diff options
author | Robert Pösel <robyer@seznam.cz> | 2017-01-23 00:12:37 +0100 |
---|---|---|
committer | Robert Pösel <robyer@seznam.cz> | 2017-01-23 00:12:37 +0100 |
commit | aabbcfa26209c2186fa20d71686a5d07d6b98598 (patch) | |
tree | c10ece7f71d388d6ca6751c56140420af65d70dd | |
parent | a379587f7147343c3fee63b8495bc725d1ee6097 (diff) |
Facebook: Optimize searching pages/persons
-rw-r--r-- | protocols/FacebookRM/src/process.cpp | 172 | ||||
-rw-r--r-- | protocols/FacebookRM/src/requests/search.h | 8 |
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; |