839 lines
29 KiB
SourcePawn
839 lines
29 KiB
SourcePawn
#pragma dynamic 4194304 // Increases stack space to 4mb, needed for encryption
|
|
|
|
#include <sourcemod>
|
|
#include <regex>
|
|
|
|
// Core includes
|
|
#include "steamcore/bigint.sp"
|
|
#include "steamcore/rsa.sp"
|
|
|
|
#define AUTOLOAD_EXTENSIONS
|
|
#define REQUIRE_EXTENSIONS
|
|
#include <SteamWorks>
|
|
|
|
#define PLUGIN_URL ""
|
|
#define PLUGIN_VERSION "1.7.1"
|
|
#define PLUGIN_NAME "SteamCore"
|
|
#define PLUGIN_AUTHOR "Statik"
|
|
|
|
public Plugin:myinfo =
|
|
{
|
|
name = PLUGIN_NAME,
|
|
author = PLUGIN_AUTHOR,
|
|
description = "Sourcemod natives to interact with Steam functions.",
|
|
version = PLUGIN_VERSION,
|
|
url = PLUGIN_URL
|
|
}
|
|
|
|
new bool:DEBUG = false;
|
|
|
|
new const TIMER_UPDATE_TIME = 6;
|
|
new const TOKEN_LIFETIME = 50;
|
|
new const Float:TIMEOUT_TIME = 10.0;
|
|
|
|
new Handle:cvarUsername;
|
|
new Handle:cvarPassword;
|
|
new Handle:cvarDebug;
|
|
|
|
new String:username[32] = "";
|
|
new String:passphrase[32] = "";
|
|
new String:sessionToken[32] = "";
|
|
new String:sessionCookie[256] = "";
|
|
new bool:isLogged = false;
|
|
new bool:isBusy = false;
|
|
new Handle:request;
|
|
|
|
new caller;
|
|
|
|
new timeSinceLastLogin;
|
|
new Handle:hTimeIncreaser;
|
|
|
|
new Handle:timeoutTimer;
|
|
new bool:connectionInterrupted;
|
|
|
|
new Handle:callbackHandle;
|
|
new Handle:callbackPlugin;
|
|
new Function:callbackFunction;
|
|
new Handle:finalRequest;
|
|
new SteamWorksHTTPRequestCompleted:finalFunction;
|
|
|
|
// ===================================================================================
|
|
// ===================================================================================
|
|
|
|
public APLRes:AskPluginLoad2(Handle:plugin, bool:late, String:error[], err_max)
|
|
{
|
|
// Native creation
|
|
CreateNative("IsSteamCoreBusy", nativeIsSteamCoreBusy);
|
|
CreateNative("SteamGroupAnnouncement", nativeGroupAnnouncement);
|
|
CreateNative("SteamGroupInvite", nativeGroupInvite);
|
|
|
|
RegPluginLibrary("steamcore");
|
|
|
|
return APLRes_Success;
|
|
}
|
|
|
|
public OnPluginStart()
|
|
{
|
|
// Callbacks
|
|
callbackHandle = CreateForward(ET_Ignore, Param_Cell, Param_Cell, Param_Cell, Param_Cell);
|
|
|
|
// Timers
|
|
hTimeIncreaser = CreateTimer(TIMER_UPDATE_TIME*60.0, timeIncreaser, INVALID_HANDLE, TIMER_REPEAT);
|
|
|
|
// Convars
|
|
CreateConVar("steamcore_version", PLUGIN_VERSION, "SteamCore Version", FCVAR_SPONLY | FCVAR_DONTRECORD | FCVAR_NOTIFY);
|
|
cvarUsername = CreateConVar("sc_username", "", "Steam login username.", FCVAR_PROTECTED);
|
|
cvarPassword = CreateConVar("sc_password", "", "Steam login password.", FCVAR_PROTECTED);
|
|
cvarDebug = CreateConVar("sc_debug", "0", "Toggles debugging.", 0, true, 0.0, true, 1.0);
|
|
|
|
HookConVarChange(cvarUsername, OnLoginInfoChange);
|
|
HookConVarChange(cvarPassword, OnLoginInfoChange);
|
|
HookConVarChange(cvarDebug, OnDebugStatusChange);
|
|
|
|
timeSinceLastLogin = TOKEN_LIFETIME;
|
|
|
|
AutoExecConfig(true, "plugin.steamcore");
|
|
}
|
|
|
|
public OnLoginInfoChange(Handle:cvar, const String:oldVal[], const String:newVal[])
|
|
{
|
|
isLogged = false;
|
|
}
|
|
|
|
public OnDebugStatusChange(Handle:cvar, const String:oldVal[], const String:newVal[])
|
|
{
|
|
DEBUG = bool:StringToInt(newVal);
|
|
}
|
|
|
|
public Action:timeIncreaser(Handle:timer)
|
|
{
|
|
timeSinceLastLogin += TIMER_UPDATE_TIME;
|
|
PrintDebug(0, "\n============================================================================\n");
|
|
PrintDebug(0, "Time since last login: %i minutes.", timeSinceLastLogin);
|
|
if (timeSinceLastLogin >= TOKEN_LIFETIME)
|
|
{
|
|
isLogged = false;
|
|
PrintDebug(0, "Expired token lifetime (%i)", TOKEN_LIFETIME);
|
|
}
|
|
return Plugin_Continue;
|
|
}
|
|
|
|
public OnConfigsExecuted()
|
|
{
|
|
DEBUG = GetConVarBool(FindConVar("sc_debug"));
|
|
if (timeSinceLastLogin > 10)
|
|
{
|
|
PrintDebug(0, "\n============================================================================\n");
|
|
PrintDebug(0, "Logging in to keep login alive...");
|
|
startRequest(0, INVALID_HANDLE, INVALID_FUNCTION, INVALID_HANDLE, INVALID_FUNCTION); // Starts an empty login request
|
|
}
|
|
}
|
|
|
|
// ===================================================================================
|
|
// ===================================================================================
|
|
|
|
public nativeIsSteamCoreBusy(Handle:plugin, numParams)
|
|
{
|
|
return _:isBusy;
|
|
}
|
|
|
|
public nativeGroupAnnouncement(Handle:plugin, numParams)
|
|
{
|
|
decl String:title[256];
|
|
decl String:body[1024];
|
|
decl String:groupID[64];
|
|
new client = GetNativeCell(1);
|
|
GetNativeString(2, title, sizeof(title));
|
|
GetNativeString(3, body, sizeof(body));
|
|
GetNativeString(4, groupID, sizeof(groupID));
|
|
|
|
decl String:URL[128];
|
|
Format(URL, sizeof(URL), "http://steamcommunity.com/gid/%s/announcements", groupID);
|
|
|
|
PrintDebug(client, "\n============================================================================\n");
|
|
PrintDebug(client, "Preparing request to: \n%s...", URL);
|
|
PrintDebug(client, "Title: \n%s", title);
|
|
PrintDebug(client, "Body: \n%s", body);
|
|
PrintDebug(client, "Verifying login...");
|
|
|
|
new Handle:_finalRequest = SteamWorks_CreateHTTPRequest(k_EHTTPMethodPOST, URL);
|
|
SteamWorks_SetHTTPRequestHeaderValue(_finalRequest, "Cookie", sessionCookie);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "action", "post");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "sessionID", sessionToken);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "headline", title);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "body", body);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "languages[0][headline]", title);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "languages[0][body]", body);
|
|
|
|
startRequest(client, _finalRequest, cbkGroupAnnouncement, plugin, GetNativeFunction(5));
|
|
}
|
|
|
|
public nativeGroupInvite(Handle:plugin, numParams)
|
|
{
|
|
decl String:invitee[64];
|
|
decl String:groupID[64];
|
|
new client = GetNativeCell(1);
|
|
GetNativeString(2, invitee, sizeof invitee);
|
|
GetNativeString(3, groupID, sizeof groupID);
|
|
|
|
decl String:URL[] = "http://steamcommunity.com/actions/GroupInvite";
|
|
|
|
PrintDebug(client, "\n============================================================================\n");
|
|
PrintDebug(client, "Preparing request to: \n%s...", URL);
|
|
PrintDebug(client, "Invitee community ID: \n%s", invitee);
|
|
PrintDebug(client, "Group community ID: \n%s", groupID);
|
|
PrintDebug(client, "Verifying login...");
|
|
|
|
new Handle:_finalRequest = SteamWorks_CreateHTTPRequest(k_EHTTPMethodGET, URL);
|
|
|
|
SteamWorks_SetHTTPRequestHeaderValue(_finalRequest, "Accept", "*/*");
|
|
SteamWorks_SetHTTPRequestHeaderValue(_finalRequest, "Accept-Encoding", "gzip, deflate");
|
|
SteamWorks_SetHTTPRequestHeaderValue(_finalRequest, "User-Agent", "Mozilla/5.0 (Windows NT 6.3; WOW64)");
|
|
SteamWorks_SetHTTPRequestHeaderValue(_finalRequest, "Cookie", sessionCookie);
|
|
|
|
//SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "json", "1");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "type", "groupInvite");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "sessionID", sessionToken);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "group", groupID);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(_finalRequest, "invitee", invitee);
|
|
|
|
startRequest(client, _finalRequest, cbkGroupInvite, plugin, GetNativeFunction(4));
|
|
}
|
|
|
|
// ===================================================================================
|
|
// ===================================================================================
|
|
|
|
startRequest(client, Handle:_finalRequest, SteamWorksHTTPRequestCompleted:_finalFunction, Handle:_callbackPlugin, Function:_callbackFunction)
|
|
{
|
|
if (isBusy)
|
|
{
|
|
PrintDebug(client, "\n============================================================================\n");
|
|
PrintDebug(client, "Plugin is busy with other task at this time, rejecting request...");
|
|
|
|
if (_callbackFunction != INVALID_FUNCTION) // There is an actual function callback
|
|
{
|
|
new pluginIteratorNumber = GetPluginIteratorNumber(_callbackPlugin);
|
|
new Handle:datapack;
|
|
CreateDataTimer(0.1, tmrBusyCallback, datapack);
|
|
WritePackCell(datapack, client);
|
|
WritePackCell(datapack, pluginIteratorNumber);
|
|
WritePackFunction(datapack, Function:_callbackFunction);
|
|
|
|
CloseHandle(_finalRequest);
|
|
}
|
|
return;
|
|
}
|
|
isBusy = true;
|
|
connectionInterrupted = false;
|
|
|
|
caller = client;
|
|
finalRequest = _finalRequest;
|
|
finalFunction = _finalFunction;
|
|
callbackPlugin = _callbackPlugin;
|
|
callbackFunction = _callbackFunction;
|
|
|
|
PrintDebug(caller, "\n============================================================================\n");
|
|
|
|
if (callbackFunction != INVALID_FUNCTION) // There is an actual function callback
|
|
{
|
|
AddToForward(callbackHandle, callbackPlugin, callbackFunction);
|
|
|
|
if (isLogged)
|
|
{
|
|
PrintDebug(caller, "Already logged in, executing request...");
|
|
SteamWorks_SetHTTPCallbacks(finalRequest, finalFunction)
|
|
SteamWorks_SendHTTPRequest(finalRequest);
|
|
startTimeoutTimer();
|
|
return;
|
|
}
|
|
}
|
|
GetConVarString(cvarUsername, username, sizeof(username));
|
|
GetConVarString(cvarPassword, passphrase, sizeof(passphrase));
|
|
|
|
if (StrEqual(username, "") || StrEqual(passphrase, ""))
|
|
{
|
|
PrintDebug(caller, "Invalid login information, check cvars. ABORTED.");
|
|
onRequestResult(caller, false, 0x03); // Invalid login information
|
|
return;
|
|
}
|
|
|
|
request = SteamWorks_CreateHTTPRequest(k_EHTTPMethodGET, "http://steamcommunity.com/login/getrsakey/");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "username", username);
|
|
SteamWorks_SetHTTPCallbacks(request, cbkRsaKeyRequest);
|
|
SteamWorks_SendHTTPRequest(request);
|
|
startTimeoutTimer();
|
|
|
|
PrintDebug(caller, "Obtaining RSA Key from steamcommunity.com/login/getrsakey...");
|
|
}
|
|
|
|
startTimeoutTimer()
|
|
{
|
|
stopTimeoutTimer();
|
|
timeoutTimer = CreateTimer(TIMEOUT_TIME, tmrTimeout);
|
|
}
|
|
|
|
stopTimeoutTimer()
|
|
{
|
|
if (timeoutTimer != INVALID_HANDLE)
|
|
{
|
|
KillTimer(timeoutTimer);
|
|
timeoutTimer = INVALID_HANDLE;
|
|
}
|
|
}
|
|
|
|
public Action:tmrTimeout(Handle:timer)
|
|
{
|
|
PrintDebug(caller, "Connection timed out.");
|
|
connectionInterrupted = true;
|
|
onRequestResult(caller, false, 0x02);
|
|
timeoutTimer = INVALID_HANDLE;
|
|
}
|
|
|
|
public Action:tmrBusyCallback(Handle:timer, Handle:pack)
|
|
{
|
|
ResetPack(pack);
|
|
new client = ReadPackCell(pack);
|
|
new pluginIteratorNumber = ReadPackCell(pack);
|
|
new Handle:callbackPl = FindPluginFromNumber(pluginIteratorNumber);
|
|
new Function:callbackFunc = ReadPackFunction(pack);
|
|
|
|
new bool:success = RemoveFromForward(callbackHandle, callbackPlugin, callbackFunction);
|
|
new functionCount = GetForwardFunctionCount(callbackHandle);
|
|
PrintDebug(caller, "Removing main callback from forward - Result: %i, - Forward Function Count: %i", success, functionCount);
|
|
|
|
success = AddToForward(callbackHandle, callbackPl, callbackFunc);
|
|
functionCount = GetForwardFunctionCount(callbackHandle);
|
|
PrintDebug(caller, "Adding temporal callback from forward - Result: %i, - Forward Function Count: %i", success, functionCount);
|
|
|
|
// Start function call
|
|
Call_StartForward(callbackHandle);
|
|
// Push parameters one at a time
|
|
Call_PushCell(client);
|
|
Call_PushCell(false);
|
|
Call_PushCell(0x01); // Plugin is busy
|
|
Call_PushCell(0);
|
|
// Finish the call
|
|
new result = Call_Finish();
|
|
PrintDebug(caller, "Temporal callback calling error code: %i (0: Success)", result);
|
|
|
|
success = RemoveFromForward(callbackHandle, callbackPl, callbackFunc);
|
|
functionCount = GetForwardFunctionCount(callbackHandle);
|
|
PrintDebug(caller, "Removing temporal callback from forward - Result: %i, - Forward Function Count: %i", success, functionCount);
|
|
|
|
success = AddToForward(callbackHandle, callbackPlugin, callbackFunction);
|
|
functionCount = GetForwardFunctionCount(callbackHandle);
|
|
PrintDebug(caller, "Re-adding main callback from forward - Result: %i, - Forward Function Count: %i", success, functionCount);
|
|
|
|
callbackPl = INVALID_HANDLE;
|
|
callbackFunc = INVALID_FUNCTION;
|
|
|
|
PrintDebug(caller, "Task rejected.");
|
|
}
|
|
|
|
public cbkRsaKeyRequest(Handle:response, bool:failure, bool:requestSuccessful, EHTTPStatusCode:statusCode)
|
|
{
|
|
stopTimeoutTimer();
|
|
if (connectionInterrupted)
|
|
{
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
|
|
if (response == INVALID_HANDLE || !requestSuccessful || statusCode != k_EHTTPStatusCode200OK)
|
|
{
|
|
PrintDebug(caller, "RSA Key request failed (%i). Status Code: %i. ABORTED", requestSuccessful, statusCode);
|
|
onRequestResult(caller, false, 0x04); // Failed http RSA Key request
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
new bodySize;
|
|
SteamWorks_GetHTTPResponseBodySize(request, bodySize);
|
|
decl String:responseBody[bodySize];
|
|
SteamWorks_GetHTTPResponseBodyData(request, responseBody, bodySize);
|
|
PrintDebug(caller, responseBody);
|
|
|
|
if (StrContains(responseBody, "\"success\":true", false) == -1)
|
|
{
|
|
PrintDebug(caller, "Could not get RSA Key, aborting...");
|
|
onRequestResult(caller, false, 0x05); // RSA Key response failed, unknown reason
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
new Handle:regex;
|
|
regex = CompileRegex("\"publickey_mod\":\"(.*?)\"");
|
|
MatchRegex(regex, responseBody);
|
|
decl String:rsaPublicMod[1024];
|
|
GetRegexSubString(regex, 1, rsaPublicMod, sizeof(rsaPublicMod));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
PrintDebug(caller, "RSA KEY MODULUS (%i): \n%s", strlen(rsaPublicMod), rsaPublicMod);
|
|
|
|
regex = CompileRegex("\"publickey_exp\":\"(.*?)\"");
|
|
MatchRegex(regex, responseBody);
|
|
decl String:rsaPublicExp[16];
|
|
GetRegexSubString(regex, 1, rsaPublicExp, sizeof(rsaPublicExp));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
PrintDebug(caller, "RSA KEY EXPONENT (%i): %s", strlen(rsaPublicExp), rsaPublicExp);
|
|
|
|
regex = CompileRegex("\"timestamp\":\"(.*?)\"");
|
|
MatchRegex(regex, responseBody);
|
|
decl String:steamTimestamp[16];
|
|
GetRegexSubString(regex, 1, steamTimestamp, sizeof(steamTimestamp));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
PrintDebug(caller, "STEAM TIMESTAMP (%i): %s", strlen(steamTimestamp), steamTimestamp);
|
|
|
|
PrintDebug(caller, "\n============================================================================\n");
|
|
|
|
PrintDebug(caller, "Encrypting passphrase ******** with RSA public key...");
|
|
decl String:encryptedPassword[1024];
|
|
rsaEncrypt(rsaPublicMod, rsaPublicExp, passphrase, encryptedPassword, sizeof(encryptedPassword));
|
|
PrintDebug(caller, "Encrypted passphrase with RSA cryptosystem (%i): \n%s", strlen(encryptedPassword), encryptedPassword);
|
|
|
|
decl numericPassword[1024];
|
|
hexString2BigInt(encryptedPassword, numericPassword, sizeof(numericPassword));
|
|
encodeBase64(numericPassword, strlen(rsaPublicMod),encryptedPassword, sizeof(encryptedPassword));
|
|
PrintDebug(caller, "Encoded encrypted passphrase with base64 algorithm (%i): \n%s", strlen(encryptedPassword), encryptedPassword);
|
|
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
|
|
PrintDebug(caller, "\n============================================================================\n");
|
|
|
|
PrintDebug(caller, "Logging in to steamcommunity.com/login/dologin/...");
|
|
request = SteamWorks_CreateHTTPRequest(k_EHTTPMethodGET, "https://steamcommunity.com/login/dologin/");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "password", encryptedPassword);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "username", username);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "twofactorcode", "");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "emailauth", "");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "loginfriendlyname", "");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "captchagid", "");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "captcha_text", "");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "emailsteamid", "");
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "rsatimestamp", steamTimestamp);
|
|
SteamWorks_SetHTTPRequestGetOrPostParameter(request, "remember_login", "false");
|
|
SteamWorks_SetHTTPCallbacks(request, cbkLoginRequest);
|
|
SteamWorks_SendHTTPRequest(request);
|
|
startTimeoutTimer();
|
|
}
|
|
|
|
public cbkLoginRequest(Handle:response, bool:failure, bool:requestSuccessful, EHTTPStatusCode:statusCode)
|
|
{
|
|
stopTimeoutTimer();
|
|
if (connectionInterrupted)
|
|
{
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
|
|
if (response == INVALID_HANDLE || !requestSuccessful || statusCode != k_EHTTPStatusCode200OK)
|
|
{
|
|
PrintDebug(caller, "Login request failed (%i). Status Code: %i. ABORTED", requestSuccessful, statusCode);
|
|
onRequestResult(caller, false, 0x06); // Failed htpps login request
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
new bodySize;
|
|
SteamWorks_GetHTTPResponseBodySize(response, bodySize);
|
|
decl String:responseBody[bodySize];
|
|
SteamWorks_GetHTTPResponseBodyData(response, responseBody, bodySize);
|
|
|
|
new Handle:regex;
|
|
regex = CompileRegex("\"success\":(.*?),");
|
|
MatchRegex(regex, responseBody);
|
|
decl String:successString[20];
|
|
GetRegexSubString(regex, 1, successString, sizeof(successString));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
if (strcmp(successString, "true") != 0) // successString != true
|
|
{
|
|
PrintDebug(caller, "Aborted logging, incorrect response body (%i): \n%s", strlen(responseBody), responseBody);
|
|
onRequestResult(caller, false, 0x07); // Incorrect login data, required captcha or e-mail confirmation (Steam Guard)
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
new cookieSize;
|
|
SteamWorks_GetHTTPResponseHeaderSize(response, "Set-Cookie", cookieSize);
|
|
SteamWorks_GetHTTPResponseHeaderValue(response, "Set-Cookie", sessionCookie, cookieSize);
|
|
|
|
// Cleaning cookie
|
|
ReplaceString(sessionCookie, sizeof sessionCookie, "path=/,", "");
|
|
ReplaceString(sessionCookie, sizeof sessionCookie, "path=/; httponly,", "");
|
|
ReplaceString(sessionCookie, sizeof sessionCookie, "path=/; secure; httponly", "");
|
|
|
|
PrintDebug(caller, "Success, got response (%i): \n%s", strlen(responseBody), responseBody);
|
|
PrintDebug(caller, "Stored Cookie (%i): \n%s", strlen(sessionCookie), sessionCookie);
|
|
|
|
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
|
|
PrintDebug(caller, "\n============================================================================\n");
|
|
|
|
PrintDebug(caller, "Logging successful, obtaining session token...");
|
|
|
|
request = SteamWorks_CreateHTTPRequest(k_EHTTPMethodGET, "http://steamcommunity.com/profiles/RedirectToHome");
|
|
SteamWorks_SetHTTPRequestHeaderValue(request, "Cookie", sessionCookie);
|
|
SteamWorks_SetHTTPCallbacks(request, cbkTokenRequest);
|
|
SteamWorks_SendHTTPRequest(request);
|
|
startTimeoutTimer();
|
|
}
|
|
|
|
public cbkTokenRequest(Handle:response, bool:failure, bool:requestSuccessful, EHTTPStatusCode:statusCode)
|
|
{
|
|
stopTimeoutTimer();
|
|
if (connectionInterrupted)
|
|
{
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
|
|
if (response == INVALID_HANDLE || !requestSuccessful || statusCode != k_EHTTPStatusCode200OK)
|
|
{
|
|
PrintDebug(caller, "Session Token request failed (%i). Status Code: %i. ABORTED", requestSuccessful, statusCode);
|
|
onRequestResult(caller, false, 0x08); // Failed http token request
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
new bodySize;
|
|
SteamWorks_GetHTTPResponseBodySize(response, bodySize);
|
|
decl String:responseBody[bodySize];
|
|
SteamWorks_GetHTTPResponseBodyData(response, responseBody, bodySize);
|
|
|
|
new Handle:regex;
|
|
regex = CompileRegex("g_steamID = (.*?);");
|
|
MatchRegex(regex, responseBody);
|
|
decl String:steamId[20];
|
|
GetRegexSubString(regex, 1, steamId, sizeof(steamId));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
regex = CompileRegex("g_sessionID = \"(.*?)\"");
|
|
MatchRegex(regex, responseBody);
|
|
GetRegexSubString(regex, 1, sessionToken, sizeof(sessionToken));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
if (strcmp(steamId, "false") == 0) // steamId == false
|
|
{
|
|
PrintDebug(caller, "Could not get session token. Got: \"%s\". Incorrect Cookie?", steamId);
|
|
onRequestResult(caller, false, 0x09); // Invalid session token. Incorrect cookie?
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
return;
|
|
}
|
|
isLogged = true;
|
|
|
|
// Cleaning cookie
|
|
ReplaceString(sessionCookie, sizeof sessionCookie, "path=/; httponly,", "");
|
|
ReplaceString(sessionCookie, sizeof sessionCookie, "path=/; secure; httponly", "");
|
|
|
|
Format(sessionCookie, sizeof sessionCookie, "Steam_Language=english; sessionid=%s; %s", sessionToken, sessionCookie);
|
|
|
|
PrintDebug(caller, "Session token successfully acquired (%i): %s", strlen(sessionToken), sessionToken);
|
|
PrintDebug(caller, "Current session for Steam ID (%i): %s", strlen(steamId), steamId);
|
|
PrintDebug(caller, "Appended session token to clean cookie, actual cookie (%i): \n%s", strlen(sessionCookie), sessionCookie);
|
|
|
|
if (finalRequest != INVALID_HANDLE)
|
|
{
|
|
PrintDebug(caller, "\n============================================================================\n");
|
|
|
|
PrintDebug(caller, "Executing final request...");
|
|
SteamWorks_SetHTTPCallbacks(finalRequest, finalFunction);
|
|
SteamWorks_SendHTTPRequest(finalRequest);
|
|
startTimeoutTimer();
|
|
}
|
|
else
|
|
{
|
|
PrintDebug(caller, "There is no final request, logged in successfully.");
|
|
onRequestResult(caller, true);
|
|
}
|
|
|
|
CloseHandle(request);
|
|
request = INVALID_HANDLE;
|
|
}
|
|
|
|
public cbkGroupAnnouncement(Handle:response, bool:failure, bool:requestSuccessful, EHTTPStatusCode:statusCode)
|
|
{
|
|
stopTimeoutTimer();
|
|
if (connectionInterrupted)
|
|
{
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
return;
|
|
}
|
|
|
|
if (response == INVALID_HANDLE || !requestSuccessful || statusCode != k_EHTTPStatusCode200OK)
|
|
{
|
|
PrintDebug(caller, "Group announcement request failed (%i). Status Code: %i", requestSuccessful, statusCode);
|
|
onRequestResult(caller, false, 0x10); // Failed http group announcement request
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
return;
|
|
}
|
|
new cookieSize;
|
|
SteamWorks_GetHTTPResponseHeaderSize(response, "Set-Cookie", cookieSize);
|
|
decl String:cookie[cookieSize];
|
|
SteamWorks_GetHTTPResponseHeaderValue(response, "Set-Cookie", cookie, cookieSize);
|
|
|
|
new bodySize;
|
|
SteamWorks_GetHTTPResponseBodySize(response, bodySize);
|
|
decl String:responseBody[bodySize];
|
|
SteamWorks_GetHTTPResponseBodyData(response, responseBody, bodySize);
|
|
|
|
new Handle:regex;
|
|
regex = CompileRegex("steamLogin=(.*?);");
|
|
MatchRegex(regex, cookie);
|
|
decl String:steamLogin[20];
|
|
GetRegexSubString(regex, 1, steamLogin, sizeof(steamLogin));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
regex = CompileRegex("<title>(.*?)</title>");
|
|
MatchRegex(regex, responseBody);
|
|
decl String:title[40];
|
|
GetRegexSubString(regex, 1, title, sizeof(title));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
if (strcmp(steamLogin, "deleted") == 0)
|
|
{
|
|
isLogged = false;
|
|
PrintDebug(caller, "Invalid steam login token.");
|
|
onRequestResult(caller, false, 0x11); // Invalid steam login token
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
return;
|
|
}
|
|
if (strcmp(title, "Steam Community :: Error") == 0)
|
|
{
|
|
PrintDebug(caller, "Form error on request.");
|
|
onRequestResult(caller, false, 0x12); // Form error on request
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
return;
|
|
}
|
|
|
|
onRequestResult(caller, true);
|
|
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
}
|
|
|
|
public cbkGroupInvite(Handle:response, bool:failure, bool:requestSuccessful, EHTTPStatusCode:statusCode)
|
|
{
|
|
stopTimeoutTimer();
|
|
if (connectionInterrupted)
|
|
{
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
return;
|
|
}
|
|
|
|
if (response == INVALID_HANDLE || !requestSuccessful || statusCode != k_EHTTPStatusCode200OK)
|
|
{
|
|
PrintDebug(caller, "Group invite request failed (%i). Status Code: %i", requestSuccessful, statusCode);
|
|
onRequestResult(caller, false, 0x20); // Failed http group invite request
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
return;
|
|
}
|
|
new bodySize;
|
|
SteamWorks_GetHTTPResponseBodySize(response, bodySize);
|
|
decl String:responseBody[bodySize];
|
|
SteamWorks_GetHTTPResponseBodyData(response, responseBody, bodySize);
|
|
|
|
new Handle:regex;
|
|
regex = CompileRegex("<results><!\\[CDATA\\[(.*?)\\]\\]><\\/results>", PCRE_DOTALL);
|
|
MatchRegex(regex, responseBody);
|
|
decl String:result[2048];
|
|
GetRegexSubString(regex, 1, result, sizeof(result));
|
|
CloseHandle(regex);
|
|
regex = INVALID_HANDLE;
|
|
|
|
if (!StrEqual(result, "OK"))
|
|
{
|
|
PrintDebug(caller, "Error: ");
|
|
PrintDebug(caller, result);
|
|
|
|
if (StrEqual(result, "The invitation to that player failed. Please try again.\n\nError code: 19"))
|
|
{
|
|
PrintDebug(caller, "Invite failed. Incorrect invitee id on request or another error.");
|
|
onRequestResult(caller, false, 0x21); // Incorrect invitee or another error
|
|
}
|
|
else if (StrEqual(result, "Missing Data"))
|
|
{
|
|
PrintDebug(caller, "Invite failed. Incorrect group id or missing data on request.");
|
|
onRequestResult(caller, false, 0x22); // Incorrect Group ID or missing data.
|
|
}
|
|
else if (StrEqual(result, "Missing or invalid form session key"))
|
|
{
|
|
isLogged = false;
|
|
PrintDebug(caller, "Invite failed. Plugin is not logged in. Try again to login.");
|
|
onRequestResult(caller, false, 0x23); // Logged out. Retry to login
|
|
}
|
|
else if (StrEqual(result, "You do not have permission to invite to the group specified."))
|
|
{
|
|
PrintDebug(caller, "Invite failed. Inviter account is not a member of the group or does not have permissions to invite.");
|
|
onRequestResult(caller, false, 0x24); // Account does not have permissions to invite.
|
|
}
|
|
else if (StrEqual(result, "Your account does not meet the requirements to use this feature. <a class=\"whiteLink\" target=\"_blank\" href=\"https://support.steampowered.com/kb_article.php?ref=3330-IAGK-7663\">Visit Steam Support</a> for more information."))
|
|
{
|
|
PrintDebug(caller, "Invite failed. Account is limited, only full Steam accounts can send group invites.");
|
|
onRequestResult(caller, false, 0x25); // Limited account. Only full Steam accounts can send Steam group invites
|
|
}
|
|
else
|
|
{
|
|
PrintDebug(caller, "Invite failed. Unknown error response received when sending the group invite.");
|
|
onRequestResult(caller, false, 0x26); // Unknown error
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (StrContains(responseBody, "<duplicate><![CDATA[1]]></duplicate>") != -1)
|
|
{
|
|
PrintDebug(caller, "Invite failed. Invitee has already received an invite or is already on the group.");
|
|
onRequestResult(caller, false, 0x27); // Invitee has already received an invite or is already on the group.
|
|
}
|
|
else
|
|
{
|
|
PrintDebug(caller, "Group invite sent.");
|
|
onRequestResult(caller, true); // Success
|
|
}
|
|
}
|
|
PrintDebug(caller, "Response body (%i):\n %s", strlen(responseBody), responseBody);
|
|
|
|
CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
}
|
|
|
|
onRequestResult(client, bool:success, errorCode=0, any:data=0)
|
|
{
|
|
isBusy = false;
|
|
|
|
PrintDebug(caller, "\n============================================================================\n");
|
|
|
|
PrintDebug(caller, "Final request result: %i - Error Code : %i", success, errorCode);
|
|
|
|
if (success)
|
|
{
|
|
timeSinceLastLogin = 0;
|
|
KillTimer(hTimeIncreaser);
|
|
hTimeIncreaser = CreateTimer(TIMER_UPDATE_TIME*60.0, timeIncreaser, INVALID_HANDLE, TIMER_REPEAT);
|
|
}
|
|
// In case there was an error before the last request was executed, they are freed.
|
|
else if (errorCode > 0 && errorCode <= 0x0A)
|
|
{
|
|
if (finalRequest != INVALID_HANDLE) CloseHandle(finalRequest);
|
|
finalRequest = INVALID_HANDLE;
|
|
finalFunction = INVALID_FUNCTION;
|
|
}
|
|
if (callbackFunction != INVALID_FUNCTION)
|
|
{
|
|
PrintDebug(caller, "Calling callback...");
|
|
// Start function call
|
|
Call_StartForward(callbackHandle);
|
|
// Push parameters one at a time
|
|
Call_PushCell(client); // Client
|
|
Call_PushCell(success); // Success
|
|
Call_PushCell(errorCode); // Error code
|
|
Call_PushCell(data); // Extra data, in this case nothing
|
|
// Finish the call
|
|
new result = Call_Finish();
|
|
PrintDebug(caller, "Callback calling error code: %i (0: Success)", result);
|
|
|
|
removeCallback();
|
|
}
|
|
caller = 0;
|
|
}
|
|
|
|
removeCallback()
|
|
{
|
|
new bool:removed = RemoveFromForward(callbackHandle, callbackPlugin, callbackFunction);
|
|
new functionCount = GetForwardFunctionCount(callbackHandle);
|
|
PrintDebug(caller, "Removing callback from forward - Result: %i, - Forward Function Count: %i", removed, functionCount);
|
|
callbackFunction = INVALID_FUNCTION;
|
|
}
|
|
|
|
// ===================================================================================
|
|
// ===================================================================================
|
|
|
|
// Obtains the plugin index in a plugin iterator
|
|
GetPluginIteratorNumber(Handle:plugin)
|
|
{
|
|
new pluginNumber = 0;
|
|
decl String:pluginName[256];
|
|
decl String:auxPluginName[256];
|
|
GetPluginFilename(plugin, pluginName, sizeof(pluginName));
|
|
new Handle:pluginIterator = GetPluginIterator();
|
|
while (MorePlugins(pluginIterator))
|
|
{
|
|
pluginNumber++;
|
|
GetPluginFilename(ReadPlugin(pluginIterator), auxPluginName, sizeof(auxPluginName));
|
|
if (StrEqual(pluginName, auxPluginName)) break;
|
|
}
|
|
CloseHandle(pluginIterator);
|
|
pluginIterator = INVALID_HANDLE;
|
|
|
|
return pluginNumber;
|
|
}
|
|
|
|
Handle:FindPluginFromNumber(pluginNumber)
|
|
{
|
|
new Handle:pluginIterator = GetPluginIterator();
|
|
new Handle:plugin;
|
|
for (new i = 0; i < pluginNumber; i++)
|
|
{
|
|
if (!MorePlugins(pluginIterator))
|
|
{
|
|
plugin = INVALID_HANDLE;
|
|
break;
|
|
}
|
|
plugin = ReadPlugin(pluginIterator);
|
|
}
|
|
CloseHandle(pluginIterator);
|
|
pluginIterator = INVALID_HANDLE;
|
|
|
|
return plugin;
|
|
}
|
|
|
|
// ===================================================================================
|
|
// ===================================================================================
|
|
|
|
PrintDebug(client, const String:format[], any:...)
|
|
{
|
|
if (DEBUG)
|
|
{
|
|
decl String:text[1024];
|
|
VFormat(text, sizeof(text), format, 3);
|
|
if (client == 0) PrintToServer(text);
|
|
else if (IsClientInGame(client)) PrintToConsole(client, text);
|
|
}
|
|
}
|
|
|
|
stock GetCaller()
|
|
{
|
|
return caller;
|
|
}
|