summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRobert Pösel <robyer@seznam.cz>2017-01-23 00:12:37 +0100
committerRobert Pösel <robyer@seznam.cz>2017-01-23 00:12:37 +0100
commitaabbcfa26209c2186fa20d71686a5d07d6b98598 (patch)
treec10ece7f71d388d6ca6751c56140420af65d70dd
parenta379587f7147343c3fee63b8495bc725d1ee6097 (diff)
Facebook: Optimize searching pages/persons
-rw-r--r--protocols/FacebookRM/src/process.cpp172
-rw-r--r--protocols/FacebookRM/src/requests/search.h8
2 files changed, 85 insertions, 95 deletions
diff --git a/protocols/FacebookRM/src/process.cpp b/protocols/FacebookRM/src/process.cpp
index bd081c8e38..33a451204e 100644
--- a/protocols/FacebookRM/src/process.cpp
+++ b/protocols/FacebookRM/src/process.cpp
@@ -1208,106 +1208,100 @@ void FacebookProto::SearchAckThread(void *targ)
{
facy.handle_entry("searchAckThread");
- // We want to search separately people and then pages
- for (int searchType = 0; searchType < 2; searchType++) {
- // searchType: 0 = search people, 1 = search pages
- bool searchPages = (searchType == 1);
+ int count = 0;
+ std::string search = utils::url::encode(T2Utf((wchar_t *)targ).str());
+ std::string ssid;
+ int pn = 1;
- int count = 0;
- std::string search = utils::url::encode(T2Utf((wchar_t *)targ).str());
- std::string ssid;
- int pn = 1;
+ while (count < 50 && !isOffline())
+ {
+ SearchRequest *request = new SearchRequest(facy.mbasicWorks, search.c_str(), count, pn, ssid.c_str());
+ http::response resp = facy.sendRequest(request);
- while (count < 50 && !isOffline())
+ if (resp.code == HTTP_CODE_OK)
{
- SearchRequest *request = new SearchRequest(facy.mbasicWorks, searchPages ? SEARCH_TYPE_PAGES : SEARCH_TYPE_PEOPLE, search.c_str(), count, pn, ssid.c_str());
- http::response resp = facy.sendRequest(request);
-
- if (resp.code == HTTP_CODE_OK)
- {
- std::string items = utils::text::source_get_value(&resp.data, 4, "<body", "</form", "<table", "</table>");
-
- 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 items = utils::text::source_get_value(&resp.data, 4, "<body", "</form", "<table", "</table>");
+
+ 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;
+ std::string type; // Type of search result: 69=group, 274=page, 844=event, 2048=contact
+ 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>");
+
+ std::string url = utils::text::source_get_value(&item, 3, "<a", "href=\"", "\"");
+ std::string sld = utils::text::source_get_value2(&url, "sld=", "&\"", true);
+ // sld is Base64 encoded and then URL encoded string. So replace potential "%3D" with "="
+ utils::text::replace_all(&sld, "%3D", "=");
+ // decode Base64 string
+ ptrA data_((char*)mir_base64_decode(sld.c_str(), 0));
+ if (data_) {
+ std::string data = data_;
+ id = utils::text::source_get_value2(&data, "\"ent_id\":", ",}");
+ type = utils::text::source_get_value2(&data, "\"result_type\":", ",}");
+ }
- std::string id = utils::text::source_get_value2(&item, "?id=", "&\"");
- if (id.empty()) {
- id = utils::text::source_get_value2(&item, "?ids=", "&\"");
+ if (type == "274") { // page
+ // When searching pages we use whole name as nick and use prefix
+ nick = m_pagePrefix + " " + name;
+ name = "";
+ if (common.empty()) {
+ // Pages has additional data in <div>, not in <span> as people
+ common = utils::text::source_get_value(&item, 4, "</a>", "<div", ">", "</div>");
}
- if (id.empty()) {
- // For pages that user already likes we must parse "sld" value
- // We could parse it for all contacts this way, but above solution is more effective
- std::string url = utils::text::source_get_value(&item, 3, "<a", "href=\"", "\"");
- std::string sld = utils::text::source_get_value2(&url, "sld=", "&\"", true);
- // sld is Base64 encoded and then URL encoded string. So replace potential "%3D" with "="
- utils::text::replace_all(&sld, "%3D", "=");
- // decode Base64 string
- ptrA data_((char*)mir_base64_decode(sld.c_str(), 0));
- if (data_) {
- std::string data = data_;
- id = utils::text::source_get_value2(&data, "\"ent_id\":", ",}");
- }
+ }
+ else if (type == "2048") { // people
+ // When searching for people we try to parse nick and split first/last name
+ 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);
}
- 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 (searchPages) {
- // When searching pages we use whole name as nick and use prefix
- nick = m_pagePrefix + " " + name;
- name = "";
- if (common.empty()) {
- // Pages has additional data in <div>, not in <span> as people
- common = utils::text::source_get_value(&item, 4, "</a>", "<div", ">", "</div>");
- }
- } else {
- // When searching for people we try to parse nick and split first/last name
- 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);
- }
+ 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;
-
- ptrW tid(mir_utf8decodeW(id.c_str()));
- ptrW tname(mir_utf8decodeW(utils::text::html_entities_decode(name).c_str()));
- ptrW tsurname(mir_utf8decodeW(utils::text::html_entities_decode(surname).c_str()));
- ptrW tnick(mir_utf8decodeW(utils::text::html_entities_decode(nick).c_str()));
- ptrW tcommon(mir_utf8decodeW(utils::text::html_entities_decode(common).c_str()));
-
- PROTOSEARCHRESULT psr = { 0 };
- psr.cbSize = sizeof(psr);
- psr.flags = PSR_UNICODE;
- psr.id.w = tid;
- psr.nick.w = tnick;
- psr.firstName.w = tname;
- psr.lastName.w = tsurname;
- psr.email.w = tcommon;
- ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, targ, (LPARAM)&psr);
+ }
+ else {
+ // This is group or event, let's ignore that
+ continue;
}
- ssid = utils::text::source_get_value(&resp.data, 3, "id=\"more_objects\"", "ssid=", "&");
- pn++; // increment page number
- if (ssid.empty())
- break; // No more results
- }
- else {
- break;
+ // ignore self contact and empty ids
+ if (id.empty() || id == facy.self_.user_id)
+ continue;
+
+ ptrW tid(mir_utf8decodeW(id.c_str()));
+ ptrW tname(mir_utf8decodeW(utils::text::html_entities_decode(name).c_str()));
+ ptrW tsurname(mir_utf8decodeW(utils::text::html_entities_decode(surname).c_str()));
+ ptrW tnick(mir_utf8decodeW(utils::text::html_entities_decode(nick).c_str()));
+ ptrW tcommon(mir_utf8decodeW(utils::text::html_entities_decode(common).c_str()));
+
+ PROTOSEARCHRESULT psr = { 0 };
+ psr.cbSize = sizeof(psr);
+ psr.flags = PSR_UNICODE;
+ psr.id.w = tid;
+ psr.nick.w = tnick;
+ psr.firstName.w = tname;
+ psr.lastName.w = tsurname;
+ psr.email.w = tcommon;
+ ProtoBroadcastAck(NULL, ACKTYPE_SEARCH, ACKRESULT_DATA, targ, (LPARAM)&psr);
}
+
+ ssid = utils::text::source_get_value(&resp.data, 3, "id=\"more_objects\"", "ssid=", "&");
+ pn++; // increment page number
+ if (ssid.empty())
+ break; // No more results
+ }
+ else {
+ break;
}
}
diff --git a/protocols/FacebookRM/src/requests/search.h b/protocols/FacebookRM/src/requests/search.h
index 6ca35b10cd..5825e61cf8 100644
--- a/protocols/FacebookRM/src/requests/search.h
+++ b/protocols/FacebookRM/src/requests/search.h
@@ -23,16 +23,12 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
#ifndef _FACEBOOK_REQUEST_SEARCH_H_
#define _FACEBOOK_REQUEST_SEARCH_H_
-#define SEARCH_TYPE_PEOPLE "people"
-#define SEARCH_TYPE_PAGES "pages"
-// other type for searching is "events" or "groups", but that's pointless to search in Miranda
-
// searching
class SearchRequest : public HttpRequest
{
public:
- SearchRequest(bool mobileBasicWorks, const char *type, const char *query, int s, int pn, const char *ssid) :
- HttpRequest(REQUEST_GET, FORMAT, "%s/search/%s/", mobileBasicWorks ? FACEBOOK_SERVER_MBASIC : FACEBOOK_SERVER_MOBILE, type)
+ SearchRequest(bool mobileBasicWorks, const char *query, int s, int pn, const char *ssid) :
+ HttpRequest(REQUEST_GET, FORMAT, "%s/search/", mobileBasicWorks ? FACEBOOK_SERVER_MBASIC : FACEBOOK_SERVER_MOBILE)
{
flags |= NLHRF_REDIRECT;