summaryrefslogtreecommitdiff
path: root/protocols/Steam/src/steam_polling.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'protocols/Steam/src/steam_polling.cpp')
-rw-r--r--protocols/Steam/src/steam_polling.cpp341
1 files changed, 144 insertions, 197 deletions
diff --git a/protocols/Steam/src/steam_polling.cpp b/protocols/Steam/src/steam_polling.cpp
index aa5a5e2c98..78ffd06c8b 100644
--- a/protocols/Steam/src/steam_polling.cpp
+++ b/protocols/Steam/src/steam_polling.cpp
@@ -2,180 +2,141 @@
#define POLLING_ERRORS_LIMIT 5
-void CSteamProto::ParsePollData(JSONNode *data)
+void CSteamProto::ParsePollData(const JSONNode &data)
{
- JSONNode *node, *item = nullptr;
-
- // FIXME: Temporary solution for receivng too many duplicated typing events; should be reworked better
- std::string typingUser;
-
std::string steamIds;
- for (size_t i = 0; i < json_size(data); i++)
+ for (const JSONNode &item : data)
{
- item = json_at(data, i);
- if (item == nullptr)
- break;
-
- node = json_get(item, "steamid_from");
- ptrA steamId(mir_u2a(ptrW(json_as_string(node))));
+ json_string steamId = item["steamid_from"].as_string();
+ time_t timestamp = _wtol(item["utc_timestamp"].as_mstring());
- node = json_get(item, "utc_timestamp");
- time_t timestamp = atol(ptrA(mir_u2a(ptrW(json_as_string(node)))));
+ MCONTACT hContact = NULL;
+ if (!IsMe(steamId.c_str()) &&
+ !(hContact = FindContact(steamId.c_str())))
+ // probably this is info about random player playing on same server, so we ignore it
+ continue;
- node = json_get(item, "type");
- ptrW type(json_as_string(node));
- if (!lstrcmpi(type, L"saytext") || !lstrcmpi(type, L"emote") ||
- !lstrcmpi(type, L"my_saytext") || !lstrcmpi(type, L"my_emote"))
+ json_string type = item["type"].as_string();
+ if (!mir_strcmpi(type.c_str(), "my_saytext") ||
+ !mir_strcmpi(type.c_str(), "my_emote"))
{
- MCONTACT hContact = FindContact(steamId);
- if (!hContact)
- continue;
+ json_string text = item["text"].as_string();
- node = json_get(item, "text");
- ptrW text(json_as_string(node));
- T2Utf szMessage(text);
+ PROTORECVEVENT recv = { 0 };
+ recv.timestamp = timestamp;
+ recv.szMessage = (char*)text.c_str();
+ recv.flags = PREF_SENT;
+ Proto_RecvMessage(hContact, &recv);
+ }
+ else if (!mir_strcmpi(type.c_str(), "saytext") ||
+ !mir_strcmpi(type.c_str(), "emote"))
+ {
+ json_string text = item["text"].as_string();
PROTORECVEVENT recv = { 0 };
recv.timestamp = timestamp;
- recv.szMessage = szMessage;
- if (wcsstr(type, L"my_") == nullptr)
- {
- ProtoChainRecvMsg(hContact, &recv);
- }
- else
- {
- recv.flags = PREF_SENT;
- Proto_RecvMessage(hContact, &recv);
- }
+ recv.szMessage = (char*)text.c_str();
+ ProtoChainRecvMsg(hContact, &recv);
+
+ CallService(MS_PROTO_CONTACTISTYPING, hContact, (LPARAM)PROTOTYPE_CONTACTTYPING_OFF);
+ m_typingTimestamps[steamId] = 0;
}
- else if (!lstrcmpi(type, L"typing"))
+ else if (!mir_strcmpi(type.c_str(), "typing") && hContact)
{
- // FIXME: Temporary solution for receivng too many duplicated typing events; should be reworked better
- if (typingUser == (char*)steamId)
- continue;
- typingUser = steamId;
-
- MCONTACT hContact = FindContact(steamId);
- if (hContact)
+ auto it = m_typingTimestamps.find(steamId);
+ if (it != m_typingTimestamps.end())
{
- CallService(MS_PROTO_CONTACTISTYPING, hContact, (LPARAM)STEAM_TYPING_TIME);
+ if ((timestamp - it->second) < STEAM_TYPING_TIME)
+ continue;
}
+ CallService(MS_PROTO_CONTACTISTYPING, hContact, (LPARAM)STEAM_TYPING_TIME);
+ m_typingTimestamps[steamId] = timestamp;
}
- else if (!lstrcmpi(type, L"personastate"))
+ else if (!mir_strcmpi(type.c_str(), "personastate"))
{
- node = json_get(item, "persona_state");
- int status = node ? SteamToMirandaStatus(json_as_int(node)) : -1;
+ JSONNode node = item["persona_state"];
+ int status = !node.isnull()
+ ? SteamToMirandaStatus(node.as_int())
+ : -1;
- if (IsMe(steamId))
+ if (IsMe(steamId.c_str()))
{
- node = json_get(item, "persona_name");
- setWString("Nick", ptrW(json_as_string(node)));
-
if (status == -1 || status == ID_STATUS_OFFLINE)
continue;
-
- if (status != m_iStatus)
- {
- debugLogW(L"CSteamProto::ParsePollData: Change own status to %i", status);
- int oldStatus = m_iStatus;
- m_iStatus = m_iDesiredStatus = status;
- ProtoBroadcastAck(NULL, ACKTYPE_STATUS, ACKRESULT_SUCCESS, (HANDLE)oldStatus, m_iStatus);
- }
-
+ SetStatus(status);
continue;
}
- MCONTACT hContact = FindContact(steamId);
- if (hContact == NULL)
- continue; // probably this is info about random player playing on same server, so we ignore it
-
if (status != -1)
SetContactStatus(hContact, status);
- node = json_get(item, "persona_name");
- setWString(hContact, "Nick", ptrW(json_as_string(node)));
-
// todo: find difference between state changing and info changing
steamIds.append(steamId).append(",");
}
- else if (!lstrcmpi(type, L"personarelationship"))
+ else if (!mir_strcmpi(type.c_str(), "personarelationship"))
{
- node = json_get(item, "persona_state");
- int state = json_as_int(node);
-
+ int state = item["persona_state"].as_int();
switch (state)
{
case 0:
{// removed
- MCONTACT hContact = FindContact(steamId);
+ MCONTACT hContact = FindContact(steamId.c_str());
if (hContact)
- {
ContactIsRemoved(hContact);
- }
}
break;
case 1:
{// ignored
- MCONTACT hContact = FindContact(steamId);
+ MCONTACT hContact = FindContact(steamId.c_str());
if (hContact)
- {
ContactIsIgnored(hContact);
- }
}
break;
case 2:
{// auth request
- MCONTACT hContact = FindContact(steamId);
+ MCONTACT hContact = FindContact(steamId.c_str());
if (hContact)
- {
ContactIsAskingAuth(hContact);
- }
else
{
// load info about this user from server
ptrA token(getStringA("TokenSecret"));
PushRequest(
- new GetUserSummariesRequest(token, steamId),
- &CSteamProto::OnAuthRequested,
- mir_strdup(steamId),
- MirFreeArg);
+ new GetUserSummariesRequest(token, steamId.c_str()),
+ &CSteamProto::OnAuthRequested);
}
}
break;
case 3:
- // add to list
- // todo
+ // todo: add to list
break;
- default: continue;
+ default:
+ continue;
}
}
- else if (!lstrcmpi(type, L"leftconversation"))
+ else if (!mir_strcmpi(type.c_str(), "leftconversation") && hContact)
{
if (!getBool("ShowChatEvents", true))
continue;
- // chatstates gone event
- MCONTACT hContact = FindContact(steamId);
- if (hContact)
- {
- BYTE bEventType = STEAM_DB_EVENT_CHATSTATES_GONE;
- DBEVENTINFO dbei = {};
- dbei.pBlob = &bEventType;
- dbei.cbBlob = 1;
- dbei.eventType = EVENTTYPE_STEAM_CHATSTATES;
- dbei.flags = DBEF_READ;
- dbei.timestamp = time(nullptr);
- dbei.szModule = m_szModuleName;
- db_event_add(hContact, &dbei);
- }
+ BYTE bEventType = STEAM_DB_EVENT_CHATSTATES_GONE;
+ DBEVENTINFO dbei = {};
+ dbei.pBlob = &bEventType;
+ dbei.cbBlob = 1;
+ dbei.eventType = EVENTTYPE_STEAM_CHATSTATES;
+ dbei.flags = DBEF_READ;
+ dbei.timestamp = time(nullptr);
+ dbei.szModule = m_szModuleName;
+ db_event_add(hContact, &dbei);
}
else
{
+ debugLogA(__FUNCTION__ ": Unknown event type \"%s\"", type.c_str());
continue;
}
}
@@ -191,122 +152,108 @@ void CSteamProto::ParsePollData(JSONNode *data)
}
}
-void CSteamProto::PollingThread(void*)
+struct PollParam
{
- debugLogW(L"CSteamProto::PollingThread: entering");
+ int errors;
+ int errorsLimit;
+};
- ptrA token(getStringA("TokenSecret"));
- ptrA umqId(getStringA("UMQID"));
- UINT32 messageId = getDword("MessageID", 0);
+void CSteamProto::OnGotPoll(const HttpResponse &response, void *arg)
+{
+ PollParam *param = (PollParam*)arg;
+ if (!response.IsSuccess())
+ {
+ param->errors++;
+ return;
+ }
- int errors = 0;
- int errorsLimit = getByte("PollingErrorsLimit", POLLING_ERRORS_LIMIT);
- while (IsOnline() && errors < errorsLimit)
+ JSONNode root = JSONNode::parse(response.Content);
+ if (root.isnull())
{
- PollRequest *request = new PollRequest(token, umqId, messageId, IdleSeconds());
- // request->nlc = m_pollingConnection;
- HttpResponse *response = request->Send(m_hNetlibUser);
- delete request;
+ param->errors++;
+ return;
+ }
- if (!ResponseHttpOk(response))
- {
- errors++;
- }
- else
- {
- ptrA body((char*)mir_calloc(response->dataLength + 2));
- mir_strncpy(body, response->pData, response->dataLength + 1);
- JSONROOT root(body);
- if (root == nullptr)
- {
- errors++;
- }
- else
- {
- JSONNode *node = json_get(root, "error");
- if (node) {
- ptrW error(json_as_string(node));
+ json_string error = root["error"].as_string();
+ if (!mir_strcmpi(error.c_str(), "Timeout"))
+ {
+ // Do nothing as this is not necessarily an error
+ }
+ else if (!mir_strcmpi(error.c_str(), "OK"))
+ {
+ // Remember last message timestamp
+ time_t timestamp = _wtoi64(root["utc_timestamp"].as_mstring());
+ if (timestamp > getDword("LastMessageTS", 0))
+ setDword("LastMessageTS", timestamp);
- if (!lstrcmpi(error, L"OK"))
- {
- // Remember last message timestamp
- node = json_get(root, "utc_timestamp");
- time_t timestamp = _wtoi64(ptrW(json_as_string(node)));
- if (timestamp > getDword("LastMessageTS", 0))
- setDword("LastMessageTS", timestamp);
+ long messageId = root["messagelast"].as_int();
+ setDword("MessageID", messageId);
- node = json_get(root, "messagelast");
- messageId = json_as_int(node);
+ JSONNode data = root["messages"].as_array();
+ ParsePollData(data);
- node = json_get(root, "messages");
- JSONNode *nroot = json_as_array(node);
+ // Reset error counter only when we've got OK
+ param->errors = 0;
- if (nroot != nullptr)
- {
- ParsePollData(nroot);
- json_delete(nroot);
- }
+ // m_pollingConnection = response->nlc;
+ }
+ else if (!mir_strcmpi(error.c_str(), "Not Logged On")) // 'else' below will handle this error, we don't need this particular check right now
+ {
+ // need to relogin
+ debugLogA(__FUNCTION__ ": Not Logged On");
- // Reset error counter only when we've got OK
- errors = 0;
+ // try to reconnect only when we're actually online (during normal logout we will still got this error anyway, but in that case our status is already offline)
+ if (!IsOnline() && !Relogin())
+ {
+ // let it jump out of further processing
+ param->errors = param->errorsLimit;
+ }
+ }
+ else
+ {
+ // something wrong
+ debugLogA(__FUNCTION__ ": %s (%d)", error.c_str(), response.GetStatusCode());
- // m_pollingConnection = response->nlc;
- }
- else if (!lstrcmpi(error, L"Timeout"))
- {
- // Do nothing as this is not necessarily an error
- }
- else if (!lstrcmpi(error, L"Not Logged On")) // 'else' below will handle this error, we don't need this particular check right now
- {
- // need to relogin
- debugLogW(L"CSteamProto::PollingThread: Not Logged On");
-
- // try to reconnect only when we're actually online (during normal logout we will still got this error anyway, but in that case our status is already offline)
- if (IsOnline() && Relogin())
- {
- // load umqId and messageId again
- umqId = getStringA("UMQID"); // it's ptrA so we can assign it without fearing a memory leak
- messageId = getDword("MessageID", 0);
- }
- else
- {
- // let it jump out of further processing
- errors = errorsLimit;
- }
- }
- else
- {
- // something wrong
- debugLogW(L"CSteamProto::PollingThread: %s (%d)", error, response->resultCode);
+ // token has expired
+ if (response.GetStatusCode() == HTTP_CODE_UNAUTHORIZED)
+ delSetting("TokenSecret");
- // token has expired
- if (response->resultCode == HTTP_CODE_UNAUTHORIZED)
- delSetting("TokenSecret");
+ // too low timeout?
+ int timeout = root["sectimeout"].as_int();
+ if (timeout < STEAM_API_TIMEOUT)
+ debugLogA(__FUNCTION__ ": Timeout is too low (%d)", timeout);
- // too low timeout?
- node = json_get(root, "sectimeout");
- int timeout = json_as_int(node);
- if (timeout < STEAM_API_TIMEOUT)
- debugLogW(L"CSteamProto::PollingThread: Timeout is too low (%d)", timeout);
+ // let it jump out of further processing
+ param->errors = param->errorsLimit;
+ }
+}
- // let it jump out of further processing
- errors = errorsLimit;
- }
- }
- }
- }
+void CSteamProto::PollingThread(void*)
+{
+ debugLogA(__FUNCTION__ ": entering");
- delete response;
- }
+ ptrA token(getStringA("TokenSecret"));
- setDword("MessageID", messageId);
+ PollParam param;
+ param.errors = 0;
+ param.errorsLimit = getByte("PollingErrorsLimit", POLLING_ERRORS_LIMIT);
+ while (IsOnline() && param.errors < param.errorsLimit)
+ {
+ // request->nlc = m_pollingConnection;
+ ptrA umqId(getStringA("UMQID"));
+ UINT32 messageId = getDword("MessageID", 0);
+ SendRequest(
+ new PollRequest(token, umqId, messageId, IdleSeconds()),
+ &CSteamProto::OnGotPoll,
+ (void*)&param);
+ }
if (IsOnline())
{
- debugLogW(L"CSteamProto::PollingThread: unexpected termination; switching protocol to offline");
+ debugLogA(__FUNCTION__ ": unexpected termination; switching protocol to offline");
SetStatus(ID_STATUS_OFFLINE);
}
m_hPollingThread = nullptr;
- debugLogW(L"CSteamProto::PollingThread: leaving");
+ debugLogA(__FUNCTION__ ": leaving");
}