//*************************************************************************************** // // Google Extension plugin for the Miranda IM's Jabber protocol // Copyright (c) 2011 bems@jabber.org, George Hazan (ghazan@jabber.ru) // // 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, write to the Free Software // Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. // //*************************************************************************************** #include "StdAfx.h" #include "handlers.h" #include "db.h" #include "inbox.h" #include "notifications.h" #include "options.h" #define JABBER_EXT_GTALK_PMUC _T("pmuc-v1") #define JABBER_IQID _T("mir_") #define JABBER_IQID_FORMAT _T("mir_%d") #define NOTIFY_FEATURE_XMLNS _T("google:mail:notify") #define SETTING_FEATURE_XMLNS _T("google:setting") #define DISCOVERY_XMLNS _T("http://jabber.org/protocol/disco#info") #define MESSAGE_URL_FORMAT_STANDARD _T("%s/#inbox/%x%08x") #define MESSAGE_URL_FORMAT_HTML _T("%s/h/?v=c&th=%x%08x") #define ATTRNAME_TYPE _T("type") #define ATTRNAME_FROM _T("from") #define ATTRNAME_TO _T("to") #define ATTRNAME_URL _T("url") #define ATTRNAME_TID _T("tid") #define ATTRNAME_UNREAD _T("unread") #define ATTRNAME_XMLNS _T("xmlns") #define ATTRNAME_ID _T("id") #define ATTRNAME_TOTAL_MATCHED _T("total-matched") #define ATTRNAME_NAME _T("name") #define ATTRNAME_ADDRESS _T("address") #define ATTRNAME_RESULT_TIME _T("result-time") #define ATTRNAME_NEWER_THAN_TIME _T("newer-than-time") #define ATTRNAME_NEWER_THAN_TID _T("newer-than-tid") #define ATTRNAME_VALUE _T("value") #define ATTRNAME_VAR _T("var") #define IQTYPE_RESULT _T("result") #define IQTYPE_SET _T("set") #define IQTYPE_GET _T("get") #define NODENAME_MAILBOX _T("mailbox") #define NODENAME_QUERY _T("query") #define NODENAME_IQ _T("iq") #define NODENAME_USERSETTING _T("usersetting") #define NODENAME_MAILNOTIFICATIONS _T("mailnotifications") #define NODENAME_SUBJECT _T("subject") #define NODENAME_SNIPPET _T("snippet") #define NODENAME_SENDERS _T("senders") #define NODENAME_FEATURE _T("feature") #define NODENAME_NEW_MAIL _T("new-mail") #define SETTING_TRUE _T("true") #define RESPONSE_TIMEOUT (1000 * 60 * 60) #define TIMER_INTERVAL (1000 * 60 * 2) LRESULT CALLBACK PopupProc(HWND wnd, UINT msg, WPARAM wParam, LPARAM lParam); XML_API xi = {0}; #include void FormatMessageUrl(LPCTSTR format, LPTSTR buf, LPCTSTR mailbox, LPCTSTR tid) { ULARGE_INTEGER iTid; iTid.QuadPart = _tstoi64(tid); int l = lstrlen(buf); mir_sntprintf(buf, l, format, mailbox, iTid.HighPart, iTid.LowPart); assert(l >= lstrlen(buf)); } void MakeUrlHex(LPTSTR url, LPCTSTR tid) { ULARGE_INTEGER iTid; iTid.QuadPart = _tstoi64(tid); LPTSTR tidInUrl = _tcsstr(url, tid); LPTSTR trail = tidInUrl + lstrlen(tid); wsprintf(tidInUrl, _T("%x%08x"), iTid.HighPart, iTid.LowPart); //!!!!!!!!!!!! wmemmove(tidInUrl + lstrlen(tidInUrl), trail, lstrlen(trail) + 1); } LPTSTR ExtractJid(LPCTSTR jidWithRes) { LPCTSTR p = _tcsrchr(jidWithRes, '/'); if (p == NULL) return mir_tstrdup(jidWithRes); size_t l = size_t(p - jidWithRes); LPTSTR result = (LPTSTR)mir_alloc((l + 1) * sizeof(TCHAR)); _tcsncpy(result, jidWithRes, l); result[l] = 0; return result; } BOOL TimerHandler(IJabberInterface *ji, HXML node, void *pUserData); BOOL InternalListHandler(IJabberInterface *ji, HXML node, LPCTSTR jid, LPCTSTR mailboxUrl) { ULONGLONG maxTid = 0; LPCTSTR sMaxTid = NULL; int unreadCount = 0; for (int i = 0; i < xi.getChildCount(node); i++) { LPCTSTR sTid = xi.getAttrValue(xi.getChild(node, i), ATTRNAME_TID); ULONGLONG tid = _tcstoui64(sTid, NULL, 10); if (tid > maxTid) { maxTid = tid; sMaxTid = sTid; } HXML senders = xi.getChildByPath(xi.getChild(node, i), NODENAME_SENDERS, FALSE); for (int j = 0; j < xi.getChildCount(senders); j++) if (xi.getAttrValue(xi.getChild(senders, j), ATTRNAME_UNREAD)) { unreadCount++; break; } } LPCSTR acc = GetJidAcc(jid); if (!acc) return FALSE; if (!unreadCount) { SetupPseudocontact(jid, xi.getAttrValue(node, ATTRNAME_TOTAL_MATCHED), acc); return TRUE; } DWORD settings = ReadNotificationSettings(acc); if (unreadCount > 5) { CloseNotifications(acc, mailboxUrl, jid, FALSE); UnreadMailNotification(acc, jid, mailboxUrl, xi.getAttrValue(node, ATTRNAME_TOTAL_MATCHED)); } else for (int i = 0; i < xi.getChildCount(node); i++) { MAIL_THREAD_NOTIFICATION mtn = {0}; HXML thread = xi.getChild(node, i); mtn.subj = xi.getText(xi.getChildByPath(thread, NODENAME_SUBJECT, FALSE)); mtn.snip = xi.getText(xi.getChildByPath(thread, NODENAME_SNIPPET, FALSE)); int threadUnreadCount = 0; HXML senders = xi.getChildByPath(thread, NODENAME_SENDERS, FALSE); for (int j = 0; threadUnreadCount < SENDER_COUNT && j < xi.getChildCount(senders); j++) { HXML sender = xi.getChild(senders, j); if (xi.getAttrValue(sender, ATTRNAME_UNREAD)) { mtn.senders[threadUnreadCount].name = xi.getAttrValue(sender, ATTRNAME_NAME); mtn.senders[threadUnreadCount].addr = xi.getAttrValue(sender, ATTRNAME_ADDRESS); threadUnreadCount++; } } LPCTSTR url = xi.getAttrValue(thread, ATTRNAME_URL); LPCTSTR tid = xi.getAttrValue(thread, ATTRNAME_TID); if ( ReadCheckbox(0, IDC_STANDARDVIEW, settings)) FormatMessageUrl(MESSAGE_URL_FORMAT_STANDARD, (LPTSTR)url, mailboxUrl, tid); else if ( ReadCheckbox(0, IDC_HTMLVIEW, settings)) FormatMessageUrl(MESSAGE_URL_FORMAT_HTML, (LPTSTR)url, mailboxUrl, tid); else MakeUrlHex((LPTSTR)url, tid); CloseNotifications(acc, url, jid, i); UnreadThreadNotification(acc, jid, url, xi.getAttrValue(node, ATTRNAME_TOTAL_MATCHED), &mtn); } LPCTSTR time = xi.getAttrValue(node, ATTRNAME_RESULT_TIME); WriteJidSetting(LAST_MAIL_TIME_FROM_JID, jid, time); WriteJidSetting(LAST_THREAD_ID_FROM_JID, jid, sMaxTid); return TRUE; } BOOL MailListHandler(IJabberInterface *ji, HXML node, void *pUserData) { LPCTSTR jidWithRes = xi.getAttrValue(node, ATTRNAME_TO); __try { if (!node || lstrcmp(xi.getAttrValue(node, ATTRNAME_TYPE), IQTYPE_RESULT)) return TRUE; LPCTSTR jid = xi.getAttrValue(node, ATTRNAME_FROM); assert(jid); node = xi.getChildByPath(node, NODENAME_MAILBOX, FALSE); if (!node) return TRUE; // empty list LPCTSTR url = xi.getAttrValue(node, ATTRNAME_URL); return InternalListHandler(ji, node, jid, url); } __finally { if (jidWithRes) ji->AddTemporaryIqHandler(TimerHandler, JABBER_IQ_TYPE_RESULT, 0, (PVOID)_tcsdup(jidWithRes), TIMER_INTERVAL); // Never get a real result stanza. Results elapsed request after WAIT_TIMER_INTERVAL ms } } void RequestMail(LPCTSTR jidWithRes, IJabberInterface *ji) { HXML child = NULL; HXML node = xi.createNode(NODENAME_IQ, NULL, FALSE); xi.addAttr(node, ATTRNAME_TYPE, IQTYPE_GET); xi.addAttr(node, ATTRNAME_FROM, jidWithRes); UINT uID = ji->SerialNext(); ptrT jid( ExtractJid(jidWithRes)); xi.addAttr(node, ATTRNAME_TO, jid); ptrT lastMailTime( ReadJidSetting(LAST_MAIL_TIME_FROM_JID, jid)), lastThreadId( ReadJidSetting(LAST_THREAD_ID_FROM_JID, jid)); TCHAR id[30]; mir_sntprintf(id, SIZEOF(id), JABBER_IQID_FORMAT, uID); xi.addAttr(node, ATTRNAME_ID, id); child = xi.addChild(node, NODENAME_QUERY, NULL); xi.addAttr(child, ATTRNAME_XMLNS, NOTIFY_FEATURE_XMLNS); xi.addAttr(child, ATTRNAME_NEWER_THAN_TIME, lastMailTime); xi.addAttr(child, ATTRNAME_NEWER_THAN_TID, lastThreadId); if (ji->SendXmlNode(node)) ji->AddTemporaryIqHandler(MailListHandler, JABBER_IQ_TYPE_RESULT, (int)uID, NULL, RESPONSE_TIMEOUT); if (child) xi.destroyNode(child); if (node) xi.destroyNode(node); } BOOL TimerHandler(IJabberInterface *ji, HXML node, void *pUserData) { RequestMail((LPCTSTR)pUserData, ji); free(pUserData); return FALSE; } BOOL NewMailHandler(IJabberInterface *ji, HXML node, void *pUserData) { HXML response = xi.createNode(NODENAME_IQ, NULL, FALSE); __try { xi.addAttr(response, ATTRNAME_TYPE, IQTYPE_RESULT); LPCTSTR attr = xi.getAttrValue(node, ATTRNAME_ID); if (!attr) return FALSE; xi.addAttr(response, ATTRNAME_ID, attr); attr = xi.getAttrValue(node, ATTRNAME_FROM); if (attr) xi.addAttr(response, ATTRNAME_TO, attr); attr = xi.getAttrValue(node, ATTRNAME_TO); if (!attr) return FALSE; xi.addAttr(response, ATTRNAME_FROM, attr); int bytesSent = ji->SendXmlNode(response); RequestMail(attr, ji); return bytesSent > 0; } __finally { xi.destroyNode(response); } } void SetNotificationSetting(LPCTSTR jidWithResource, IJabberInterface *ji) { HXML child = NULL; HXML node = xi.createNode(NODENAME_IQ, NULL, FALSE); xi.addAttr(node, ATTRNAME_TYPE, IQTYPE_SET); xi.addAttr(node, ATTRNAME_FROM, jidWithResource); ptrT jid( ExtractJid(jidWithResource)); xi.addAttr(node, ATTRNAME_TO, jid); TCHAR id[30]; mir_sntprintf(id, SIZEOF(id), JABBER_IQID_FORMAT, ji->SerialNext()); xi.addAttr(node, ATTRNAME_ID, id); child = xi.addChild(node, NODENAME_USERSETTING, NULL); xi.addAttr(child, ATTRNAME_XMLNS, SETTING_FEATURE_XMLNS); child = xi.addChild(child, NODENAME_MAILNOTIFICATIONS, NULL); xi.addAttr(child, ATTRNAME_VALUE, SETTING_TRUE); ji->SendXmlNode(node); if (child) xi.destroyNode(child); if (node) xi.destroyNode(node); } BOOL DiscoverHandler(IJabberInterface *ji, HXML node, void *pUserData) { if (!node) return FALSE; LPCTSTR jid = xi.getAttrValue(node, ATTRNAME_TO); assert(jid); node = xi.getChildByAttrValue(node, NODENAME_QUERY, ATTRNAME_XMLNS, DISCOVERY_XMLNS); HXML child = xi.getChildByAttrValue(node, NODENAME_FEATURE, ATTRNAME_VAR, SETTING_FEATURE_XMLNS); if (child) SetNotificationSetting(jid, ji); child = xi.getChildByAttrValue(node, NODENAME_FEATURE, ATTRNAME_VAR, NOTIFY_FEATURE_XMLNS); if (child) { ji->AddIqHandler(NewMailHandler, JABBER_IQ_TYPE_SET, NOTIFY_FEATURE_XMLNS, NODENAME_NEW_MAIL); RequestMail(jid, ji); } return FALSE; } extern DWORD itlsRecursion; BOOL SendHandler(IJabberInterface *ji, HXML node, void *pUserData) { HXML queryNode = xi.getChildByAttrValue(node, NODENAME_QUERY, ATTRNAME_XMLNS, DISCOVERY_XMLNS); if (!queryNode) return FALSE; if ( lstrcmp(xi.getName(node), NODENAME_IQ) || lstrcmp(xi.getAttrValue(node, ATTRNAME_TYPE), IQTYPE_GET)) return FALSE; if (TlsGetValue(itlsRecursion)) return FALSE; TlsSetValue(itlsRecursion, (PVOID)TRUE); UINT id = ji->SerialNext(); HXML newNode = xi.createNode(NODENAME_IQ, NULL, FALSE); xi.addAttr(newNode, ATTRNAME_TYPE, IQTYPE_GET); xi.addAttr(newNode, ATTRNAME_TO, xi.getAttrValue(node, ATTRNAME_TO)); TCHAR idAttr[30]; mir_sntprintf(idAttr, SIZEOF(idAttr), JABBER_IQID_FORMAT, id); xi.addAttr(newNode, ATTRNAME_ID, idAttr); xi.addAttr(xi.addChild(newNode, NODENAME_QUERY, NULL), ATTRNAME_XMLNS, DISCOVERY_XMLNS); ji->SendXmlNode(newNode); xi.destroyNode(newNode); ji->AddTemporaryIqHandler(DiscoverHandler, JABBER_IQ_TYPE_RESULT, id, NULL, RESPONSE_TIMEOUT); TlsSetValue(itlsRecursion, (PVOID)FALSE); return FALSE; } ///////////////////////////////////////////////////////////////////////////////////////// // adds Google extensions into the caps list int OnExtListInit(WPARAM wParam, LPARAM lParam) { IJabberInterface *japi = (IJabberInterface*)lParam; if (g_accs.indexOf(japi) != -1) { LIST *pList = (LIST*)wParam; pList->insert(JABBER_EXT_GTALK_PMUC); } return 0; } ///////////////////////////////////////////////////////////////////////////////////////// // for our pseudo contact only our own popups are allowed // 0 = allowed, 1 = disallowed int OnFilterPopup(WPARAM wParam, LPARAM lParam) { HANDLE hContact = (HANDLE)wParam; if ( !db_get_b(hContact, SHORT_PLUGIN_NAME, PSEUDOCONTACT_FLAG, 0)) return 0; return (lParam != (LPARAM)&PopupProc); } ///////////////////////////////////////////////////////////////////////////////////////// int AccListChanged(WPARAM wParam, LPARAM lParam) { if (wParam == PRAC_ADDED) { IJabberInterface *ji = getJabberApi(((PROTOACCOUNT*)lParam)->szModuleName); if (ji) { g_accs.insert(ji); ji->AddSendHandler(SendHandler); } } return 0; } ///////////////////////////////////////////////////////////////////////////////////////// IJabberInterface* IsGoogleAccount(LPCSTR szModuleName) { IJabberInterface *japi = getJabberApi(szModuleName); if (!japi) return NULL; ptrA host( db_get_sa(NULL, szModuleName, "ManualHost")); if (host == NULL) return NULL; return ( !strcmp(host, "talk.google.com")) ? japi : NULL; } int ModulesLoaded(WPARAM wParam, LPARAM lParam) { RenewPseudocontactHandles(); int count; PROTOACCOUNT **protos; ProtoEnumAccounts(&count, &protos); for (int i=0; i < count; i++) { IJabberInterface *ji = IsGoogleAccount(protos[i]->szModuleName); if (ji) { g_accs.insert(ji); ji->AddSendHandler(SendHandler); } } HookEvent(ME_POPUP_FILTER, OnFilterPopup); HookEvent(ME_JABBER_EXTLISTINIT, OnExtListInit); HookEvent(ME_OPT_INITIALISE, OptionsInitialization); return 0; }