/* Copyright (c) 2015 Miranda NG project (http://miranda-ng.org) 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 version 2 of the License. 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" /* HISTORY SYNC */ void CSkypeProto::OnGetServerHistory(const NETLIBHTTPREQUEST *response) { if (response == NULL) return; JSONROOT root(response->pData); if (root == NULL) return; JSONNODE *metadata = json_get(root, "_metadata"); JSONNODE *conversations = json_as_array(json_get(root, "messages")); int totalCount = json_as_int(json_get(metadata, "totalCount")); ptrA syncState(mir_t2a(ptrT(json_as_string(json_get(metadata, "syncState"))))); if (totalCount >= 99 || json_size(conversations) >= 99) PushRequest(new GetHistoryOnUrlRequest(syncState, RegToken), &CSkypeProto::OnGetServerHistory); for (size_t i = 0; i < json_size(conversations); i++) { JSONNODE *message = json_at(conversations, i); ptrA clientMsgId(mir_t2a(ptrT(json_as_string(json_get(message, "clientmessageid"))))); ptrA skypeEditedId(mir_t2a(ptrT(json_as_string(json_get(message, "skypeeditedid"))))); ptrA messageType(mir_t2a(ptrT(json_as_string(json_get(message, "messagetype"))))); ptrA from(mir_t2a(ptrT(json_as_string(json_get(message, "from"))))); ptrA content(mir_t2a(ptrT(json_as_string(json_get(message, "content"))))); ptrT composeTime(json_as_string(json_get(message, "composetime"))); ptrA conversationLink(mir_t2a(ptrT(json_as_string(json_get(message, "conversationLink"))))); time_t timestamp = IsoToUnixTime(composeTime); bool isEdited = (json_get(message, "skypeeditedid") != NULL); if (conversationLink != NULL && strstr(conversationLink, "/8:")) { if (!mir_strcmpi(messageType, "Text") || !mir_strcmpi(messageType, "RichText")) { int emoteOffset = json_as_int(json_get(message, "skypeemoteoffset")); int flags = DBEF_UTF | DBEF_READ; ptrA skypename(ContactUrlToName(from)); bool isMe = IsMe(skypename); if (isMe) flags |= DBEF_SENT; MCONTACT hContact = FindContact(ptrA(ContactUrlToName(conversationLink))); ptrA message(RemoveHtml(content)); MEVENT dbevent = GetMessageFromDb(hContact, skypeEditedId); if (isEdited && dbevent != NULL) { DBEVENTINFO dbei = { sizeof(dbei) }; CMStringA msg; dbei.cbBlob = db_event_getBlobSize(dbevent); mir_ptr blob((PBYTE)mir_alloc(dbei.cbBlob)); dbei.pBlob = blob; db_event_get(dbevent, &dbei); time_t dbEventTimestamp = dbei.timestamp; ptrA dbMsgText((char *)mir_alloc(dbei.cbBlob)); mir_strcpy(dbMsgText, (char*)dbei.pBlob); msg.AppendFormat("%s\n%s [%s]:\n%s", dbMsgText, Translate("Edited at"), ptrA(mir_t2a(composeTime)), message); db_event_delete(hContact, dbevent); AddMessageToDb(hContact, dbEventTimestamp, flags, clientMsgId, msg.GetBuffer(), emoteOffset); } else AddMessageToDb(hContact, timestamp, flags, clientMsgId, message, emoteOffset); } else if (!mir_strcmpi(messageType, "Event/Call")) { //content=user name6 //Echo / Sound Test Service6 //content=user name MCONTACT hContact = FindContact(ptrA(ContactUrlToName(conversationLink))); ptrA name; int iType = 3, iDuration = 0; ptrA skypename(ContactUrlToName(from)); bool isMe = IsMe(skypename); int flags = DBEF_UTF | DBEF_READ; if (isMe) flags |= DBEF_SENT; HXML xml = xi.parseString(ptrT(mir_a2t(content)), 0, _T("partlist")); if (xml != NULL) { ptrA type(mir_t2a(xi.getAttrValue(xml, _T("type")))); if (!mir_strcmpi(type, "ended")) iType = 0; else if (!mir_strcmpi(type, "started")) iType = 1; HXML xmlNode = xi.getChildByPath(xml, _T("part"), 0); HXML duration = xmlNode == NULL ? NULL : xi.getChildByPath(xmlNode, _T("duration"), 0); iDuration = duration != NULL ? atoi(mir_t2a(xi.getText(duration))) : NULL; xi.destroyNode(xml); } CMStringA text = ""; if (iType == 1) text.Append(Translate("Call started")); else if (iType == 0) { CMStringA chours = "", cmins = "", csec = ""; int hours=0, mins=0, sec=0; if (iDuration != NULL) { hours = iDuration / 3600; mins = iDuration / 60; sec = iDuration % 60; } else hours=mins=sec=0; if (hours < 10) chours.AppendFormat("0%d", hours); else chours.AppendFormat("%d", hours); if (mins < 10) cmins.AppendFormat("0%d", mins); else cmins.AppendFormat("%d", mins); if (sec < 10) csec.AppendFormat("0%d", sec); else csec.AppendFormat("%d", sec); text.AppendFormat("%s\n%s: %s:%s:%s", Translate("Call ended"), Translate("Duration"), chours, cmins, csec); } AddMessageToDb(hContact, timestamp, flags, clientMsgId, text.GetBuffer()); } } else if (conversationLink != NULL && strstr(conversationLink, "/19:")) { ptrA chatname(ChatUrlToName(conversationLink)); if (!mir_strcmpi(messageType, "Text") || !mir_strcmpi(messageType, "RichText")) { GCDEST gcd = { m_szModuleName, _A2T(chatname), GC_EVENT_MESSAGE }; GCEVENT gce = { sizeof(GCEVENT), &gcd }; gce.bIsMe = IsMe(ContactUrlToName(from)); gce.ptszUID = mir_a2t(ContactUrlToName(from)); gce.time = timestamp; gce.ptszNick = mir_a2t(ContactUrlToName(from)); gce.ptszText = mir_a2t(RemoveHtml(content)); gce.dwFlags = GCEF_NOTNOTIFY; CallServiceSync(MS_GC_EVENT, 0, (LPARAM)&gce); } } } } INT_PTR CSkypeProto::GetContactHistory(WPARAM hContact, LPARAM) { PushRequest(new GetHistoryRequest(RegToken, ptrA(db_get_sa(hContact, m_szModuleName, SKYPE_SETTINGS_ID)), 100, false, 0, Server), &CSkypeProto::OnGetServerHistory); return 0; } void CSkypeProto::SyncHistory() { PushRequest(new SyncHistoryFirstRequest(RegToken, 100, Server), &CSkypeProto::OnSyncHistory); } void CSkypeProto::OnSyncHistory(const NETLIBHTTPREQUEST *response) { if (response == NULL) return; JSONROOT root(response->pData); if (root == NULL) return; JSONNODE *metadata = json_get(root, "_metadata"); JSONNODE *conversations = json_as_array(json_get(root, "conversations")); int totalCount = json_as_int(json_get(metadata, "totalCount")); ptrA syncState(mir_t2a(ptrT(json_as_string(json_get(metadata, "syncState"))))); if (totalCount >= 99 || json_size(conversations) >= 99) PushRequest(new SyncHistoryFirstRequest(syncState, RegToken), &CSkypeProto::OnSyncHistory); for (size_t i = 0; i < json_size(conversations); i++) { JSONNODE *conversation = json_at(conversations, i); JSONNODE *lastMessage = json_get(conversation, "lastMessage"); JSONNODE *threadProperties = json_get(conversation, "threadProperties"); if (json_empty(lastMessage)) continue; char *clientMsgId = mir_t2a(json_as_string(json_get(lastMessage, "clientmessageid"))); char *skypeEditedId = mir_t2a(json_as_string(json_get(lastMessage, "skypeeditedid"))); bool isEdited = (skypeEditedId != NULL); char *conversationLink = mir_t2a(json_as_string(json_get(lastMessage, "conversationLink"))); time_t composeTime(IsoToUnixTime(ptrT(json_as_string(json_get(lastMessage, "conversationLink"))))); bool isChat = false; ptrA skypename; ptrT topic; if (conversationLink != NULL && strstr(conversationLink, "/8:")) { skypename = ContactUrlToName(conversationLink); } else if (conversationLink != NULL && strstr(conversationLink, "/19:")) { skypename = ChatUrlToName(conversationLink); isChat = true; topic = json_as_string(json_get(threadProperties, "topic")); StartChatRoom(_A2T(skypename), topic); } else continue; MCONTACT hContact = isChat ? NULL : AddContact(skypename); if (hContact == NULL || GetMessageFromDb(hContact, clientMsgId, composeTime) == NULL) PushRequest(new GetHistoryRequest(RegToken, skypename, !isChat ? 100 : 15, isChat, 0,Server), &CSkypeProto::OnGetServerHistory); } }