/* 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" #include "JSON_CAJUN/reader.h" #include "JSON_CAJUN/writer.h" #include "JSON_CAJUN/elements.h" int facebook_json_parser::parse_buddy_list(void* data, List::List< facebook_user >* buddy_list) { using namespace json; try { facebook_user* current = NULL; std::string buddyData = static_cast< std::string* >(data)->substr(9); std::istringstream sDocument(buddyData); Object objDocument; Reader::Read(objDocument, sDocument); const Object& objRoot = objDocument; /* const Array& wasAvailableIDs = objRoot["payload"]["buddy_list"]["wasAvailableIDs"]; for (Array::const_iterator itWasAvailable(wasAvailableIDs.Begin()); itWasAvailable != wasAvailableIDs.End(); ++itWasAvailable) { const Number& member = *itWasAvailable; char was_id[32]; lltoa(member.Value(), was_id, 10); current = buddy_list->find(std::string(was_id)); if (current != NULL) current->status_id = ID_STATUS_OFFLINE; }*/ // Facebook removed support for "wasAvailableIDs" // Set all contacts in map to offline for (List::Item< facebook_user >* i = buddy_list->begin(); i != NULL; i = i->next) { i->data->status_id = ID_STATUS_OFFLINE; } // Load last active times const Object& lastActive = objRoot["payload"]["buddy_list"]["last_active_times"]; for (Object::const_iterator itLastActive(lastActive.Begin()); itLastActive != lastActive.End(); ++itLastActive) { const Object::Member& member = *itLastActive; current = buddy_list->find(member.name); if (current == NULL) { buddy_list->insert(std::make_pair(member.name, new facebook_user())); current = buddy_list->find(member.name); current->user_id = member.name; } const Number& timestamp = member.element; current->last_active = timestamp.Value(); } // Find mobile friends const Array& mobileFriends = objRoot["payload"]["buddy_list"]["mobile_friends"]; for (Array::const_iterator buddy(mobileFriends.Begin()); buddy != mobileFriends.End(); ++buddy) { const Number& member = *buddy; char was_id[32]; lltoa(member.Value(), was_id, 10); std::string id = was_id; if (!id.empty()) { current = buddy_list->find(id); if (current == NULL) { buddy_list->insert(std::make_pair(id, new facebook_user())); current = buddy_list->find(id); current->user_id = id; } current->status_id = ID_STATUS_ONTHEPHONE; } } // Find now awailable contacts const Object& nowAvailableList = objRoot["payload"]["buddy_list"]["nowAvailableList"]; for (Object::const_iterator itAvailable(nowAvailableList.Begin()); itAvailable != nowAvailableList.End(); ++itAvailable) { const Object::Member& member = *itAvailable; const Object& objMember = member.element; const Boolean idle = objMember["i"]; // In new version of Facebook "i" means "offline" current = buddy_list->find(member.name); if (current == NULL) { if (idle) continue; // Just little optimalization buddy_list->insert(std::make_pair(member.name, new facebook_user())); current = buddy_list->find(member.name); current->user_id = current->real_name = member.name; } current->status_id = (idle ? ID_STATUS_OFFLINE : ID_STATUS_ONLINE); } // Get aditional informations about contacts (if available) const Object& userInfosList = objRoot["payload"]["buddy_list"]["userInfos"]; for (Object::const_iterator itUserInfo(userInfosList.Begin()); itUserInfo != userInfosList.End(); ++itUserInfo) { const Object::Member& member = *itUserInfo; current = buddy_list->find(member.name); if (current == NULL) continue; const Object& objMember = member.element; const String& realName = objMember["name"]; const String& imageUrl = objMember["thumbSrc"]; current->real_name = utils::text::slashu_to_utf8( utils::text::special_expressions_decode(realName.Value())); current->image_url = utils::text::slashu_to_utf8( utils::text::special_expressions_decode(imageUrl.Value())); } } catch (Reader::ParseException& e) { proto->Log("!!!!! Caught json::ParseException: %s", e.what()); proto->Log(" Line/offset: %d/%d", e.m_locTokenBegin.m_nLine + 1, e.m_locTokenBegin.m_nLineOffset + 1); } catch (const Exception& e) { proto->Log("!!!!! Caught json::Exception: %s", e.what()); } catch (const std::exception& e) { proto->Log("!!!!! Caught std::exception: %s", e.what()); } return EXIT_SUCCESS; } int facebook_json_parser::parse_friends(void* data, std::map< std::string, facebook_user* >* friends) { using namespace json; try { std::string buddyData = static_cast< std::string* >(data)->substr(9); std::istringstream sDocument(buddyData); Object objDocument; Reader::Read(objDocument, sDocument); const Object& objRoot = objDocument; const Object& payload = objRoot["payload"]; for (Object::const_iterator payload_item(payload.Begin()); payload_item != payload.End(); ++payload_item) { const Object::Member& member = *payload_item; const Object& objMember = member.element; const String& realName = objMember["name"]; const String& imageUrl = objMember["thumbSrc"]; //const String& vanity = objMember["vanity"]; //const String& uri = objMember["uri"]; const Number& gender = objMember["gender"]; //const Boolean& isFriend = objMember["is_friend"]; //const String& type = objMember["type"]; // "friend" = existing classic contact, "user" = contact with disabled/deleted account facebook_user *fbu = new facebook_user(); fbu->user_id = member.name; fbu->real_name = utils::text::slashu_to_utf8( utils::text::special_expressions_decode(realName.Value())); fbu->image_url = utils::text::slashu_to_utf8( utils::text::special_expressions_decode(imageUrl.Value())); if (gender.Value() == 1) { fbu->gender = 70; // female } else if (gender.Value() == 2) { fbu->gender = 77; // male } friends->insert(std::make_pair(member.name, fbu)); } } catch (Reader::ParseException& e) { proto->Log("!!!!! Caught json::ParseException: %s", e.what()); proto->Log(" Line/offset: %d/%d", e.m_locTokenBegin.m_nLine + 1, e.m_locTokenBegin.m_nLineOffset + 1); } catch (const Exception& e) { proto->Log("!!!!! Caught json::Exception: %s", e.what()); } catch (const std::exception& e) { proto->Log("!!!!! Caught std::exception: %s", e.what()); } return EXIT_SUCCESS; } int facebook_json_parser::parse_notifications(void *data, std::vector< facebook_notification* > *notifications) { using namespace json; try { std::string notificationsData = static_cast< std::string* >(data)->substr(9); std::istringstream sDocument(notificationsData); Object objDocument; Reader::Read(objDocument, sDocument); const Object& objRoot = objDocument; const Object& payload = objRoot["payload"]["notifications"]; for (Object::const_iterator payload_item(payload.Begin()); payload_item != payload.End(); ++payload_item) { const Object::Member& member = *payload_item; const Object& objMember = member.element; const String& content = objMember["markup"]; const Number& unread = objMember["unread"]; if (unread.Value() == 0) // ignore old notifications continue; std::string text = utils::text::slashu_to_utf8( utils::text::special_expressions_decode(content.Value())); facebook_notification* notification = new facebook_notification(); notification->text = utils::text::remove_html(utils::text::source_get_value(&text, 1, "link = utils::text::source_get_value(&text, 3, "id = member.name; notifications->push_back(notification); } } catch (Reader::ParseException& e) { proto->Log("!!!!! Caught json::ParseException: %s", e.what()); proto->Log(" Line/offset: %d/%d", e.m_locTokenBegin.m_nLine + 1, e.m_locTokenBegin.m_nLineOffset + 1); } catch (const Exception& e) { proto->Log("!!!!! Caught json::Exception: %s", e.what()); } catch (const std::exception& e) { proto->Log("!!!!! Caught std::exception: %s", e.what()); } return EXIT_SUCCESS; } bool ignore_duplicits(FacebookProto *proto, std::string mid, std::string text) { std::map::iterator it = proto->facy.messages_ignore.find(mid); if (it != proto->facy.messages_ignore.end()) { std::string msg = "????? Ignoring duplicit/sent message\n" + text; proto->Log(msg.c_str()); it->second = true; // mark to delete it at the end return true; } // remember this id to ignore duplicits proto->facy.messages_ignore.insert(std::make_pair(mid, true)); return false; } int facebook_json_parser::parse_messages(void* data, std::vector< facebook_message* >* messages, std::vector< facebook_notification* >* notifications) { using namespace json; try { std::string messageData = static_cast< std::string* >(data)->substr(9); std::istringstream sDocument(messageData); Object objDocument; Reader::Read(objDocument, sDocument); const Object& objRoot = objDocument; const Array& messagesArray = objRoot["ms"]; for (Array::const_iterator itMessage(messagesArray.Begin()); itMessage != messagesArray.End(); ++itMessage) { const Object& objMember = *itMessage; const String& type = objMember["type"]; if (type.Value() == "msg" || type.Value() == "offline_msg") // direct message { const Number& from = objMember["from"]; char was_id[32]; lltoa(from.Value(), was_id, 10); const Object& messageContent = objMember["msg"]; const String& text = messageContent["text"]; const String& message_id = messageContent["messageId"]; //"tab_type":"friend", objMember["tab_type"] const Number& time_sent = messageContent["time"]; // ignore duplicits or messages sent from miranda if (ignore_duplicits(proto, message_id.Value(), text.Value())) continue; std::string message_text = utils::text::trim(utils::text::special_expressions_decode(utils::text::slashu_to_utf8(text.Value())), true); if (message_text.empty()) continue; if (was_id == proto->facy.self_.user_id) { const Number& to = objMember["to"]; char to_id[32]; lltoa(to.Value(), to_id, 10); HANDLE hContact = proto->ContactIDToHContact(to_id); if (!hContact) continue; DBEVENTINFO dbei = {0}; dbei.cbSize = sizeof(dbei); dbei.eventType = EVENTTYPE_MESSAGE; dbei.flags = DBEF_SENT | DBEF_UTF; dbei.szModule = proto->m_szModuleName; bool local_time = proto->getByte(FACEBOOK_KEY_LOCAL_TIMESTAMP, 0) != 0; dbei.timestamp = local_time ? ::time(NULL) : utils::time::fix_timestamp(time_sent.Value()); dbei.cbBlob = (DWORD)message_text.length() + 1; dbei.pBlob = (PBYTE)message_text.c_str(); db_event_add(hContact, &dbei); continue; } if ((messageContent.Find("truncated") != messageContent.End()) && (((const Number &)messageContent["truncated"]).Value() == 1)) { // If we got truncated message, we can ignore it, because we should get it again as "messaging" type std::string msg = "????? We got truncated message so we ignore it\n"; msg += utils::text::special_expressions_decode(utils::text::slashu_to_utf8(text.Value())); proto->Log(msg.c_str()); } else { facebook_message* message = new facebook_message(); message->message_text = message_text; message->time = utils::time::fix_timestamp(time_sent.Value()); message->user_id = was_id; message->message_id = message_id; messages->push_back(message); } } else if (type.Value() == "messaging") { const String& type = objMember["event"]; if (type.Value() == "read_receipt") { // user read message const Number& reader = objMember["reader"]; const Number& time = objMember["time"]; char user_id[32]; lltoa(reader.Value(), user_id, 10); // TODO: add check for chat contacts HANDLE hContact = proto->ContactIDToHContact(user_id); if (hContact) { TCHAR ttime[64], tstr[100]; _tcsftime(ttime, SIZEOF(ttime), _T("%X"), utils::conversion::fbtime_to_timeinfo(time.Value())); mir_sntprintf(tstr, SIZEOF(tstr), TranslateT("Message read: %s"), ttime); CallService(MS_MSG_SETSTATUSTEXT, (WPARAM)hContact, (LPARAM)tstr); } } else if (type.Value() == "deliver") { // inbox message (multiuser or direct) const Object& messageContent = objMember["message"]; const Number& sender_fbid = messageContent["sender_fbid"]; const String& sender_name = messageContent["sender_name"]; const String& text = messageContent["body"]; const String& tid = messageContent["tid"]; const String& mid = messageContent["mid"]; const Number& time_sent = messageContent["timestamp"]; char was_id[32]; lltoa(sender_fbid.Value(), was_id, 10); // Ignore messages from myself if (was_id == proto->facy.self_.user_id) continue; // Ignore duplicits or messages sent from miranda if (ignore_duplicits(proto, mid.Value(), text.Value())) continue; std::string message_text = utils::text::trim(utils::text::special_expressions_decode(utils::text::slashu_to_utf8(text.Value())), true); if (message_text.empty()) continue; facebook_message* message = new facebook_message(); message->message_text = message_text; message->sender_name = utils::text::special_expressions_decode(utils::text::slashu_to_utf8(sender_name.Value())); message->time = utils::time::fix_timestamp(time_sent.Value()); message->user_id = was_id; // TODO: Check if we have contact with this ID in friendlist and otherwise do something different? message->message_id = mid.Value(); messages->push_back(message); } } else if (type.Value() == "thread_msg") // multiuser message { const String& from_name = objMember["from_name"]; const String& to_name = objMember["to_name"]["__html"]; const String& to_id = objMember["to"]; const Number& from = objMember["from"]; char was_id[32]; lltoa(from.Value(), was_id, 10); const Object& messageContent = objMember["msg"]; const String& text = messageContent["text"]; const String& mid = messageContent["messageId"]; // Ignore messages from myself if (was_id == proto->facy.self_.user_id) continue; // Ignore duplicits or messages sent from miranda if (ignore_duplicits(proto, mid.Value(), text.Value())) continue; std::string message_text = utils::text::trim(utils::text::special_expressions_decode(utils::text::slashu_to_utf8(text.Value())), true); if (message_text.empty()) continue; std::string title = utils::text::special_expressions_decode(utils::text::slashu_to_utf8(to_name.Value())); std::string url = "/?action=read&sk=inbox&page&query&tid=" + to_id.Value(); std::string popup_text = utils::text::special_expressions_decode(utils::text::slashu_to_utf8(from_name.Value())); popup_text += ": " + message_text; proto->Log(" Got multichat message"); TCHAR* szTitle = mir_utf8decodeT(title.c_str()); TCHAR* szText = mir_utf8decodeT(popup_text.c_str()); proto->NotifyEvent(szTitle, szText, NULL, FACEBOOK_EVENT_OTHER, &url); mir_free(szTitle); mir_free(szText); } else if (type.Value() == "notification_json") // event notification { if (!proto->getByte(FACEBOOK_KEY_EVENT_NOTIFICATIONS_ENABLE, DEFAULT_EVENT_NOTIFICATIONS_ENABLE)) continue; const Array& notificationsArray = objMember["nodes"]; for (Array::const_iterator itNotification(notificationsArray.Begin()); itNotification != notificationsArray.End(); ++itNotification) { const Object& objNotification = *itNotification; //const String& text = objNotification["title"]["text"]; const String& text = objNotification["unaggregatedTitle"]["text"]; const String& link = objNotification["url"]; const String& id = objNotification["alert_id"]; const Number& time_sent = objNotification["timestamp"]["time"]; if (time_sent.Value() > proto->facy.last_notification_time_) // Check against duplicit notifications { proto->facy.last_notification_time_ = time_sent.Value(); facebook_notification* notification = new facebook_notification(); notification->text = utils::text::slashu_to_utf8(text.Value()); notification->link = utils::text::special_expressions_decode(link.Value()); std::string::size_type pos = id.Value().find(":"); notification->id = (pos != std::string::npos) ? id.Value().substr(pos+1) : id.Value(); notifications->push_back(notification); } } } else if (type.Value() == "typ") // chat typing notification { const Number& from = objMember["from"]; char user_id[32]; lltoa(from.Value(), user_id, 10); facebook_user fbu; fbu.user_id = user_id; HANDLE hContact = proto->AddToContactList(&fbu, CONTACT_FRIEND); if (proto->getWord(hContact, "Status", 0) == ID_STATUS_OFFLINE) proto->setWord(hContact, "Status", ID_STATUS_ONLINE); const Number& state = objMember["st"]; if (state.Value() == 1) CallService(MS_PROTO_CONTACTISTYPING, (WPARAM)hContact, (LPARAM)60); else CallService(MS_PROTO_CONTACTISTYPING, (WPARAM)hContact, (LPARAM)PROTOTYPE_CONTACTTYPING_OFF); } else if (type.Value() == "privacy_changed") { const String& event_type = objMember["event"]; const Object& event_data = objMember["data"]; if (event_type.Value() == "visibility_update") { // change of chat status const Boolean visibility = event_data["visibility"]; proto->Log(" Requested chat switch to %s", visibility ? "Online" : "Offline"); proto->SetStatus(visibility ? ID_STATUS_ONLINE : ID_STATUS_INVISIBLE); } } else continue; } // remove received messages from map for (std::map::iterator it = proto->facy.messages_ignore.begin(); it != proto->facy.messages_ignore.end(); ++it) { if (it->second) proto->facy.messages_ignore.erase(it); } } catch (Reader::ParseException& e) { proto->Log("!!!!! Caught json::ParseException: %s", e.what()); proto->Log(" Line/offset: %d/%d", e.m_locTokenBegin.m_nLine + 1, e.m_locTokenBegin.m_nLineOffset + 1); } catch (const Exception& e) { proto->Log ("!!!!! Caught json::Exception: %s", e.what()); } return EXIT_SUCCESS; }