/*
Facebook plugin for Miranda Instant Messenger
_____________________________________________
Copyright � 2009-11 Michal Zelinka, 2011-16 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 "stdafx.h"
int facebook_json_parser::parse_buddy_list(std::string *data, List::List< facebook_user >* buddy_list)
{
facebook_user* current = NULL;
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &list = root["payload"].at("buddy_list");
if (!list)
return EXIT_FAILURE;
// Set all contacts in map to offline (and reset client)
for (List::Item< facebook_user >* i = buddy_list->begin(); i != NULL; i = i->next) {
i->data->status_id = ID_STATUS_OFFLINE;
if (i->data->client == 0)
i->data->client = CLIENT_WEB;
}
// Load last active times
const JSONNode &lastActive = list["last_active_times"];
for (auto it = lastActive.begin(); it != lastActive.end(); ++it) {
const char *id = (*it).name();
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->last_active = (*it).as_int();
}
// Find mobile friends
const JSONNode &mobileFriends = list["mobile_friends"];
for (auto it = mobileFriends.begin(); it != mobileFriends.end(); ++it) {
std::string id = (*it).as_string();
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_OFFLINE;
current->client = CLIENT_MOBILE;
}
time_t now = ::time(NULL);
// Find now available contacts
const JSONNode &nowAvailable = list["nowAvailableList"];
for (auto it = nowAvailable.begin(); it != nowAvailable.end(); ++it) {
const char *id = (*it).name();
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 = (current->client == CLIENT_MOBILE || current->client == CLIENT_MESSENGER) ? ID_STATUS_ONTHEPHONE : ID_STATUS_ONLINE;
// Set contacts that were last active more than 1 minute ago as away
if (current->status_id == ID_STATUS_ONLINE && current->last_active > 0 && (now - current->last_active) > 60) {
current->status_id = ID_STATUS_AWAY;
}
}
// Get aditional informations about contacts (if available)
const JSONNode &userInfos = list["userInfos"];
for (auto it = userInfos.begin(); it != userInfos.end(); ++it) {
const char *id = (*it).name();
current = buddy_list->find(id);
if (current == NULL)
continue;
std::string name = (*it)["name"].as_string();
if (!name.empty())
current->real_name = utils::text::slashu_to_utf8(name);
std::string thumbSrc = (*it)["thumbSrc"].as_string();
if (!thumbSrc.empty())
current->image_url = utils::text::slashu_to_utf8(thumbSrc);
}
return EXIT_SUCCESS;
}
void parseUser(const JSONNode &it, facebook_user *fbu)
{
fbu->user_id = it.name();
std::string id = it["id"].as_string();
if (id.empty() || id == "0") {
// this user has deleted account or is just unavailable for us (e.g., ignore list) -> don't read dummy name and avatar and rather ignore that completely
return;
}
std::string alternateName = it["alternateName"].as_string(); // nickname
std::string name = it["name"].as_string();
std::string thumbSrc = it["thumbSrc"].as_string();
std::string vanity = it["vanity"].as_string(); // username
int gender = it["gender"].as_int();
//const JSONNode &uri = it["uri"); // profile url
//const JSONNode &is_friend = it["is_friend"); // e.g. "True"
//const JSONNode &type = it["type"); // e.g. "friend" (classic contact) or "user" (disabled/deleted account)
if (!name.empty())
fbu->real_name = utils::text::slashu_to_utf8(name);
if (!thumbSrc.empty())
fbu->image_url = utils::text::slashu_to_utf8(thumbSrc);
if (!vanity.empty())
fbu->username = utils::text::slashu_to_utf8(vanity);
if (!alternateName.empty())
fbu->nick = alternateName;
if (gender) {
switch (gender) {
case 1: // female
fbu->gender = 70;
break;
case 2: // male
fbu->gender = 77;
break;
// case 7: not available female?
}
}
}
int facebook_json_parser::parse_friends(std::string *data, std::map< std::string, facebook_user* >* friends)
{
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &payload = root["payload"];
if (!payload)
return EXIT_FAILURE;
for (auto it = payload.begin(); it != payload.end(); ++it) {
facebook_user *fbu = new facebook_user();
parseUser(*it, fbu);
friends->insert(std::make_pair(fbu->user_id, fbu));
}
return EXIT_SUCCESS;
}
int facebook_json_parser::parse_notifications(std::string *data, std::map< std::string, facebook_notification* > *notifications)
{
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &list = root["payload"].at("nodes");
if (!list)
return EXIT_FAILURE;
// Create notifications chatroom (if it doesn't exists), because we will be writing to it new notifications here
proto->PrepareNotificationsChatRoom();
for (auto it = list.begin(); it != list.end(); ++it) {
const JSONNode &id_ = (*it)["alert_id"];
const JSONNode &state_ = (*it)["seen_state"];
const JSONNode &time_ = (*it)["timestamp"]["time"];
const JSONNode &text_ = (*it)["title"]["text"];
const JSONNode &url_ = (*it)["url"];
// Ignore empty and old notifications
if (!text_ || !state_ || state_.as_string() == "SEEN_AND_READ" || !time_)
continue;
facebook_notification* notification = new facebook_notification();
notification->id = id_.as_string();
// Fix notification ID
std::string::size_type pos = notification->id.find(":");
if (pos != std::string::npos)
notification->id = notification->id.substr(pos + 1);
notification->link = url_.as_string();
notification->text = utils::text::html_entities_decode(utils::text::slashu_to_utf8(text_.as_string()));
notification->time = utils::time::from_string(time_.as_string());
// Write notification to chatroom
proto->UpdateNotificationsChatRoom(notification);
// If it's unseen, remember it, otherwise forget it
if (notifications->find(notification->id) == notifications->end())
notifications->insert(std::make_pair(notification->id, notification));
else
delete notification;
}
return EXIT_SUCCESS;
}
bool ignore_duplicits(FacebookProto *proto, const std::string &mid, const std::string &)
{
ScopedLock s(proto->facy.send_message_lock_);
std::map::iterator it = proto->facy.messages_ignore.find(mid);
if (it != proto->facy.messages_ignore.end()) {
proto->debugLogA("??? Ignoring duplicit/sent message ID: %s", mid.c_str());
it->second++; // increase counter (for deleting it later)
return true;
}
// remember this id to ignore duplicits
proto->facy.messages_ignore.insert(std::make_pair(mid, 1));
return false;
}
std::string absolutizeUrl(std::string &url) {
if (url.find("/") == 0) {
url = HTTP_PROTO_SECURE FACEBOOK_SERVER_REGULAR + url;
}
return url;
}
void parseAttachments(FacebookProto *proto, std::string *message_text, const JSONNode &it, const std::string &thread_id, std::string other_user_fbid)
{
// Process attachements and stickers
const JSONNode &has_attachment = it["has_attachment"];
if (!has_attachment || !has_attachment.as_bool())
return;
// Append attachements
std::string type;
std::string attachments_text;
const JSONNode &attachments = it["attachments"];
for (auto itAttachment = attachments.begin(); itAttachment != attachments.end(); ++itAttachment) {
const JSONNode &attach_type = (*itAttachment)["attach_type"]; // "sticker", "photo", "file", "share"
if (attach_type) {
// Get attachment type - "file" has priority over other types
if (type.empty() || type != "file")
type = attach_type.as_string();
}
if (type == "photo") {
std::string filename = (*itAttachment)["name"].as_string();
std::string link = (*itAttachment)["hires_url"].as_string();
const JSONNode &metadata = (*itAttachment)["metadata"];
if (metadata) {
std::string id = metadata["fbid"].as_string();
const JSONNode &data = it["attachment_map"][id.c_str()];
filename = data["filename"].as_string();
link = data["image_data"]["url"].as_string();
}
if (!link.empty()) {
attachments_text += "\n" + (!filename.empty() ? "<" + filename + "> " : "") + absolutizeUrl(link) + "\n";
}
}
else if (type == "file") {
std::string filename = (*itAttachment)["name"].as_string();
std::string link = (*itAttachment)["url"].as_string();
if (!link.empty()) {
attachments_text += "\n" + (!filename.empty() ? "<" + filename + "> " : "") + absolutizeUrl(link) + "\n";
}
}
else if (type == "share") {
const JSONNode &share = (*itAttachment)["share"];
if (share) {
std::string title = share["title"].as_string();
std::string description = share["description"].as_string();
std::string link = share["uri"].as_string();
if (link.find("l." FACEBOOK_SERVER_DOMAIN) != std::string::npos) {
// de-facebook this link
link = utils::url::decode(utils::text::source_get_value2(&link, "l.php?u=", "&", true));
}
if (!link.empty()) {
attachments_text += "\n";
if (!title.empty())
attachments_text += title + "\n";
if (!description.empty())
attachments_text += description + "\n";
attachments_text += absolutizeUrl(link) + "\n";
}
}
}
else if (type == "sticker") {
std::string link = (*itAttachment)["url"].as_string();
if (!link.empty()) {
attachments_text += "\n" + absolutizeUrl(link) + "\n";
}
const JSONNode &metadata = (*itAttachment)["metadata"];
if (metadata) {
const JSONNode &stickerId_ = metadata["stickerID"];
if (stickerId_) {
std::string sticker = "[[sticker:" + stickerId_.as_string() + "]]\n";
attachments_text += sticker;
if (other_user_fbid.empty() && !thread_id.empty())
other_user_fbid = proto->ThreadIDToContactID(thread_id);
// Stickers as smileys
if (proto->getByte(FACEBOOK_KEY_CUSTOM_SMILEYS, DEFAULT_CUSTOM_SMILEYS)) {
// FIXME: rewrite smileyadd to use custom smileys per protocol and not per contact and then remove this ugliness
if (!other_user_fbid.empty()) {
MCONTACT hContact = proto->ContactIDToHContact(other_user_fbid);
proto->StickerAsSmiley(sticker, link, hContact);
}
}
}
}
}
}
// TODO: have this as extra event, not replace or append message content
if (!message_text->empty())
*message_text += "\n\n";
if (!attachments_text.empty()) {
// we can't use this as offline messages doesn't have it
/* const JSONNode &admin_snippet = it["admin_snippet");
if (admin_snippet != NULL) {
*message_text += admin_snippet);
} */
std::tstring newText;
if (type == "sticker")
newText = TranslateT("a sticker");
else if (type == "share")
newText = TranslateT("a link");
else if (type == "file")
newText = (attachments.size() > 1) ? TranslateT("files") : TranslateT("a file");
else if (type == "photo")
newText = (attachments.size() > 1) ? TranslateT("photos") : TranslateT("a photo");
else
newText = _A2T(type.c_str());
TCHAR title[200];
mir_sntprintf(title, TranslateT("User sent %s:"), newText.c_str());
*message_text += T2Utf(title);
*message_text += attachments_text;
}
else {
*message_text += T2Utf(TranslateT("User sent an unsupported attachment. Open your browser to see it."));
}
}
void parseAttachments2(FacebookProto *proto, std::string *message_text, const JSONNode &it, std::string other_user_fbid)
{
std::string attachments_text;
std::string type;
const JSONNode &attach_ = it["attachments"]["mercury"];
if (!attach_)
return;
/* const JSONNode &attachments = it["attachments"];
for (auto itAttachment = attachments.begin(); itAttachment != attachments.end(); ++itAttachment) {
const JSONNode &attach_ = (*itAttachment)["mercury"];*/
type = attach_["attach_type"].as_string(); // "sticker", "photo", "file", "share"
if (type == "photo") {
std::string filename = attach_["name"].as_string();
std::string link = attach_["hires_url"].as_string();
const JSONNode &metadata = attach_["metadata"];
if (metadata) {
std::string id = metadata["fbid"].as_string();
const JSONNode &data = it["attachment_map"][id.c_str()];
filename = data["filename"].as_string();
link = data["image_data"]["url"].as_string();
}
if (!link.empty()) {
attachments_text += "\n" + (!filename.empty() ? "<" + filename + "> " : "") + absolutizeUrl(link) + "\n";
}
}
else if (type == "file") {
std::string filename = attach_["name"].as_string();
std::string link = attach_["url"].as_string();
if (!link.empty()) {
attachments_text += "\n" + (!filename.empty() ? "<" + filename + "> " : "") + absolutizeUrl(link) + "\n";
}
}
else if (type == "share") {
const JSONNode &share = attach_["share"];
if (share) {
std::string title = share["title"].as_string();
std::string description = share["description"].as_string();
std::string link = share["uri"].as_string();
if (link.find("l." FACEBOOK_SERVER_DOMAIN) != std::string::npos) {
// de-facebook this link
link = utils::url::decode(utils::text::source_get_value2(&link, "l.php?u=", "&", true));
}
if (!link.empty()) {
attachments_text += "\n";
if (!title.empty())
attachments_text += title + "\n";
if (!description.empty())
attachments_text += description + "\n";
attachments_text += absolutizeUrl(link) + "\n";
}
}
}
else if (type == "sticker") {
std::string link = attach_["url"].as_string();
if (!link.empty()) {
attachments_text += "\n" + absolutizeUrl(link) + "\n";
}
const JSONNode &metadata = attach_["metadata"];
if (metadata) {
const JSONNode &stickerId_ = metadata["stickerID"];
if (stickerId_) {
std::string sticker = "[[sticker:" + stickerId_.as_string() + "]]\n";
attachments_text += sticker;
// Stickers as smileys
if (proto->getByte(FACEBOOK_KEY_CUSTOM_SMILEYS, DEFAULT_CUSTOM_SMILEYS)) {
// FIXME: rewrite smileyadd to use custom smileys per protocol and not per contact and then remove this ugliness
if (!other_user_fbid.empty()) {
MCONTACT hContact = proto->ContactIDToHContact(other_user_fbid);
proto->StickerAsSmiley(sticker, link, hContact);
}
}
}
}
}
else {
proto->debugLogA("json::parseAttachments2 - Unknown attachment type '%s'", type.c_str());
}
// }
// TODO: have this as extra event, not replace or append message content
if (!message_text->empty())
*message_text += "\n\n";
if (!attachments_text.empty()) {
// we can't use this as offline messages doesn't have it
/* const JSONNode &admin_snippet = it["admin_snippet");
if (admin_snippet != NULL) {
*message_text += admin_snippet);
} */
std::tstring newText;
if (type == "sticker")
newText = TranslateT("a sticker");
else if (type == "share")
newText = TranslateT("a link");
else if (type == "file")
// newText = (attachments.size() > 1) ? TranslateT("files") : TranslateT("a file");
newText = TranslateT("a file");
else if (type == "photo")
// newText = (attachments.size() > 1) ? TranslateT("photos") : TranslateT("a photo");
newText = TranslateT("a photo");
else
newText = _A2T(type.c_str());
TCHAR title[200];
mir_sntprintf(title, TranslateT("User sent %s:"), newText.c_str());
*message_text += T2Utf(title);
*message_text += attachments_text;
}
else {
*message_text += T2Utf(TranslateT("User sent an unsupported attachment. Open your browser to see it."));
}
}
int facebook_json_parser::parse_messages(std::string *pData, std::vector< facebook_message* >* messages, std::map< std::string, facebook_notification* >* notifications, bool inboxOnly)
{
// remove old received messages from map
for (std::map::iterator it = proto->facy.messages_ignore.begin(); it != proto->facy.messages_ignore.end();) {
if (it->second > FACEBOOK_IGNORE_COUNTER_LIMIT) {
it = proto->facy.messages_ignore.erase(it);
}
else {
it->second++; // increase counter on each request
++it;
}
}
JSONNode root = JSONNode::parse(pData->substr(9).c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &ms = root["ms"];
if (!ms)
return EXIT_FAILURE;
for (auto it = ms.begin(); it != ms.end(); ++it) {
const JSONNode &type = (*it)["type"];
if (!type)
continue;
std::string t = type.as_string();
if (t == "delta") {
// new messaging stuff
const JSONNode &delta_ = (*it)["delta"];
if (!delta_)
continue;
const JSONNode &cls_ = delta_["class"];
std::string cls = cls_.as_string();
if (cls == "NewMessage") {
// TODO: Support for "folders" was probably removed, we should remove the "inboxOnly" option too - but first check how does that "allow messages request" works
const JSONNode &meta_ = delta_["messageMetadata"];
if (!meta_) {
proto->debugLogA("json::parse_messages - No messageMetadata element");
continue;
}
const JSONNode &sender_fbid = meta_["actorFbId"]; // who send the message
const JSONNode &body = delta_["body"];
const JSONNode &mid = meta_["messageId"];
const JSONNode ×tamp = meta_["timestamp"];
if (!sender_fbid || !body || !mid || !timestamp)
continue;
std::string id = sender_fbid.as_string();
std::string message_id = mid.as_string();
std::string message_text = body.as_string();
std::string other_user_id = meta_["threadKey"]["otherUserFbId"].as_string(); // for whom the message is (probably only for single conversations)
std::string tid = meta_["threadKey"]["threadFbId"].as_string(); // this is probably only for chats
// Process attachements and stickers
parseAttachments2(proto, &message_text, delta_, other_user_id); // FIXME: Rework and fix parsing attachments
// Ignore duplicits or messages sent from miranda
if (ignore_duplicits(proto, message_id, message_text))
continue;
message_text = utils::text::trim(utils::text::slashu_to_utf8(message_text), true);
if (message_text.empty()) {
proto->debugLogA("json::parse_messages - Received empty message. Received only some attachment?");
// continue;
}
facebook_message* message = new facebook_message();
message->isIncoming = (id != proto->facy.self_.user_id);
message->isUnread = message->isIncoming;
message->message_text = message_text;
message->time = utils::time::from_string(timestamp.as_string());
message->user_id = other_user_id;
message->message_id = message_id;
message->thread_id = tid;
message->isChat = other_user_id.empty(); // FIXME: Determine whether this is chat or not
messages->push_back(message);
}
else if (cls == "ReadReceipt") {
// user read message
// when read
const JSONNode &time_ = delta_["actionTimestampMs"];
if (!time_)
continue;
time_t timestamp = utils::time::from_string(time_.as_string());
// for multi chats (not available for single)
const JSONNode &actor_ = delta_["actorFbId"]; // who read the message
const JSONNode &thread_ = delta_["threadKey"]["threadFbId"]; // chat thread
// for single chats (not available for multi)
const JSONNode &reader_ = delta_["threadKey"]["otherUserFbId"]; // who read the message
if (actor_ && thread_) {
// multi chat
// ignore if disabled
if (!proto->m_enableChat)
continue;
std::string readerId = actor_.as_string();
std::string tid = "id." + thread_.as_string(); // NOTE: threadFbId means just numeric id of thread, without "id." prefix. We add it here to have it consistent with other methods (where threadId is used)
MCONTACT hContact = proto->ChatIDToHContact(tid);
proto->facy.insert_reader(hContact, timestamp, readerId);
}
else if (reader_) {
// single chat
std::string userId = reader_.as_string();
MCONTACT hContact = proto->ContactIDToHContact(userId);
proto->facy.insert_reader(hContact, timestamp);
}
}
else if (cls == "NoOp") {
// contains numNoOps=1 (or probably other number) in delta element, but I don't know what is it for
continue;
}
else {
proto->debugLogA("json::parse_messages - Unknown class '%s'", cls.c_str());
}
}
else if (t == "messaging") {
// various messaging stuff (received and sent messages, getting seen info)
const JSONNode &ev_ = (*it)["event"];
if (!ev_)
continue;
std::string ev = ev_.as_string();
if (ev == "deliver") {
// inbox message (multiuser or direct)
const JSONNode &msg = (*it)["message"];
const JSONNode &folder = (*it)["folder"];
if (inboxOnly && folder.as_string() != "inbox")
continue;
const JSONNode &sender_fbid = msg["sender_fbid"];
const JSONNode &sender_name = msg["sender_name"];
const JSONNode &body = msg["body"];
// looks like there is either "tid" or "other_user_fbid" (or both)
const JSONNode &tid = msg["tid"];
const JSONNode &mid = msg["mid"];
const JSONNode ×tamp = msg["timestamp"];
if (!sender_fbid || !sender_name || !body || !mid || !timestamp)
continue;
const JSONNode &is_filtered = (*it)["is_filtered_content"]; // TODO: is it still here? perhaps it is replaced with msg["is_spoof_warning"] or something else?
//const JSONNode &is_spoof_warning = msg["is_spoof_warning"];
//const JSONNode &is_silent = msg["is_silent"];
//const JSONNode &is_unread = msg["is_unread"];
std::string id = sender_fbid.as_string();
std::string message_id = mid.as_string();
std::string message_text = body.as_string();
std::string thread_id = tid.as_string();
std::string other_user_id = msg["other_user_fbid"].as_string();
// Process attachements and stickers
parseAttachments(proto, &message_text, msg, thread_id, other_user_id);
// Ignore duplicits or messages sent from miranda
if (!body || ignore_duplicits(proto, message_id, message_text))
continue;
if (is_filtered.as_bool() && message_text.empty())
message_text = Translate("This message is no longer available, because it was marked as abusive or spam.");
message_text = utils::text::trim(utils::text::slashu_to_utf8(message_text), true);
if (message_text.empty())
continue;
facebook_message* message = new facebook_message();
message->isUnread = true;
message->isIncoming = (id != proto->facy.self_.user_id);
message->message_text = message_text;
message->time = utils::time::from_string(timestamp.as_string());
message->user_id = id;
message->message_id = message_id;
message->sender_name = utils::text::slashu_to_utf8(sender_name.as_string()); // TODO: or if not incomming use my own name from facy.self_ ?
message->thread_id = tid.as_string(); // TODO: or if not incomming use my own id from facy.self_ ?
const JSONNode >hreadinfo = msg["group_thread_info"];
message->isChat = (gthreadinfo && gthreadinfo.as_string() != "null");
if (!message->isChat && !message->isIncoming) {
message->sender_name.clear();
message->user_id = !other_user_id.empty() ? other_user_id : proto->ThreadIDToContactID(message->thread_id); // TODO: Check if we have contact with this user_id in friendlist and otherwise do something different?
}
messages->push_back(message);
}
}
else if (t == "notification_json") {
// event notification
const JSONNode &nodes = (*it)["nodes"];
// Create notifications chatroom (if it doesn't exists), because we will be writing to it new notifications here
proto->PrepareNotificationsChatRoom();
for (auto itNodes = nodes.begin(); itNodes != nodes.end(); ++itNodes) {
const JSONNode &text_ = (*itNodes)["unaggregatedTitle"]; // notifications one by one, not grouped
if (!text_)
continue;
const JSONNode &text = text_["text"];
const JSONNode &url = (*itNodes)["url"];
const JSONNode &alert_id = (*itNodes)["alert_id"];
const JSONNode &time_ = (*itNodes)["timestamp"];
if (!time_)
continue;
const JSONNode &time = time_["time"];
if (!time || !text || !url || !alert_id)
continue;
time_t timestamp = utils::time::from_string(time.as_string());
if (timestamp > proto->facy.last_notification_time_) {
// Only new notifications
proto->facy.last_notification_time_ = timestamp;
facebook_notification* notification = new facebook_notification();
notification->text = utils::text::slashu_to_utf8(text.as_string());
notification->link = url.as_string();
notification->id = alert_id.as_string();
notification->time = timestamp;
// Fix notification ID
std::string::size_type pos = notification->id.find(":");
if (pos != std::string::npos)
notification->id = notification->id.substr(pos + 1);
// Write notification to chatroom
proto->UpdateNotificationsChatRoom(notification);
// If it's unseen, remember it, otherwise forget it (here it will always be unseen)
if (notifications->find(notification->id) == notifications->end() && !notification->seen)
notifications->insert(std::make_pair(notification->id, notification));
else
delete notification;
}
}
}
else if (t == "m_notification") {
const JSONNode &data = (*it)["data"];
if (!data)
continue;
const JSONNode &appId_ = data["app_id"];
const JSONNode &type_ = data["type"];
if (appId_.as_string() == "2356318349" || type_.as_string() == "friend_confirmed") {
// Friendship notifications
const JSONNode &body_ = data["body"];
const JSONNode &html_ = body_["__html"];
const JSONNode &href_ = data["href"];
const JSONNode &unread_ = data["unread"];
const JSONNode &alertId_ = data["alert_id"];
if (!html_ || !href_ || !unread_ || unread_.as_int() == 0)
continue;
std::string text = utils::text::remove_html(utils::text::slashu_to_utf8(html_.as_string()));
std::string url = href_.as_string();
std::string alert_id = alertId_.as_string();
proto->NotifyEvent(proto->m_tszUserName, ptrT(mir_utf8decodeT(text.c_str())), NULL, FACEBOOK_EVENT_FRIENDSHIP, &url, alert_id.empty() ? NULL : &alert_id);
}
}
else if (t == "jewel_requests_add") {
// New friendship request, load them all with real names (because there is only user_id in "from" field)
proto->ForkThread(&FacebookProto::ProcessFriendRequests, NULL);
}
else if (t == "typ") {
// chat typing notification
const JSONNode &from = (*it)["from"];
if (!from)
continue;
facebook_user fbu;
fbu.user_id = from.as_string();
MCONTACT hContact = proto->AddToContactList(&fbu, CONTACT_FRIEND); // only friends are able to send typing notifications
const JSONNode &st = (*it)["st"];
if (st.as_int() == 1)
proto->StartTyping(hContact);
else
proto->StopTyping(hContact);
}
else if (t == "ttyp") {
// multi chat typing notification
if (!proto->m_enableChat)
continue;
const JSONNode &from_ = (*it)["from"];
const JSONNode &thread_ = (*it)["thread"];
const JSONNode &st_ = (*it)["st"];
if (!from_ || !thread_ || !st_)
continue;
std::string tid = thread_.as_string();
std::string from_id = from_.as_string();
auto itRoom = proto->facy.chat_rooms.find(thread_.as_string());
if (itRoom != proto->facy.chat_rooms.end()) {
facebook_chatroom *chatroom = itRoom->second;
std::map participants = chatroom->participants;
auto participant = participants.find(from_id);
if (participant == participants.end()) {
// TODO: load name of this participant
std::string name = from_id;
proto->AddChatContact(tid.c_str(), from_id.c_str(), name.c_str());
}
participant = participants.find(from_id);
if (participant != participants.end()) {
MCONTACT hChatContact = proto->ChatIDToHContact(tid);
ptrT name(mir_utf8decodeT(participant->second.c_str()));
if (st_.as_int() == 1) {
StatusTextData st = { 0 };
st.cbSize = sizeof(st);
mir_sntprintf(st.tszText, TranslateT("%s is typing a message..."), name);
CallService(MS_MSG_SETSTATUSTEXT, (WPARAM)hChatContact, (LPARAM)&st);
}
else CallService(MS_MSG_SETSTATUSTEXT, (WPARAM)hChatContact);
// TODO: support proper MS_PROTO_CONTACTISTYPING service for chatrooms (when it will be implemented)
}
}
}
else if (t == "privacy_changed") {
// settings changed
const JSONNode &event_type = (*it)["event"];
const JSONNode &event_data = (*it)["data"];
if (!event_type || !event_data)
continue;
std::string et = event_type.as_string();
if (et == "visibility_update") {
// change of chat status
const JSONNode &visibility = event_data["visibility"];
bool isVisible = visibility && visibility.as_bool();
proto->debugLogA(" Requested chat switch to %s", isVisible ? "Online" : "Offline");
// If actual status is not what server says, change it (handle also local away status, which means online)
if (isVisible != (proto->m_iStatus != ID_STATUS_INVISIBLE))
proto->SetStatus(isVisible ? ID_STATUS_ONLINE : ID_STATUS_INVISIBLE);
}
}
else if (t == "buddylist_overlay") {
// additional info about user status (used client)
const JSONNode &overlay = (*it)["overlay"];
if (!overlay)
continue;
for (auto itNodes = overlay.begin(); itNodes != overlay.end(); ++itNodes) {
std::string id = (*itNodes).as_string();
const JSONNode &p_ = (*itNodes)["p"];
if (!p_)
continue;
// TODO: This is now supported also via /ajax/mercury/tabs_presence.php request
const JSONNode &p = (*it)["p"];
if (p) {
facebook_user* current = proto->facy.buddies.find(id); // HACKISH-WAY to get buddies...
if (current == NULL)
continue;
std::string status = p["status"].as_string(); // this seems to be "active" or "invisible" or null
std::string webStatus = p["webStatus"].as_string(); // "active", "idle" or "offline"
std::string fbAppStatus = p["fbAppStatus"].as_string(); // "offline" or "active" or "invisible" or null
std::string messengerStatus = p["messengerStatus"].as_string(); // "offline" or "active" or "invisible" or null
std::string otherStatus = p["otherStatus"].as_string(); // "offline" or "active" or "invisible" or null - this seems to be "active" when webStatus is "idle" or "active" only
// this may never happen
if (status != "active")
current->status_id = ID_STATUS_OFFLINE;
bool b;
// "webStatus" and "otherStatus" are marked as "WEB" on FB website
if ((b = (webStatus == "active")) || otherStatus == "active") {
current->status_id = ID_STATUS_ONLINE;
current->client = b ? CLIENT_WEB : CLIENT_OTHER;
}
// "fbAppStatus" and "messengerStatus" are marked as "MOBILE" on FB website
if ((b = (fbAppStatus == "active")) || messengerStatus == "active") {
current->status_id = ID_STATUS_ONTHEPHONE;
current->client = b ? CLIENT_APP : CLIENT_MESSENGER;
}
// this is not marked anyhow on website (yet?)
current->idle = webStatus == "idle" || otherStatus == "idle" || fbAppStatus == "idle" || messengerStatus == "idle";
// TODO: save that info to DB
}
}
} else if (t == "ticker_update:home") {
const JSONNode &actor_ = (*it)["actor"];
const JSONNode &story_ = (*it)["story_xhp"];
std::string text = story_.as_string();
text = utils::text::html_entities_decode(utils::text::slashu_to_utf8(text));
std::string url = utils::text::source_get_value(&text, 3, "\"tickerStoryLink\"", "href=\"", "\"");
std::string story_type = utils::text::source_get_value2(&text, "\"type\":\"", "\"");
std::string story_class = utils::text::source_get_value2(&text, "\"entstory_class\":\"", "\"");
text = utils::text::trim(utils::text::remove_html(text));
std::string userId = actor_.as_string();
MCONTACT hContact = proto->ContactIDToHContact(userId);
proto->debugLogA("+++ Got ticker type='%s' class='%s'", story_type.c_str(), story_class.c_str());
if (!text.empty())
proto->NotifyEvent(proto->m_tszUserName, ptrT(mir_utf8decodeT(text.c_str())), hContact, FACEBOOK_EVENT_TICKER, &url);
}
else if (t == "mercury") {
// rename multi user chat, video call, ...
const JSONNode &actions_ = (*it)["actions"];
if (!actions_)
continue;
for (unsigned int i = 0; i < actions_.size(); i++) {
const JSONNode &action_ = actions_[i];
const JSONNode &thread_id_ = action_["thread_id"];
const JSONNode &log_body_ = action_["log_message_body"];
const JSONNode &log_data_ = action_["log_message_data"];
const JSONNode &log_type_ = action_["log_message_type"];
if (!log_data_ || !log_body_ || !thread_id_ || !log_type_)
continue;
std::string thread_id = thread_id_.as_string();
std::string logType = log_type_.as_string();
std::string message_text = log_body_.as_string();
if (logType == "log:video-call") {
std::string id = action_["other_user_fbid"].as_string();
std::string message_id = action_["message_id"].as_string();
facebook_message* message = new facebook_message();
message->isChat = false;
message->isUnread = true;
message->isIncoming = (id != proto->facy.self_.user_id);
message->message_text = message_text;
message->time = utils::time::from_string(action_["timestamp"].as_string());
message->user_id = id;
message->message_id = message_id;
message->sender_name.clear();
message->thread_id = thread_id;
message->type = CALL;
messages->push_back(message);
}
else {
// TODO: check for other types, now we expect this is rename chat
if (!proto->m_enableChat)
continue;
std::string name = log_data_["name"].as_string();
// proto->RenameChat(thread_id.c_str(), name.c_str()); // this don't work, why?
proto->setStringUtf(proto->ChatIDToHContact(thread_id), FACEBOOK_KEY_NICK, name.c_str());
proto->UpdateChat(thread_id.c_str(), NULL, NULL, message_text.c_str());
}
}
}
else if (t == "notifications_read" || t == "notifications_seen") {
ScopedLock s(proto->facy.notifications_lock_);
const JSONNode &alerts = (*it)["alert_ids"];
for (auto itAlerts = alerts.begin(); itAlerts != alerts.end(); ++itAlerts) {
std::string id = (*itAlerts).as_string();
auto itAlert = notifications->find(id);
if (itAlert != notifications->end()) {
if (itAlert->second->hWndPopup != NULL)
PUDeletePopup(itAlert->second->hWndPopup); // close popup
delete itAlert->second;
notifications->erase(itAlert);
}
}
}
else
continue;
}
return EXIT_SUCCESS;
}
int facebook_json_parser::parse_unread_threads(std::string *data, std::vector< std::string >* threads, bool inboxOnly)
{
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &unread_threads = root["payload"].at("unread_thread_ids");
if (!unread_threads)
return EXIT_FAILURE;
for (auto it = unread_threads.begin(); it != unread_threads.end(); ++it) {
const JSONNode &folder = (*it)["folder"];
const JSONNode &thread_ids = (*it)["thread_ids"];
if (inboxOnly && folder.as_string() != "inbox")
continue;
for (auto jt = thread_ids.begin(); jt != thread_ids.end(); ++jt)
threads->push_back((*jt).as_string());
}
return EXIT_SUCCESS;
}
int facebook_json_parser::parse_thread_messages(std::string *data, std::vector< facebook_message* >* messages, std::map< std::string, facebook_chatroom* >* chatrooms, bool unreadOnly, bool inboxOnly)
{
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &payload = root["payload"];
if (!payload)
return EXIT_FAILURE;
const JSONNode &actions = payload["actions"];
const JSONNode &threads = payload["threads"];
if (!actions || !threads)
return EXIT_FAILURE;
const JSONNode &roger = payload["roger"];
for (auto it = roger.begin(); it != roger.end(); ++it) {
std::string id = (*it).name();
// Ignore "wrong" (duplicit) identifiers - these that doesn't begin with "id."
if (id.substr(0, 3) == "id.") {
facebook_chatroom *room = new facebook_chatroom(id);
chatrooms->insert(std::make_pair(id, room));
}
}
std::map thread_ids;
for (auto it = threads.begin(); it != threads.end(); ++it) {
const JSONNode &is_canonical_user = (*it)["is_canonical_user"];
const JSONNode &other_user_fbid = (*it)["other_user_fbid"]; // other_user_fbid is better than thread_fbid, because even multi chat has thread_fbid, but they have other_user_fbid=null
const JSONNode &thread_id = (*it)["thread_id"];
const JSONNode &name = (*it)["name"];
//const JSONNode &message_count = (*it)["message_count");
//const JSONNode &unread_count = (*it)["unread_count"); // TODO: use it to check against number of loaded messages... but how?
const JSONNode &folder = (*it)["folder"];
if (!other_user_fbid || !thread_id) {
proto->debugLogA("!!! Missing other_user_fbid/thread_id");
continue;
}
std::string id = other_user_fbid.as_string();
std::string tid = thread_id.as_string();
std::map::iterator iter = chatrooms->find(tid);
if (iter != chatrooms->end()) {
if (is_canonical_user.as_bool()) {
chatrooms->erase(iter); // this is not chatroom
}
else {
iter->second->chat_name = std::tstring(ptrT(mir_utf8decodeT(name.as_string().c_str()))); // TODO: create name from users if there is no name...
const JSONNode &participants = (*it)["participants"];
for (auto jt = participants.begin(); jt != participants.end(); ++jt) {
std::string user_id = (*jt).name();
iter->second->participants.insert(std::make_pair(user_id, user_id)); // TODO: get name somehow
}
}
}
iter = chatrooms->find(id);
if (iter != chatrooms->end())
chatrooms->erase(iter); // this is not chatroom
if (inboxOnly && folder.as_string() != "inbox")
continue;
if (id == "null")
continue;
thread_ids.insert(std::make_pair(tid, id));
}
for (auto it = actions.begin(); it != actions.end(); ++it) {
const JSONNode &author = (*it)["author"];
const JSONNode &author_email = (*it)["author_email"];
const JSONNode &other_user_fbid = (*it)["other_user_fbid"];
const JSONNode &body = (*it)["body"];
const JSONNode &tid = (*it)["thread_id"];
const JSONNode &mid = (*it)["message_id"];
const JSONNode ×tamp = (*it)["timestamp"];
const JSONNode &filtered = (*it)["is_filtered_content"];
const JSONNode &folder = (*it)["folder"];
const JSONNode &is_unread = (*it)["is_unread"];
if (!author || !body || !mid || !tid || !timestamp)
continue;
if (inboxOnly && folder.as_string() != "inbox")
continue;
std::string thread_id = tid.as_string();
std::string message_id = mid.as_string();
std::string message_text = body.as_string();
std::string author_id = author.as_string();
std::string other_user_id = other_user_fbid ? other_user_fbid.as_string() : "";
std::string::size_type pos = author_id.find(":");
if (pos != std::string::npos)
author_id = author_id.substr(pos + 1);
// Process attachements and stickers
parseAttachments(proto, &message_text, *it, thread_id, other_user_id);
if (filtered.as_bool() && message_text.empty())
message_text = Translate("This message is no longer available, because it was marked as abusive or spam.");
message_text = utils::text::trim(utils::text::slashu_to_utf8(message_text), true);
if (message_text.empty())
continue;
bool isUnread = is_unread.as_bool();
// Ignore read messages if we want only unread messages
if (unreadOnly && !isUnread)
continue;
facebook_message* message = new facebook_message();
message->message_text = message_text;
if (author_email)
message->sender_name = author_email.as_string();
message->time = utils::time::from_string(timestamp.as_string());
message->thread_id = thread_id;
message->message_id = message_id;
message->isIncoming = (author_id != proto->facy.self_.user_id);
message->isUnread = isUnread;
if (chatrooms->find(thread_id) != chatrooms->end()) {
// this is chatroom message
message->isChat = true;
message->user_id = author_id;
}
else {
// this is standard message
message->isChat = false;
auto iter = thread_ids.find(thread_id);
if (iter != thread_ids.end())
message->user_id = iter->second; // TODO: Check if we have contact with this ID in friendlist and otherwise do something different?
else if (!other_user_id.empty())
message->user_id = other_user_id;
else {
delete message;
continue;
}
}
messages->push_back(message);
}
return EXIT_SUCCESS;
}
int facebook_json_parser::parse_thread_info(std::string *data, std::string* user_id)
{
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &threads = root["payload"].at("threads");
if (!threads)
return EXIT_FAILURE;
//std::map thread_ids;
for (auto it = threads.begin(); it != threads.end(); ++it) {
const JSONNode &canonical = (*it)["canonical_fbid"];
const JSONNode &thread_id = (*it)["thread_id"];
//const JSONNode &message_count = (*it)["message_count"); // TODO: this could be useful for loading history from server
if (!canonical || !thread_id)
continue;
std::string id = canonical.as_string();
if (id == "null")
continue;
*user_id = id;
}
return EXIT_SUCCESS;
}
int facebook_json_parser::parse_user_info(std::string *data, facebook_user* fbu)
{
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &profiles = root["payload"].at("profiles");
if (!profiles)
return EXIT_FAILURE;
//std::map user_ids;
for (auto it = profiles.begin(); it != profiles.end(); ++it) {
// TODO: allow more users to parse at once
std::string id = (*it).name();
if (fbu->user_id == id) {
parseUser(*it, fbu);
break;
}
}
return EXIT_SUCCESS;
}
int facebook_json_parser::parse_chat_info(std::string *data, facebook_chatroom* fbc)
{
std::string jsonData = data->substr(9);
JSONNode root = JSONNode::parse(jsonData.c_str());
if (!root)
return EXIT_FAILURE;
const JSONNode &threads = root["payload"].at("threads");
if (!threads)
return EXIT_FAILURE;
/*const JSONNode &roger = payload, "roger");
if (roger) {
for (unsigned int i = 0; i < json_size(roger); i++) {
const JSONNode &it = json_at(roger, i);
std::tstring id = _A2T(json_name(it));
// Ignore "wrong" (duplicit) identifiers - these that doesn't begin with "id."
if (id.substr(0, 3) == _T("id.")) {
facebook_chatroom *room = new facebook_chatroom();
room->thread_id = id;
chatrooms->insert(std::make_pair((char*)_T2A(room->thread_id.c_str()), room));
}
}
}*/
for (auto it = threads.begin(); it != threads.end(); ++it) {
const JSONNode &is_canonical_user_ = (*it)["is_canonical_user"];
const JSONNode &thread_id_ = (*it)["thread_id"];
const JSONNode &name_ = (*it)["name"];
//const JSONNode &message_count_ = (*it)["message_count");
//const JSONNode &unread_count_ = (*it)["unread_count"); // TODO: use it to check against number of loaded messages... but how?
if (!thread_id_ || !is_canonical_user_ || is_canonical_user_.as_bool())
continue;
std::string tid = thread_id_.as_string();
// TODO: allow more users to parse at once
if (fbc->thread_id != tid)
continue;
const JSONNode &participants = (*it)["participants"];
for (auto jt = participants.begin(); jt != participants.end(); ++jt) {
std::string user_id = (*jt).as_string();
fbc->participants.insert(std::make_pair(user_id.substr(5), ""));
}
fbc->chat_name = std::tstring(ptrT(mir_utf8decodeT(name_.as_string().c_str())));
}
return EXIT_SUCCESS;
}