2016-02-17 16:21:42 -05:00
|
|
|
#include "STDInclude.hpp"
|
|
|
|
|
|
|
|
namespace Components
|
|
|
|
{
|
|
|
|
RCon::Container RCon::BackdoorContainer;
|
|
|
|
Utils::Cryptography::ECDSA::Key RCon::BackdoorKey;
|
|
|
|
|
2016-02-19 10:02:43 -05:00
|
|
|
std::string RCon::Password;
|
|
|
|
|
2016-02-17 16:21:42 -05:00
|
|
|
RCon::RCon()
|
|
|
|
{
|
2016-02-18 12:04:47 -05:00
|
|
|
// TODO: Maybe execute that for clients as well, when we use triangular natting.
|
|
|
|
if (!Dedicated::IsDedicated()) return;
|
|
|
|
|
2016-02-17 16:40:02 -05:00
|
|
|
// TODO: Load public key
|
|
|
|
RCon::BackdoorKey.Set("");
|
2016-02-17 16:21:42 -05:00
|
|
|
|
|
|
|
RCon::BackdoorContainer.timestamp = 0;
|
|
|
|
|
|
|
|
Network::Handle("rconRequest", [] (Network::Address address, std::string data)
|
|
|
|
{
|
|
|
|
RCon::BackdoorContainer.address = address;
|
|
|
|
RCon::BackdoorContainer.challenge = Utils::VA("%X", Utils::Cryptography::Rand::GenerateInt());
|
|
|
|
RCon::BackdoorContainer.timestamp = Game::Com_Milliseconds();
|
|
|
|
|
|
|
|
Network::SendCommand(address, "rconAuthorization", RCon::BackdoorContainer.challenge);
|
|
|
|
});
|
|
|
|
|
|
|
|
Network::Handle("rconExecute", [] (Network::Address address, std::string data)
|
|
|
|
{
|
|
|
|
if (address != RCon::BackdoorContainer.address) return; // Invalid IP
|
|
|
|
if (!RCon::BackdoorContainer.timestamp || (Game::Com_Milliseconds() - RCon::BackdoorContainer.timestamp) > (1000 * 10)) return; // Timeout
|
|
|
|
RCon::BackdoorContainer.timestamp = 0;
|
|
|
|
|
|
|
|
Proto::RCon::Command command;
|
|
|
|
command.ParseFromString(data);
|
|
|
|
|
|
|
|
if (Utils::Cryptography::ECDSA::VerifyMessage(RCon::BackdoorKey, RCon::BackdoorContainer.challenge, command.signature()))
|
|
|
|
{
|
2016-02-17 16:40:02 -05:00
|
|
|
RCon::BackdoorContainer.output.clear();
|
|
|
|
Logger::PipeOutput([] (std::string output)
|
|
|
|
{
|
|
|
|
RCon::BackdoorContainer.output.append(output);
|
|
|
|
});
|
|
|
|
|
2016-02-17 16:21:42 -05:00
|
|
|
Command::Execute(command.commands(), true);
|
2016-02-17 16:40:02 -05:00
|
|
|
|
|
|
|
Logger::PipeOutput(nullptr);
|
|
|
|
|
|
|
|
Network::SendCommand(address, "rconResponse", RCon::BackdoorContainer.output);
|
|
|
|
RCon::BackdoorContainer.output.clear();
|
2016-02-17 16:21:42 -05:00
|
|
|
}
|
|
|
|
});
|
2016-02-19 10:02:43 -05:00
|
|
|
|
|
|
|
Command::Add("rcon", [] (Command::Params params)
|
|
|
|
{
|
|
|
|
if (params.Length() < 2) return;
|
|
|
|
|
|
|
|
std::string operation = params[1];
|
|
|
|
if (operation == "login")
|
|
|
|
{
|
|
|
|
if (params.Length() < 3) return;
|
|
|
|
RCon::Password = params[2];
|
|
|
|
}
|
|
|
|
else if (operation == "logout")
|
|
|
|
{
|
|
|
|
RCon::Password.clear();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!RCon::Password.empty() && *reinterpret_cast<int*>(0xB2C540) >= 5) // Get our state
|
|
|
|
{
|
|
|
|
Network::Address target(reinterpret_cast<Game::netadr_t*>(0xA5EA44));
|
|
|
|
|
|
|
|
if (target.IsValid())
|
|
|
|
{
|
|
|
|
Network::SendCommand(target, "rcon", RCon::Password + " " + params.Join(1));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Logger::Print("You are connected to an invalid server\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Logger::Print("You need to be logged in and connected to a server!\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2016-02-17 16:21:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
RCon::~RCon()
|
|
|
|
{
|
2016-02-19 10:02:43 -05:00
|
|
|
RCon::Password.clear();
|
2016-02-17 16:21:42 -05:00
|
|
|
}
|
|
|
|
}
|