/*
Copyright (c) 2025 Miranda NG team (https://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 "stdafx.h"
void CTeamsProto::SendCreateEndpoint()
{
auto *pReq = new AsyncHttpRequest(REQUEST_POST, HOST_DEFAULT, "/users/ME/endpoints", &CTeamsProto::OnEndpointCreated);
pReq->flags |= NLHRF_REDIRECT;
pReq->m_szParam = "{\"endpointFeatures\":\"Agent,Presence2015,MessageProperties,CustomUserProperties,Casts,ModernBots,AutoIdleForWebApi,secureThreads,notificationStream,InviteFree,SupportsReadReceipts,ued\"}";
pReq->AddHeader("Origin", "https://web.skype.com");
pReq->AddHeader("Referer", "https://web.skype.com/");
pReq->AddAuthentication(this);
PushRequest(pReq);
}
void CTeamsProto::OnEndpointCreated(MHttpResponse *response, AsyncHttpRequest*)
{
if (IsStatusConnecting(m_iStatus))
m_iStatus++;
if (response == nullptr) {
debugLogA(__FUNCTION__ ": failed to get create endpoint");
ProtoBroadcastAck(NULL, ACKTYPE_LOGIN, ACKRESULT_FAILED, NULL, 1001);
SetStatus(ID_STATUS_OFFLINE);
return;
}
switch (response->resultCode) {
case 200:
case 201: // okay, endpoint created
break;
case 401: // unauthorized
default:
delSetting("TokenExpiresIn");
ProtoBroadcastAck(NULL, ACKTYPE_LOGIN, ACKRESULT_FAILED, NULL, 1001);
SetStatus(ID_STATUS_OFFLINE);
return;
}
// Succeeded, decode the answer
int oldStatus = m_iStatus;
m_iStatus = m_iDesiredStatus;
ProtoBroadcastAck(NULL, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)oldStatus, m_iStatus);
if (auto *hdr = response->FindHeader("Set-RegistrationToken")) {
CMStringA szValue = hdr;
int iStart = 0;
while (true) {
CMStringA szToken = szValue.Tokenize(";", iStart).Trim();
if (iStart == -1)
break;
int iStart2 = 0;
CMStringA name = szToken.Tokenize("=", iStart2);
CMStringA val = szToken.Mid(iStart2);
if (name == "registrationToken")
m_szToken = val.Detach();
else if (name == "endpointId") {
val.Replace("{", "");
val.Replace("}", "");
m_szEndpoint = val.Detach();
}
}
}
SetServerStatus(m_iDesiredStatus);
StartTrouter();
PushRequest(new CreateSubscriptionsRequest());
}
void CTeamsProto::OnEndpointDeleted(MHttpResponse *, AsyncHttpRequest *)
{
m_szEndpoint = nullptr;
m_szToken = nullptr;
}
void CTeamsProto::OnSubscriptionsCreated(MHttpResponse *response, AsyncHttpRequest*)
{
if (response == nullptr) {
debugLogA(__FUNCTION__ ": failed to create subscription");
ProtoBroadcastAck(NULL, ACKTYPE_LOGIN, ACKRESULT_FAILED, NULL, 1001);
SetStatus(ID_STATUS_OFFLINE);
return;
}
SendPresence();
}
/////////////////////////////////////////////////////////////////////////////////////////
void CTeamsProto::OnCapabilitiesSended(MHttpResponse *response, AsyncHttpRequest *)
{
if (response == nullptr || response->body.IsEmpty()) {
ProtoBroadcastAck(NULL, ACKTYPE_LOGIN, ACKRESULT_FAILED, NULL, 1001);
SetStatus(ID_STATUS_OFFLINE);
return;
}
LIST skypenames(1);
for (auto &hContact : AccContacts())
if (!isChatRoom(hContact))
skypenames.insert(getId(hContact).Detach());
PushRequest(new CreateContactsSubscriptionRequest(skypenames));
FreeList(skypenames);
skypenames.destroy();
ReceiveAvatar(0);
PushRequest(new AsyncHttpRequest(REQUEST_GET, HOST_CONTACTS, "/users/SELF/contacts", &CTeamsProto::LoadContactList));
PushRequest(new SyncConversations());
JSONNode root = JSONNode::parse(response->body);
if (root)
m_szOwnSkypeId = UrlToSkypeId(root["selfLink"].as_string().c_str()).Detach();
PushRequest(new GetProfileRequest(this, 0));
}
void CTeamsProto::SendPresence()
{
ptrA epname;
if (!m_bUseHostnameAsPlace && m_wstrPlace && *m_wstrPlace)
epname = mir_utf8encodeW(m_wstrPlace);
else {
wchar_t compName[MAX_COMPUTERNAME_LENGTH + 1];
DWORD size = _countof(compName);
GetComputerName(compName, &size);
epname = mir_utf8encodeW(compName);
}
JSONNode privateInfo; privateInfo.set_name("privateInfo");
privateInfo << CHAR_PARAM("epname", epname);
JSONNode publicInfo; publicInfo.set_name("publicInfo");
publicInfo << CHAR_PARAM("capabilities", "Audio|Video") << INT_PARAM("typ", 125)
<< CHAR_PARAM("skypeNameVersion", "Miranda NG Skype") << CHAR_PARAM("nodeInfo", "xx") << CHAR_PARAM("version", g_szMirVer);
JSONNode node;
node << CHAR_PARAM("id", "messagingService") << CHAR_PARAM("type", "EndpointPresenceDoc")
<< CHAR_PARAM("selfLink", "uri") << privateInfo << publicInfo;
auto *pReq = new AsyncHttpRequest(REQUEST_PUT, HOST_DEFAULT, "/users/ME/endpoints/" + mir_urlEncode(m_szEndpoint) + "/presenceDocs/messagingService",
&CTeamsProto::OnCapabilitiesSended);
pReq->m_szParam = node.write().c_str();
PushRequest(pReq);
}
/////////////////////////////////////////////////////////////////////////////////////////
void CTeamsProto::OnStatusChanged(MHttpResponse *response, AsyncHttpRequest*)
{
if (response == nullptr || response->resultCode != 201) {
debugLogA(__FUNCTION__ ": failed to change status");
ProtoBroadcastAck(NULL, ACKTYPE_LOGIN, ACKRESULT_FAILED, NULL, 1001);
SetStatus(ID_STATUS_OFFLINE);
return;
}
int oldStatus = m_iStatus;
m_iStatus = m_iDesiredStatus;
ProtoBroadcastAck(NULL, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)oldStatus, m_iStatus);
}
void CTeamsProto::SetServerStatus(int iStatus)
{
auto *pReq = new AsyncHttpRequest(REQUEST_PUT, HOST_PRESENCE, "/me/endpoints", &CTeamsProto::OnStatusChanged);
const char *pszAvailability, *pszActivity;
switch (iStatus) {
case ID_STATUS_OFFLINE:
pszAvailability = pszActivity = "Offline";
break;
case ID_STATUS_NA:
case ID_STATUS_AWAY:
pszAvailability = pszActivity = "Away";
break;
case ID_STATUS_DND:
pszAvailability = "DoNotDisturb";
pszActivity = "Presenting";
break;
case ID_STATUS_OCCUPIED:
pszAvailability = "Busy";
pszActivity = "InACall";
break;
default:
pszAvailability = pszActivity = "Available";
}
JSONNode node(JSON_NODE);
node << CHAR_PARAM("id", m_szEndpoint) << CHAR_PARAM("availability", pszAvailability)
<< CHAR_PARAM("activity", pszActivity) << CHAR_PARAM("activityReporting", "Transport") << CHAR_PARAM("deviceType", "Desktop");
pReq->m_szParam = node.write().c_str();
PushRequest(pReq);
}