/* Jabber Protocol Plugin for Miranda NG Copyright (c) 2002-04 Santithorn Bunchua Copyright (c) 2005-12 George Hazan Copyright (c) 2007 Maxim Mluhov Copyright (C) 2012-23 Miranda NG team 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 "jabber_iq.h" #include "jabber_caps.h" #include "jabber_disco.h" #include "jabber_secur.h" #pragma warning(disable:4355) static int compareTransports(const char *p1, const char *p2) { return mir_strcmpi(p1, p2); } static int compareListItems(const JABBER_LIST_ITEM *p1, const JABBER_LIST_ITEM *p2) { if (p1->list != p2->list) return p1->list - p2->list; // for bookmarks, temporary contacts & groupchat members // resource must be used in the comparison if ((p1->list == LIST_ROSTER && (p1->bUseResource == true || p2->bUseResource == true)) || (p1->list == LIST_BOOKMARK) || (p1->list == LIST_VCARD_TEMP)) return mir_strcmpi(p1->jid, p2->jid); char szp1[JABBER_MAX_JID_LEN], szp2[JABBER_MAX_JID_LEN]; JabberStripJid(p1->jid, szp1, _countof(szp1)); JabberStripJid(p2->jid, szp2, _countof(szp2)); return mir_strcmpi(szp1, szp2); } CJabberProto::CJabberProto(const char *aProtoName, const wchar_t *aUserName) : PROTO(aProtoName, aUserName), m_impl(*this), m_omemo(this), m_arChatMarks(50, NumericKeySortT), m_arAuthMechs(1, &TJabberAuth::compare), m_lstTransports(50, compareTransports), m_lstRoster(50, compareListItems), m_iqManager(this), m_messageManager(this), m_presenceManager(this), m_sendManager(this), m_adhocManager(this), m_privacyListManager(this), m_privacyMenuServiceAllocated(-1), m_priorityMenuVal(0), m_priorityMenuValSet(false), m_hPrivacyMenuRoot(nullptr), m_hPrivacyMenuItems(10), m_lstJabberFeatCapPairsDynamic(2), m_uEnabledFeatCapsDynamic(0), m_StrmMgmt(this), m_bAcceptHttpAuth(this, "AcceptHttpAuth", true), m_bAcceptNotes(this, "AcceptNotes", true), m_bAllowTimeReplies(this, "AllowTimeReplies", true), m_bAllowVersionRequests(this, "AllowVersionRequests", true), m_bAutoAcceptAuthorization(this, "AutoAcceptAuthorization", false), m_bAutoAcceptMUC(this, "AutoAcceptMUC", false), m_bAutoAdd(this, "AutoAdd", true), m_bAutoJoinBookmarks(this, "AutoJoinBookmarks", true), m_bAutoJoinConferences(this, "AutoJoinConferences", false), m_bAutoJoinHidden(this, "AutoJoinHidden", true), m_bAutosaveNotes(this, "AutosaveNotes", false), m_bBsDirect(this, "BsDirect", true), m_bBsDirectManual(this, "BsDirectManual", false), m_bBsProxyManual(this, "BsProxyManual", false), m_bDisable3920auth(this, "Disable3920auth", false), m_bEnableAvatars(this, "EnableAvatars", true), m_bEnableCarbons(this, "EnableCarbons", true), m_bEnableChatStates(this, "EnableChatStates", true), m_bEnableMam(this, "EnableMam", true), m_bEnableMsgArchive(this, "EnableMsgArchive", false), m_bEnableRemoteControl(this, "EnableRemoteControl", false), m_bEnableStreamMgmt(this, "UseStreamMgmt", false), m_bEnableUserActivity(this, "EnableUserActivity", true), m_bEnableUserMood(this, "EnableUserMood", true), m_bEnableUserTune(this, "EnableUserTune", false), m_bEnableVOIP(this, "EnableVOIP", false), m_bEnableZlib(this, "EnableZlib", true), m_bFixIncorrectTimestamps(this, "FixIncorrectTimestamps", true), m_bGcLogAffiliations(this, "GcLogAffiliations", false), m_bGcLogBans(this, "GcLogBans", true), m_bGcLogChatHistory(this, "GcLogChatHistory", true), m_bGcLogConfig(this, "GcLogConfig", false), m_bGcLogRoles(this, "GcLogRoles", false), m_bGcLogStatuses(this, "GcLogStatuses", false), m_bHostNameAsResource(this, "HostNameAsResource", false), m_bIgnoreMUCInvites(this, "IgnoreMUCInvites", false), m_bIgnoreRoster(this, "IgnoreRosterGroups", false), m_bInlinePictures(this, "InlinePictures", false), m_bKeepAlive(this, "KeepAlive", true), m_bLogChatstates(this, "LogChatstates", false), m_bLogPresence(this, "LogPresence", true), m_bLogPresenceErrors(this, "LogPresenceErrors", false), m_bManualConnect(this, "ManualConnect", false), m_bMsgAck(this, "MsgAck", true), m_bProcessXMPPLinks(this, "ProcessXMPPLinks", false), m_bEmbraceUrls(this, "EmbraceUrls", false), m_bRcMarkMessagesAsRead(this, "RcMarkMessagesAsRead", true), m_bRosterSync(this, "RosterSync", false), m_bSavePassword(this, "SavePassword", true), m_bShowForeignResourceInMirVer(this, "ShowForeignResourceInMirVer", false), m_bShowOSVersion(this, "ShowOSVersion", false), m_bShowTransport(this, "ShowTransport", true), m_bUseDomainLogin(this, "UseDomainLogin", false), m_bUseHttpUpload(this, "UseHttpUpload", false), m_bUseOMEMO(this, "UseOMEMO", false), m_bUsePopups(this, "UsePopups", true), m_bUseSSL(this, "UseSSL", false), m_bUseTLS(this, "UseTLS", true), m_bUseTlsExport(this, "UseTlsExport", false), m_iMamMode(this, "MamMode", 0), m_iConnectionKeepAliveInterval(this, "ConnectionKeepAliveInterval", 60000), m_iConnectionKeepAliveTimeout(this, "ConnectionKeepAliveTimeout", 50000) { debugLogA("Setting protocol/module name to '%s'", m_szModuleName); // Protocol services and events... m_hEventNudge = CreateProtoEvent(PE_NUDGE); m_hEventXStatusIconChanged = CreateProtoEvent(JE_CUSTOMSTATUS_EXTRAICON_CHANGED); m_hEventXStatusChanged = CreateProtoEvent(JE_CUSTOMSTATUS_CHANGED); CreateProtoService(PS_GETAVATARINFO, &CJabberProto::JabberGetAvatarInfo); CreateProtoService(PS_GETMYAWAYMSG, &CJabberProto::GetMyAwayMsg); CreateProtoService(PS_SET_LISTENINGTO, &CJabberProto::OnSetListeningTo); CreateProtoService(PS_JOINCHAT, &CJabberProto::OnJoinChat); CreateProtoService(PS_LEAVECHAT, &CJabberProto::OnLeaveChat); CreateProtoService(PS_GETCUSTOMSTATUSEX, &CJabberProto::OnGetXStatusEx); CreateProtoService(PS_SETCUSTOMSTATUSEX, &CJabberProto::OnSetXStatusEx); CreateProtoService(PS_GETCUSTOMSTATUSICON, &CJabberProto::OnGetXStatusIcon); CreateProtoService(PS_GETADVANCEDSTATUSICON, &CJabberProto::OnGetAdvancedStatusIcon); CreateProtoService(JS_HTTP_AUTH, &CJabberProto::OnHttpAuthRequest); CreateProtoService(JS_INCOMING_NOTE_EVENT, &CJabberProto::OnIncomingNoteEvent); CreateProtoService(JS_SENDXML, &CJabberProto::ServiceSendXML); CreateProtoService(PS_GETMYAVATAR, &CJabberProto::JabberGetAvatar); CreateProtoService(PS_GETAVATARCAPS, &CJabberProto::JabberGetAvatarCaps); CreateProtoService(PS_SETMYAVATAR, &CJabberProto::JabberSetAvatar); CreateProtoService(PS_SETMYNICKNAME, &CJabberProto::JabberSetNickname); CreateProtoService(PS_MENU_REQAUTH, &CJabberProto::OnMenuHandleRequestAuth); CreateProtoService(PS_MENU_GRANTAUTH, &CJabberProto::OnMenuHandleGrantAuth); CreateProtoService(PS_MENU_REVOKEAUTH, &CJabberProto::OnMenuHandleRevokeAuth); CreateProtoService(PS_MENU_LOADHISTORY, &CJabberProto::OnMenuLoadHistory); CreateProtoService(JS_DB_GETEVENTTEXT_CHATSTATES, &CJabberProto::OnGetEventTextChatStates); CreateProtoService(JS_DB_GETEVENTTEXT_PRESENCE, &CJabberProto::OnGetEventTextPresence); CreateProtoService(JS_GETJABBERAPI, &CJabberProto::JabberGetApi); // Voice m_hVoiceEvent = CreateProtoEvent(PE_VOICE_CALL_STATE); CreateProtoService(PS_VOICE_CALL, &CJabberProto::JabberVOIP_call); CreateProtoService(PS_VOICE_ANSWERCALL, &CJabberProto::JabberVOIP_answercall); CreateProtoService(PS_VOICE_DROPCALL, &CJabberProto::JabberVOIP_dropcall); // XEP-0224 support (Attention/Nudge) CreateProtoService(PS_SEND_NUDGE, &CJabberProto::JabberSendNudge); // Offline file transfer CreateProtoService(PS_OFFLINEFILE, &CJabberProto::OnOfflineFile); // service to get from protocol chat buddy info CreateProtoService(MS_GC_PROTO_GETTOOLTIPTEXT, &CJabberProto::JabberGCGetToolTipText); // XMPP URI parser service for "File Association Manager" plugin CreateProtoService(JS_PARSE_XMPP_URI, &CJabberProto::JabberServiceParseXmppURI); HookProtoEvent(ME_LANGPACK_CHANGED, &CJabberProto::OnLangChanged); HookProtoEvent(ME_OPT_INITIALISE, &CJabberProto::OnOptionsInit); HookProtoEvent(ME_SKIN_ICONSCHANGED, &CJabberProto::OnReloadIcons); HookProtoEvent(ME_DB_CONTACT_SETTINGCHANGED, &CJabberProto::OnDbSettingChanged); m_iqManager.FillPermanentHandlers(); m_messageManager.FillPermanentHandlers(); m_adhocManager.FillDefaultNodes(); IconsInit(); m_pepServices.insert(new CPepMood(this)); m_pepServices.insert(new CPepActivity(this)); if (m_bUseOMEMO) { db_set_resident(m_szModuleName, "OmemoSessionChecked"); OmemoInitDevice(); } // avatars CreateDirectoryTreeW(GetAvatarPath()); // network initialization NETLIBUSER nlu = {}; nlu.flags = NUF_OUTGOING | NUF_INCOMING | NUF_HTTPCONNS | NUF_UNICODE; nlu.szDescriptiveName.w = m_tszUserName; nlu.szSettingsModule = m_szModuleName; m_hNetlibUser = Netlib_RegisterUser(&nlu); // group chats GCREGISTER gcr = {}; gcr.dwFlags = GC_TYPNOTIF | GC_CHANMGR; gcr.ptszDispName = m_tszUserName; gcr.pszModule = m_szModuleName; Chat_Register(&gcr); HookProtoEvent(ME_GC_EVENT, &CJabberProto::JabberGcEventHook); HookProtoEvent(ME_GC_BUILDMENU, &CJabberProto::JabberGcMenuHook); // resident settings db_set_resident(m_szModuleName, DBSETTING_XSTATUSID); db_set_resident(m_szModuleName, DBSETTING_XSTATUSNAME); db_set_resident(m_szModuleName, DBSETTING_XSTATUSMSG); db_set_resident(m_szModuleName, DBSETTING_DISPLAY_UID); db_set_resident(m_szModuleName, "SubscriptionText"); db_set_resident(m_szModuleName, "Subscription"); db_set_resident(m_szModuleName, "Auth"); db_set_resident(m_szModuleName, "Grant"); if ((m_tszSelectedLang = getUStringA("XmlLang")) == nullptr) m_tszSelectedLang = mir_strdup("en"); g_plugin.addPopupOption(CMStringW(FORMAT, TranslateT("%s error notifications"), m_tszUserName), m_bUsePopups); } CJabberProto::~CJabberProto() { ConsoleUninit(); GlobalMenuUninit(); if (m_hPopupClass) Popup_UnregisterClass(m_hPopupClass); // Events DestroyHookableEvent(m_hEventNudge); DestroyHookableEvent(m_hEventXStatusIconChanged); DestroyHookableEvent(m_hEventXStatusChanged); DestroyHookableEvent(m_hVoiceEvent); // Voice if (hasJingle()) InitVoip(false); // Lists & strings ListWipe(); mir_free(m_tszSelectedLang); mir_free(m_modeMsgs.szOnline); mir_free(m_modeMsgs.szAway); mir_free(m_modeMsgs.szNa); mir_free(m_modeMsgs.szDnd); mir_free(m_modeMsgs.szFreechat); mir_free(m_transportProtoTableStartIndex); for (auto &it : m_lstTransports) mir_free(it); for (auto &it : m_lstJabberFeatCapPairsDynamic) { mir_free(it->szExt); mir_free(it->szFeature); mir_free(it->szDescription); delete it; } } //////////////////////////////////////////////////////////////////////////////////////// // OnModulesLoadedEx - performs hook registration void CJabberProto::OnModulesLoaded() { HookProtoEvent(ME_USERINFO_INITIALISE, &CJabberProto::OnUserInfoInit); XStatusInit(); m_pepServices.InitGui(); InitPopups(); GlobalMenuInit(); UpdateFeatHash(); StatusIconData sid = {}; sid.szModule = m_szModuleName; sid.hIcon = IcoLib_GetIconByHandle(m_hProtoIcon); sid.flags = MBF_HIDDEN; sid.szTooltip.a = LPGEN("Jabber Resource"); Srmm_AddIcon(&sid, &g_plugin); HookProtoEvent(ME_MSG_ICONPRESSED, &CJabberProto::OnProcessSrmmIconClick); HookProtoEvent(ME_MSG_WINDOWEVENT, &CJabberProto::OnProcessSrmmEvent); DBEVENTTYPEDESCR dbEventType = {}; dbEventType.module = m_szModuleName; dbEventType.eventType = EVENTTYPE_JABBER_CHATSTATES; dbEventType.descr = "Chat state notifications"; DbEvent_RegisterType(&dbEventType); dbEventType.eventType = EVENTTYPE_JABBER_PRESENCE; dbEventType.descr = "Presence notifications"; DbEvent_RegisterType(&dbEventType); HookProtoEvent(ME_IDLE_CHANGED, &CJabberProto::OnIdleChanged); CheckAllContactsAreTransported(); // Voip if (hasJingle()) InitVoip(true); // Set all contacts to offline for (auto &hContact : AccContacts()) { SetContactOfflineStatus(hContact); if (getByte(hContact, "IsTransport", 0)) { ptrA jid(getUStringA(hContact, "jid")); if (jid == nullptr) continue; char *resourcepos = strchr(jid, '/'); if (resourcepos != nullptr) *resourcepos = '\0'; m_lstTransports.insert(mir_strdup(jid)); } } } /////////////////////////////////////////////////////////////////////////////// // OnPreShutdown - prepares Miranda to be shut down void CJabberProto::OnShutdown() { m_bShutdown = true; UI_SAFE_CLOSE_HWND(m_hwndJabberChangePassword); UI_SAFE_CLOSE_HWND(m_hwndJabberAddBookmark); UI_SAFE_CLOSE_HWND(m_hwndPrivacyRule); UI_SAFE_CLOSE(m_pDlgPrivacyLists); UI_SAFE_CLOSE(m_pDlgBookmarks); UI_SAFE_CLOSE(m_pDlgServiceDiscovery); UI_SAFE_CLOSE(m_pDlgJabberJoinGroupchat); UI_SAFE_CLOSE(m_pDlgNotes); AgentShutdown(); MucShutdown(); m_iqManager.ExpireAll(); ConsoleUninit(); Srmm_RemoveIcon(m_szModuleName, 0); } //////////////////////////////////////////////////////////////////////////////////////// // JabberAddToList - adds a contact to the contact list MCONTACT CJabberProto::AddToListByJID(const char *newJid, uint32_t flags) { debugLogA("AddToListByJID jid = %s", newJid); MCONTACT hContact = DBCreateContact(newJid, nullptr, true, false); if (flags & PALF_TEMPORARY) Contact::Hide(hContact); return hContact; } MCONTACT CJabberProto::AddToList(int flags, PROTOSEARCHRESULT* psr) { if (psr->cbSize != sizeof(PROTOSEARCHRESULT) && psr->id.w == nullptr) return 0; return AddToListByJID(T2Utf(psr->id.w), flags); } MCONTACT CJabberProto::AddToListByEvent(int flags, int /*iContact*/, MEVENT hDbEvent) { debugLogA("AddToListByEvent"); DB::EventInfo dbei(hDbEvent); if (!dbei) return 0; if (mir_strcmp(dbei.szModule, m_szModuleName)) return 0; if (dbei.eventType != EVENTTYPE_AUTHREQUEST) return 0; DB::AUTH_BLOB blob(dbei.pBlob); return AddToListByJID(blob.get_email(), flags); } //////////////////////////////////////////////////////////////////////////////////////// // JabberAuthAllow - processes the successful authorization int CJabberProto::Authorize(MEVENT hDbEvent) { if (!m_bJabberOnline) return 1; DB::EventInfo dbei(hDbEvent); if (!dbei) return 1; if (dbei.eventType != EVENTTYPE_AUTHREQUEST) return 1; if (mir_strcmp(dbei.szModule, m_szModuleName)) return 1; DB::AUTH_BLOB blob(dbei.pBlob); debugLogA("Send 'authorization allowed' to %s", blob.get_email()); m_ThreadInfo->send(XmlNode("presence") << XATTR("to", blob.get_email()) << XATTR("type", "subscribed")); // Automatically add this user to my roster if option is enabled if (m_bAutoAdd) { JABBER_LIST_ITEM *item = ListGetItemPtr(LIST_ROSTER, blob.get_email()); if (item == nullptr || (item->subscription != SUB_BOTH && item->subscription != SUB_TO)) { debugLogA("Try adding contact automatically jid = %s", blob.get_email()); if (MCONTACT hContact = AddToListByJID(blob.get_email(), 0)) { // Trigger actual add by removing the "NotOnList" added by AddToListByJID() // See AddToListByJID() and JabberDbSettingChanged(). Contact::PutOnList(hContact); } } } return 0; } //////////////////////////////////////////////////////////////////////////////////////// // JabberAuthDeny - handles the unsuccessful authorization int CJabberProto::AuthDeny(MEVENT hDbEvent, const wchar_t*) { if (!m_bJabberOnline) return 1; debugLogA("Entering AuthDeny"); DB::EventInfo dbei(hDbEvent); if (!dbei) return 1; if (dbei.eventType != EVENTTYPE_AUTHREQUEST) return 1; if (mir_strcmp(dbei.szModule, m_szModuleName)) return 1; char *nick = (char*)(dbei.pBlob + sizeof(uint32_t) * 2); char *firstName = nick + mir_strlen(nick) + 1; char *lastName = firstName + mir_strlen(firstName) + 1; char *jid = lastName + mir_strlen(lastName) + 1; debugLogA("Send 'authorization denied' to %s", jid); ptrA newJid(dbei.flags & DBEF_UTF ? mir_strdup(jid) : mir_utf8encode(jid)); m_ThreadInfo->send(XmlNode("presence") << XATTR("to", newJid) << XATTR("type", "unsubscribed")); return 0; } //////////////////////////////////////////////////////////////////////////////////////// // JabberAuthRecv - receives a auth int CJabberProto::AuthRecv(MCONTACT, PROTORECVEVENT *pre) { return Proto_AuthRecv(m_szModuleName, pre); } //////////////////////////////////////////////////////////////////////////////////////// // JabberFileAllow - starts a file transfer HANDLE CJabberProto::FileAllow(MCONTACT /*hContact*/, HANDLE hTransfer, const wchar_t *szPath) { if (!m_bJabberOnline) return nullptr; filetransfer *ft = (filetransfer*)hTransfer; ft->std.szWorkingDir.w = mir_wstrdup(szPath); size_t len = mir_wstrlen(ft->std.szWorkingDir.w) - 1; if (ft->std.szWorkingDir.w[len] == '/' || ft->std.szWorkingDir.w[len] == '\\') ft->std.szWorkingDir.w[len] = 0; switch (ft->type) { case FT_OOB: ForkThread((MyThreadFunc)&CJabberProto::FileReceiveThread, ft); break; case FT_HTTP: ForkThread((MyThreadFunc)&CJabberProto::FileReceiveHttpThread, ft); break; case FT_BYTESTREAM: FtAcceptSiRequest(ft); break; case FT_IBB: FtAcceptIbbRequest(ft); break; } return hTransfer; } //////////////////////////////////////////////////////////////////////////////////////// // JabberFileCancel - cancels a file transfer int CJabberProto::FileCancel(MCONTACT, HANDLE hTransfer) { filetransfer *ft = (filetransfer*)hTransfer; HANDLE hEvent; debugLogA("Invoking FileCancel()"); if (ft->type == FT_OOB) { if (ft->s) { debugLogA("FT canceled"); debugLogA("Closing ft->s = %d", ft->s); ft->state = FT_ERROR; Netlib_CloseHandle(ft->s); ft->s = nullptr; if (ft->hFileEvent != nullptr) { hEvent = ft->hFileEvent; ft->hFileEvent = nullptr; SetEvent(hEvent); } debugLogA("ft->s is now nullptr, ft->state is now FT_ERROR"); } } else FtCancel(ft); return 0; } //////////////////////////////////////////////////////////////////////////////////////// // JabberFileDeny - denies a file transfer int CJabberProto::FileDeny(MCONTACT, HANDLE hTransfer, const wchar_t *) { if (!m_bJabberOnline) return 1; filetransfer *ft = (filetransfer*)hTransfer; switch (ft->type) { case FT_OOB: m_ThreadInfo->send(XmlNodeIq("error", ft->szId, ft->jid) << XCHILD("error", "File transfer refused") << XATTRI("code", 406)); break; case FT_BYTESTREAM: case FT_IBB: m_ThreadInfo->send( XmlNodeIq("error", ft->szId, ft->jid) << XCHILD("error", "File transfer refused") << XATTRI("code", 403) << XATTR("type", "cancel") << XCHILDNS("forbidden", "urn:ietf:params:xml:ns:xmpp-stanzas") << XCHILD("text", "File transfer refused") << XATTR("xmlns", "urn:ietf:params:xml:ns:xmpp-stanzas")); break; } delete ft; return 0; } //////////////////////////////////////////////////////////////////////////////////////// // JabberFileResume - processes file renaming etc int CJabberProto::FileResume(HANDLE hTransfer, int, const wchar_t *szFilename) { filetransfer *ft = (filetransfer*)hTransfer; if (!m_bJabberOnline || ft == nullptr) return 1; if (szFilename != nullptr) replaceStrW(ft->std.szCurrentFile.w, szFilename); SetEvent(ft->hWaitEvent); return 0; } //////////////////////////////////////////////////////////////////////////////////////// // GetCaps - return protocol capabilities bits INT_PTR CJabberProto::GetCaps(int type, MCONTACT hContact) { uint32_t dwFlags; switch (type) { case PFLAGNUM_1: return PF1_IM | PF1_AUTHREQ | PF1_CHAT | PF1_SERVERCLIST | PF1_MODEMSG | PF1_BASICSEARCH | PF1_EXTSEARCH | PF1_FILE | PF1_CONTACT; case PFLAGNUM_2: return PF2_ONLINE | PF2_INVISIBLE | PF2_SHORTAWAY | PF2_LONGAWAY | PF2_HEAVYDND | PF2_FREECHAT; case PFLAGNUM_3: return PF2_ONLINE | PF2_SHORTAWAY | PF2_LONGAWAY | PF2_HEAVYDND | PF2_FREECHAT; case PFLAGNUM_4: dwFlags = PF4_FORCEAUTH | PF4_NOCUSTOMAUTH | PF4_NOAUTHDENYREASON | PF4_SUPPORTTYPING | PF4_AVATARS | PF4_READNOTIFY; if (m_bUseHttpUpload || m_bInlinePictures) dwFlags |= PF4_OFFLINEFILES; if (m_ThreadInfo && (m_ThreadInfo->jabberServerCaps & JABBER_CAPS_MAM)) dwFlags |= PF4_SERVERMSGID; return dwFlags; case PFLAG_UNIQUEIDTEXT: return (INT_PTR)TranslateT("JID"); case PFLAG_MAXCONTACTSPERPACKET: char szClientJid[JABBER_MAX_JID_LEN]; if (GetClientJID(hContact, szClientJid, _countof(szClientJid))) { JabberCapsBits jcb = GetResourceCapabilities(szClientJid); return ((~jcb & JABBER_CAPS_ROSTER_EXCHANGE) ? 0 : 50); } break; case PFLAG_GETCURRENTENCRYPTION: return (INT_PTR)((OmemoIsEnabled(hContact) && m_bUseOMEMO && !isChatRoom(hContact)) ? "OMEMO" : nullptr); } return 0; } //////////////////////////////////////////////////////////////////////////////////////// // GetInfo - retrieves a contact info int CJabberProto::GetInfo(MCONTACT hContact, int /*infoType*/) { if (!m_bJabberOnline || isChatRoom(hContact)) return 1; ptrA jid(getUStringA(hContact, "jid")); if (!jid) return 1; char szBareJid[JABBER_MAX_JID_LEN]; JabberStripJid(jid, szBareJid, _countof(szBareJid)); if (m_ThreadInfo) { auto *pInfo = AddIQ(&CJabberProto::OnIqResultEntityTime, JABBER_IQ_TYPE_GET, jid); pInfo->SetParamsToParse(JABBER_IQ_PARSE_HCONTACT); m_ThreadInfo->send(XmlNodeIq(pInfo) << XCHILDNS("time", JABBER_FEAT_ENTITY_TIME)); // XEP-0012, last logoff time pInfo = AddIQ(&CJabberProto::OnIqResultLastActivity, JABBER_IQ_TYPE_GET, jid); pInfo->SetParamsToParse(JABBER_IQ_PARSE_FROM); m_ThreadInfo->send(XmlNodeIq(pInfo) << XQUERY(JABBER_FEAT_LAST_ACTIVITY)); JABBER_LIST_ITEM *item = nullptr; if ((item = ListGetItemPtr(LIST_VCARD_TEMP, jid)) == nullptr) item = ListGetItemPtr(LIST_ROSTER, jid); if (item == nullptr) { bool bHasResource = mir_strcmp(jid, szBareJid) != 0; JABBER_LIST_ITEM *tmpItem = nullptr; if (bHasResource && (tmpItem = ListGetItemPtr(LIST_CHATROOM, szBareJid))) { pResourceStatus him(tmpItem->findResource(szBareJid + mir_strlen(szBareJid) + 1)); if (him) { item = ListAdd(LIST_VCARD_TEMP, jid, hContact); ListAddResource(LIST_VCARD_TEMP, jid, him->m_iStatus, him->m_szStatusMessage, him->m_iPriority); } } else item = ListAdd(LIST_VCARD_TEMP, jid, hContact); } if (item != nullptr) { if (item->arResources.getCount()) { for (auto &it : item->arResources) { pResourceStatus r(it); CMStringA tmp(MakeJid(szBareJid, r->m_szResourceName)); if (r->m_jcbCachedCaps & JABBER_CAPS_DISCO_INFO) { pInfo = AddIQ(&CJabberProto::OnIqResultCapsDiscoInfo, JABBER_IQ_TYPE_GET, tmp); pInfo->SetParamsToParse(JABBER_IQ_PARSE_FROM | JABBER_IQ_PARSE_CHILD_TAG_NODE | JABBER_IQ_PARSE_HCONTACT); m_ThreadInfo->send(XmlNodeIq(pInfo) << XQUERY(JABBER_FEAT_DISCO_INFO)); } if (mir_strcmp(tmp, jid)) { // skip current resource, we've already sent this iq to it pInfo = AddIQ(&CJabberProto::OnIqResultLastActivity, JABBER_IQ_TYPE_GET, tmp); pInfo->SetParamsToParse(JABBER_IQ_PARSE_FROM); m_ThreadInfo->send(XmlNodeIq(pInfo) << XQUERY(JABBER_FEAT_LAST_ACTIVITY)); } } } } } SendGetVcard(hContact); OmemoRequestDeviceList(jid); return 0; } //////////////////////////////////////////////////////////////////////////////////////// // SearchBasic - searches the contact by JID struct JABBER_SEARCH_BASIC { int hSearch; wchar_t jid[128]; }; void __cdecl CJabberProto::BasicSearchThread(JABBER_SEARCH_BASIC *jsb) { Thread_SetName("Jabber: BasicSearchThread"); Sleep(100); PROTOSEARCHRESULT psr = { 0 }; psr.cbSize = sizeof(psr); psr.flags = PSR_UNICODE; psr.nick.w = jsb->jid; psr.firstName.w = L""; psr.lastName.w = L""; psr.id.w = jsb->jid; ProtoBroadcastAck(0, ACKTYPE_SEARCH, ACKRESULT_DATA, (HANDLE)jsb->hSearch, (LPARAM)&psr); ProtoBroadcastAck(0, ACKTYPE_SEARCH, ACKRESULT_SUCCESS, (HANDLE)jsb->hSearch); mir_free(jsb); } HANDLE CJabberProto::SearchBasic(const wchar_t *szJid) { debugLogA("JabberBasicSearch called with lParam = '%s'", szJid); JABBER_SEARCH_BASIC *jsb; if (!m_bJabberOnline || (jsb = (JABBER_SEARCH_BASIC*)mir_alloc(sizeof(JABBER_SEARCH_BASIC))) == nullptr) return nullptr; if (wcschr(szJid, '@') == nullptr) { wchar_t *szServer = mir_a2u(m_ThreadInfo->conn.server); const wchar_t *p = wcsstr(szJid, szServer); if (p == nullptr) { bool numericjid = true; for (const wchar_t *i = szJid; *i && numericjid; i++) numericjid = (*i >= '0') && (*i <= '9'); mir_free(szServer); szServer = getWStringA(0, "LoginServer"); if (szServer == nullptr) szServer = mir_wstrdup(L"jabber.org"); else if (numericjid && !mir_wstrcmpi(szServer, L"S.ms")) { mir_free(szServer); szServer = mir_wstrdup(L"sms"); } mir_snwprintf(jsb->jid, L"%s@%s", szJid, szServer); } else wcsncpy_s(jsb->jid, szJid, _TRUNCATE); mir_free(szServer); } else wcsncpy_s(jsb->jid, szJid, _TRUNCATE); debugLogA("Adding '%s' without validation", jsb->jid); jsb->hSearch = SerialNext(); ForkThread((MyThreadFunc)&CJabberProto::BasicSearchThread, jsb); return (HANDLE)jsb->hSearch; } //////////////////////////////////////////////////////////////////////////////////////// // SearchByEmail - searches the contact by its e-mail HANDLE CJabberProto::SearchByEmail(const wchar_t *email) { ptrA szServerName(getStringA("Jud")); if (!m_bJabberOnline || email == nullptr || szServerName == nullptr) return nullptr; CJabberIqInfo *pInfo = AddIQ(&CJabberProto::OnIqResultSetSearch, JABBER_IQ_TYPE_SET, szServerName); m_ThreadInfo->send(XmlNodeIq(pInfo) << XQUERY(JABBER_FEAT_JUD) << XCHILD("email", T2Utf(email))); return (HANDLE)pInfo->GetIqId(); } //////////////////////////////////////////////////////////////////////////////////////// // JabberSearchByName - searches the contact by its first or last name, or by a nickname HANDLE CJabberProto::SearchByName(const wchar_t *nick, const wchar_t *firstName, const wchar_t *lastName) { ptrA szServerName(getStringA("Jud")); if (!m_bJabberOnline || szServerName == nullptr) return nullptr; CJabberIqInfo *pInfo = AddIQ(&CJabberProto::OnIqResultExtSearch, JABBER_IQ_TYPE_SET, szServerName); XmlNodeIq iq(pInfo); TiXmlElement *query = iq << XQUERY(JABBER_FEAT_JUD); if (m_tszSelectedLang) iq << XATTR("xml:lang", m_tszSelectedLang); TiXmlElement *x = query << XCHILDNS("x", JABBER_FEAT_DATA_FORMS) << XATTR("type", "submit"); if (nick[0] != '\0') x << XCHILD("field") << XATTR("var", "user") << XATTR("value", T2Utf(nick)); if (firstName[0] != '\0') x << XCHILD("field") << XATTR("var", "fn") << XATTR("value", T2Utf(firstName)); if (lastName[0] != '\0') x << XCHILD("field") << XATTR("var", "given") << XATTR("value", T2Utf(lastName)); m_ThreadInfo->send(iq); return (HANDLE)pInfo->GetIqId(); } MEVENT CJabberProto::RecvFile(MCONTACT hContact, PROTORECVFILE *pre) { MEVENT hEvent = CSuper::RecvFile(hContact, pre); if (hEvent) { auto *ft = (filetransfer *)pre->lParam; if (ft && ft->type == FT_HTTP && ft->httpPath) { DBVARIANT dbv = { DBVT_UTF8 }; dbv.pszVal = ft->httpPath; db_event_setJson(hEvent, "u", &dbv); } } return hEvent; } //////////////////////////////////////////////////////////////////////////////////////// // SendContacts int CJabberProto::SendContacts(MCONTACT hContact, int, int nContacts, MCONTACT *hContactsList) { if (!m_bJabberOnline) return 0; char szClientJid[JABBER_MAX_JID_LEN]; if (!GetClientJID(hContact, szClientJid, _countof(szClientJid))) return 0; JabberCapsBits jcb = GetResourceCapabilities(szClientJid); if (~jcb & JABBER_CAPS_ROSTER_EXCHANGE) return 0; XmlNode m("message"); TiXmlElement *x = m << XCHILDNS("x", JABBER_FEAT_ROSTER_EXCHANGE); for (int i = 0; i < nContacts; i++) { ptrA jid(getUStringA(hContactsList[i], "jid")); if (jid != nullptr) x << XCHILD("item") << XATTR("action", "add") << XATTR("jid", jid); } m << XATTR("to", szClientJid) << XATTRID(SerialNext()); m_ThreadInfo->send(m); return 1; } //////////////////////////////////////////////////////////////////////////////////////// // SendFile - sends a file HANDLE CJabberProto::SendFile(MCONTACT hContact, const wchar_t *szDescription, wchar_t** ppszFiles) { if (!m_bJabberOnline) return nullptr; ptrA jid(ContactToJID(hContact)); if (jid == nullptr) { debugLogA("invalid contact passed for a file transfer: %d", hContact); return nullptr; } JABBER_LIST_ITEM *item = ListGetItemPtr(LIST_ROSTER, jid); if (item == nullptr) if ((item = ListGetItemPtr(LIST_CHATROOM, jid)) == nullptr) { debugLogA("invalid jid passed for a file transfer: %s", jid); return nullptr; } // Check if another file transfer session request is pending (waiting for disco result) if (item->ft != nullptr) { debugLogA("another file transfer is active for %s, exiting", item->jid); return nullptr; } filetransfer *ft = new filetransfer(this, item); ft->std.hContact = hContact; while (ppszFiles[ft->std.totalFiles] != nullptr) ft->std.totalFiles++; ft->std.pszFiles.w = (wchar_t**)mir_calloc(sizeof(wchar_t*)* ft->std.totalFiles); ft->fileSize = (unsigned __int64*)mir_calloc(sizeof(unsigned __int64)* ft->std.totalFiles); int i, j; for (i = j = 0; i < ft->std.totalFiles; i++) { struct _stat64 statbuf; if (_wstat64(ppszFiles[i], &statbuf)) debugLogW(L"'%s' is an invalid filename", ppszFiles[i]); else { ft->std.pszFiles.w[j] = mir_wstrdup(ppszFiles[i]); ft->fileSize[j] = statbuf.st_size; j++; ft->std.totalBytes += statbuf.st_size; } } if (j == 0) { debugLogA("no valid files to send, exiting"); delete ft; return nullptr; } ft->std.szCurrentFile.w = mir_wstrdup(ppszFiles[0]); ft->szDescription = mir_wstrdup(szDescription); ft->jid = mir_strdup(jid); FtInitiate(ft); return ft; } //////////////////////////////////////////////////////////////////////////////////////// // JabberSendMessage - sends a message int CJabberProto::SendMsg(MCONTACT hContact, int /*flags*/, const char *pszSrc) { XmlNode m("message"); return SendMsgEx(hContact, pszSrc, m); } int CJabberProto::SendMsgEx(MCONTACT hContact, const char *pszSrc, XmlNode &m) { char szClientJid[JABBER_MAX_JID_LEN]; if (!m_bJabberOnline || !GetClientJID(hContact, szClientJid, _countof(szClientJid))) { ProtoBroadcastAsync(hContact, ACKTYPE_MESSAGE, ACKRESULT_FAILED, 0, (LPARAM)TranslateT("Protocol is offline or no JID")); return -1; } if (m_bUseOMEMO && OmemoIsEnabled(hContact)) { if (!OmemoCheckSession(hContact)) { OmemoPutMessageToOutgoingQueue(hContact, pszSrc); int id = SerialNext(); ProtoBroadcastAsync(hContact, ACKTYPE_MESSAGE, ACKRESULT_SUCCESS, (HANDLE)id); return id; } } CMStringA szBody; int isEncrypted, id = SerialNext(); if (!strncmp(pszSrc, PGP_PROLOG, mir_strlen(PGP_PROLOG))) { const char *szEnd = strstr(pszSrc, PGP_EPILOG); size_t nStrippedLength = mir_strlen(pszSrc) - mir_strlen(PGP_PROLOG) - (szEnd ? mir_strlen(szEnd) : 0) + 1; szBody.Append(pszSrc + mir_strlen(PGP_PROLOG), (int)nStrippedLength); szBody.Replace("\r\n", ""); pszSrc = szBody; isEncrypted = 1; } else isEncrypted = 0; char *msgType; if (ListGetItemPtr(LIST_CHATROOM, szClientJid) && strchr(szClientJid, '/') == nullptr) msgType = "groupchat"; else msgType = "chat"; // omemo enabled in options, omemo enabled for contact if (m_bUseOMEMO && OmemoIsEnabled(hContact) && !mir_strcmp(msgType, "chat")) { if (!OmemoEncryptMessage(m, pszSrc, hContact)) { ProtoBroadcastAsync(hContact, ACKTYPE_MESSAGE, ACKRESULT_FAILED, 0, (LPARAM)TranslateT("No valid OMEMO session exists")); return 0; } } else { XmlAddAttr(m, "type", msgType); if (!isEncrypted) m << XCHILD("body", pszSrc); else { m << XCHILD("body", "[This message is encrypted.]"); m << XCHILD("x", pszSrc) << XATTR("xmlns", "jabber:x:encrypted"); } } JabberCapsBits jcb = GetResourceCapabilities(szClientJid); if (jcb & JABBER_RESOURCE_CAPS_ERROR) jcb = JABBER_RESOURCE_CAPS_NONE; if (m_bEnableChatStates && (jcb & JABBER_CAPS_CHATSTATES)) m << XCHILDNS("active", JABBER_FEAT_CHATSTATES); m << XATTR("to", szClientJid); bool bSendReceipt = IsSendAck(hContact) && (jcb & (JABBER_CAPS_CHAT_MARKERS | JABBER_CAPS_MESSAGE_RECEIPTS)); if (bSendReceipt) { m << XCHILDNS("request", JABBER_FEAT_MESSAGE_RECEIPTS); m << XCHILDNS("markable", JABBER_FEAT_CHAT_MARKERS); } if ( // if message delivery check disabled by entity caps manager (jcb & JABBER_CAPS_MESSAGE_EVENTS_NO_DELIVERY) || // if client knows nothing about delivery !(jcb & JABBER_CAPS_MESSAGE_RECEIPTS) || // if message sent to groupchat !mir_strcmp(msgType, "groupchat") || // if message delivery check disabled in settings !bSendReceipt) { if (mir_strcmp(msgType, "groupchat")) XmlAddAttrID(m, id); m_ThreadInfo->send(m); ProtoBroadcastAsync(hContact, ACKTYPE_MESSAGE, ACKRESULT_SUCCESS, (HANDLE)id); } else { XmlAddAttrID(m, id); m_ThreadInfo->send(m); } return id; } //////////////////////////////////////////////////////////////////////////////////////// // JabberSetApparentMode - sets the visibility status int CJabberProto::SetApparentMode(MCONTACT hContact, int mode) { if (mode != 0 && mode != ID_STATUS_ONLINE && mode != ID_STATUS_OFFLINE) return 1; int oldMode = getWord(hContact, "ApparentMode", 0); if (mode == oldMode) return 1; setWord(hContact, "ApparentMode", (uint16_t)mode); if (!m_bJabberOnline) return 0; ptrA jid(getUStringA(hContact, "jid")); if (jid == nullptr) return 0; switch (mode) { case ID_STATUS_ONLINE: if (m_iStatus == ID_STATUS_INVISIBLE || oldMode == ID_STATUS_OFFLINE) m_ThreadInfo->send(XmlNode("presence") << XATTR("to", jid)); break; case ID_STATUS_OFFLINE: if (m_iStatus != ID_STATUS_INVISIBLE || oldMode == ID_STATUS_ONLINE) SendPresenceTo(ID_STATUS_INVISIBLE, jid); break; case 0: if (oldMode == ID_STATUS_ONLINE && m_iStatus == ID_STATUS_INVISIBLE) SendPresenceTo(ID_STATUS_INVISIBLE, jid); else if (oldMode == ID_STATUS_OFFLINE && m_iStatus != ID_STATUS_INVISIBLE) SendPresenceTo(m_iStatus, jid); break; } // TODO: update the zebra list (jabber:iq:privacy) return 0; } //////////////////////////////////////////////////////////////////////////////////////// // JabberSetStatus - sets the protocol status int CJabberProto::SetStatus(int iNewStatus) { debugLogA("PS_SETSTATUS(%d,%d) => %d", m_iStatus, m_iDesiredStatus, iNewStatus); if (m_iDesiredStatus == iNewStatus) return 0; int oldStatus = m_iStatus; m_iDesiredStatus = iNewStatus; if (iNewStatus == ID_STATUS_OFFLINE) { if (m_ThreadInfo) { if (m_bEnableStreamMgmt) m_StrmMgmt.SendAck(); m_ThreadInfo->send(""); m_ThreadInfo->shutdown(); } m_StrmMgmt.ResetState(); m_iDesiredStatus = ID_STATUS_OFFLINE; } else if (!m_ThreadInfo && !IsStatusConnecting(m_iStatus)) { m_iStatus = ID_STATUS_CONNECTING; ProtoBroadcastAck(0, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)oldStatus, m_iStatus); ForkThread((MyThreadFunc)&CJabberProto::ServerThread); } else if (m_bJabberOnline) SetServerStatus(iNewStatus); else { // race condition: old thread is dying, but we're already offline. report failure m_iDesiredStatus = oldStatus; ProtoBroadcastAck(0, ACKTYPE_STATUS, ACKRESULT_FAILED, (HANDLE)oldStatus, m_iStatus); } return 0; } //////////////////////////////////////////////////////////////////////////////////////// // JabberGetAwayMsg - returns a contact's away message void __cdecl CJabberProto::GetAwayMsgThread(void *param) { Thread_SetName("Jabber: GetAwayMsgThread"); MCONTACT hContact = (DWORD_PTR)param; ptrA jid(getUStringA(hContact, "jid")); if (jid != nullptr) { JABBER_LIST_ITEM *item = ListGetItemPtr(LIST_ROSTER, jid); if (item != nullptr) { if (item->arResources.getCount() > 0) { int nRes = 0; CMStringA str, strLast; for (auto &r : item->arResources) if (r->m_szStatusMessage) { nRes++; strLast = r->m_szStatusMessage; str.AppendFormat("(%s): %s\r\n", r->m_szResourceName, r->m_szStatusMessage); } if (nRes == 1) str = strLast; str.TrimRight(); ProtoBroadcastAck(hContact, ACKTYPE_AWAYMSG, ACKRESULT_SUCCESS, (HANDLE)1, Utf2T(str)); return; } Utf2T tszStatusMsg(item->getTemp()->m_szStatusMessage); if (tszStatusMsg != nullptr) { ProtoBroadcastAck(hContact, ACKTYPE_AWAYMSG, ACKRESULT_SUCCESS, (HANDLE)1, tszStatusMsg); return; } } } ProtoBroadcastAck(hContact, ACKTYPE_AWAYMSG, ACKRESULT_SUCCESS, (HANDLE)1); } HANDLE CJabberProto::GetAwayMsg(MCONTACT hContact) { ForkThread(&CJabberProto::GetAwayMsgThread, (void*)hContact); return (HANDLE)1; } //////////////////////////////////////////////////////////////////////////////////////// // JabberSetAwayMsg - sets the away status message int CJabberProto::SetAwayMsg(int status, const wchar_t *msg) { char **szMsg; mir_cslockfull lck(m_csModeMsgMutex); switch (status) { case ID_STATUS_ONLINE: szMsg = &m_modeMsgs.szOnline; break; case ID_STATUS_AWAY: szMsg = &m_modeMsgs.szAway; status = ID_STATUS_AWAY; break; case ID_STATUS_NA: szMsg = &m_modeMsgs.szNa; break; case ID_STATUS_DND: case ID_STATUS_OCCUPIED: szMsg = &m_modeMsgs.szDnd; status = ID_STATUS_DND; break; case ID_STATUS_FREECHAT: szMsg = &m_modeMsgs.szFreechat; break; default: return 1; } T2Utf szNewMsg(msg); if ((*szMsg == nullptr && msg == nullptr) || (*szMsg != nullptr && msg != nullptr && !mir_strcmp(*szMsg, szNewMsg))) return 0; // Message is the same, no update needed // Update with the new mode message replaceStr(*szMsg, szNewMsg); // Send a presence update if needed lck.unlock(); if (status == m_iStatus) SendPresence(m_iStatus, true); return 0; } ///////////////////////////////////////////////////////////////////////////////////////// // JabberUserIsTyping - sends a UTN notification int CJabberProto::UserIsTyping(MCONTACT hContact, int type) { if (!m_bJabberOnline) return 0; char szClientJid[JABBER_MAX_JID_LEN]; if (!GetClientJID(hContact, szClientJid, _countof(szClientJid))) return 0; JABBER_LIST_ITEM *item = ListGetItemPtr(LIST_ROSTER, szClientJid); if (item == nullptr) return 0; JabberCapsBits jcb = GetResourceCapabilities(szClientJid); if (jcb & JABBER_RESOURCE_CAPS_ERROR) jcb = JABBER_RESOURCE_CAPS_NONE; XmlNode m("message"); XmlAddAttr(m, "to", szClientJid); if (m_bEnableChatStates && (jcb & JABBER_CAPS_CHATSTATES)) { m << XATTR("type", "chat") << XATTRID(SerialNext()); switch (type) { case PROTOTYPE_SELFTYPING_OFF: m << XCHILDNS("paused", JABBER_FEAT_CHATSTATES); m_ThreadInfo->send(m); break; case PROTOTYPE_SELFTYPING_ON: m << XCHILDNS("composing", JABBER_FEAT_CHATSTATES); m_ThreadInfo->send(m); break; } } return 0; }