rpcsx/rpcs3/Emu/NP/rpcn_client.cpp

1866 lines
55 KiB
C++
Raw Normal View History

2020-12-05 13:08:24 +01:00
#include "stdafx.h"
2020-08-27 21:47:04 +02:00
#include <string>
#include <mutex>
#include <thread>
#include <chrono>
#include <queue>
#include "rpcn_client.h"
#include "np_structs_extra.h"
#include "Utilities/StrUtil.h"
#include "Utilities/Thread.h"
#include "Emu/IdManager.h"
#include "Emu/System.h"
2020-11-17 02:19:17 +01:00
#include "Emu/NP/rpcn_config.h"
2020-08-27 21:47:04 +02:00
#include "util/asm.hpp"
2020-08-27 21:47:04 +02:00
#include "generated/np2_structs_generated.h"
#ifdef _WIN32
#include <winsock2.h>
#include <WS2tcpip.h>
#else
#include <errno.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netinet/tcp.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <fcntl.h>
#include <poll.h>
#include <netdb.h>
#endif
LOG_CHANNEL(rpcn_log, "rpcn");
2020-11-17 02:19:17 +01:00
// Those are defined here to avoid including sys_net.h
s32 send_packet_from_p2p_port(const std::vector<u8>& data, const sockaddr_in& addr);
std::vector<std::vector<u8>> get_rpcn_msgs();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
namespace rpcn
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
constexpr u32 RPCN_PROTOCOL_VERSION = 13;
constexpr usz RPCN_HEADER_SIZE = 13;
constexpr usz COMMUNICATION_ID_SIZE = 9;
// Constructor, destructor & singleton manager
rpcn_client::rpcn_client()
: sem_connected(0), sem_authentified(0), sem_reader(0), sem_writer(0), sem_rpcn(0),
thread_rpcn(std::thread(&rpcn_client::rpcn_thread, this)), thread_rpcn_reader(std::thread(&rpcn_client::rpcn_reader_thread, this)),
thread_rpcn_writer(std::thread(&rpcn_client::rpcn_writer_thread, this))
{
2020-08-27 21:47:04 +02:00
#ifdef _WIN32
2020-11-17 02:19:17 +01:00
WSADATA wsa_data;
WSAStartup(MAKEWORD(2, 2), &wsa_data);
2020-08-27 21:47:04 +02:00
#endif
2020-11-17 02:19:17 +01:00
g_cfg_rpcn.load();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
sem_rpcn.release();
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
rpcn_client::~rpcn_client()
{
std::lock_guard lock(inst_mutex);
terminate = true;
sem_rpcn.release();
sem_reader.release();
sem_writer.release();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
thread_rpcn.join();
thread_rpcn_reader.join();
thread_rpcn_writer.join();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
disconnect();
sem_connected.release();
sem_authentified.release();
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
std::shared_ptr<rpcn_client> rpcn_client::get_instance()
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::shared_ptr<rpcn_client> sptr;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
std::lock_guard lock(inst_mutex);
sptr = instance.lock();
if (!sptr)
{
sptr = std::shared_ptr<rpcn_client>(new rpcn_client());
instance = sptr;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return sptr;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
// inform rpcn that the server infos have been updated and signal rpcn_thread to try again
void rpcn_client::server_infos_updated()
{
if (connected)
{
disconnect();
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
sem_rpcn.release();
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
// RPCN thread
void rpcn_client::rpcn_reader_thread()
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
while (true)
{
sem_reader.acquire();
{
std::lock_guard lock(mutex_read);
while (true)
{
if (terminate)
{
return;
}
if (!handle_input())
{
break;
}
}
}
}
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
void rpcn_client::rpcn_writer_thread()
{
while (true)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
sem_writer.acquire();
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::lock_guard lock(mutex_write);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (terminate)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
return;
}
if (!handle_output())
{
break;
2020-08-27 21:47:04 +02:00
}
}
2020-11-17 02:19:17 +01:00
}
}
void rpcn_client::rpcn_thread()
{
while (true)
{
sem_rpcn.acquire();
while (true)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
if (terminate)
{
return;
}
// By default is the object is alive we should be connected
if (!connected)
{
bool result_connect;
{
std::lock_guard lock(mutex_connected);
result_connect = connect(g_cfg_rpcn.get_host());
}
sem_connected.release();
if (!result_connect)
{
break;
}
continue;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!authentified)
{
if (want_auth)
{
bool result_login;
{
std::lock_guard lock(mutex_authentified);
result_login = login(g_cfg_rpcn.get_npid(), g_cfg_rpcn.get_password(), g_cfg_rpcn.get_token());
}
sem_authentified.release();
if (!result_login)
{
break;
}
continue;
}
else
{
// Nothing to do while waiting for auth request so wait for main sem
break;
}
}
if (authentified && !Emu.IsStopped())
{
// Ping the UDP Signaling Server if we're authentified & ingame
auto now = steady_clock::now();
auto rpcn_msgs = get_rpcn_msgs();
for (const auto& msg : rpcn_msgs)
{
if (msg.size() == 6)
{
addr_sig = *utils::bless<le_t<u32>>(&msg[0]);
port_sig = *utils::bless<be_t<u16>>(&msg[4]);
last_pong_time = now;
}
else
{
rpcn_log.error("Received faulty RPCN UDP message!");
}
}
// Send a packet every 5 seconds and then every 500 ms until reply is received
if (now - last_pong_time >= 5s && now - last_ping_time > 500ms)
{
std::vector<u8> ping(9);
ping[0] = 1;
*utils::bless<le_t<s64, 1>>(&ping[1]) = user_id;
if (send_packet_from_p2p_port(ping, addr_rpcn_udp) == -1)
{
rpcn_log.error("Failed to send ping to rpcn!");
}
last_ping_time = now;
}
else
{
std::chrono::nanoseconds duration;
if ((now - last_pong_time) < 5s)
{
duration = 5s - (now - last_pong_time);
}
else
{
duration = 500ms - (now - last_ping_time);
}
sem_rpcn.try_acquire_for(duration);
}
}
}
2020-08-27 21:47:04 +02:00
}
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::handle_input()
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
u8 header[RPCN_HEADER_SIZE];
auto res_recvn = recvn(header, RPCN_HEADER_SIZE);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
switch (res_recvn)
{
case recvn_result::recvn_noconn: return error_and_disconnect("Disconnected");
case recvn_result::recvn_fatal:
case recvn_result::recvn_timeout: return error_and_disconnect("Failed to read a packet header on socket");
case recvn_result::recvn_nodata: return true;
case recvn_result::recvn_success: break;
case recvn_result::recvn_terminate: return error_and_disconnect("Recvn was forcefully aborted");
}
const u8 packet_type = header[0];
const u16 command = *utils::bless<le_t<u16>>(&header[1]);
const u16 packet_size = *utils::bless<le_t<u16>>(&header[3]);
const u64 packet_id = *utils::bless<le_t<u64>>(&header[5]);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (packet_size < RPCN_HEADER_SIZE)
return error_and_disconnect("Invalid packet size");
std::vector<u8> data;
if (packet_size > RPCN_HEADER_SIZE)
{
const u16 data_size = packet_size - RPCN_HEADER_SIZE;
data.resize(data_size);
if (recvn(data.data(), data_size) != recvn_result::recvn_success)
return error_and_disconnect("Failed to receive a whole packet");
}
switch (static_cast<PacketType>(packet_type))
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
case PacketType::Request: return error_and_disconnect("Client shouldn't receive request packets!");
case PacketType::Reply:
{
if (data.empty())
return error_and_disconnect("Reply packet without result");
// Those commands are handled synchronously and won't be forwarded to NP Handler
if (command == CommandType::Login || command == CommandType::GetServerList || command == CommandType::Create ||
command == CommandType::AddFriend || command == CommandType::RemoveFriend ||
command == CommandType::AddBlock || command == CommandType::RemoveBlock || command == CommandType::SendMessage)
{
std::lock_guard lock(mutex_replies_sync);
replies_sync.insert(std::make_pair(packet_id, std::make_pair(command, std::move(data))));
}
else
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
if (packet_id < 0x100000000)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::lock_guard lock(mutex_replies);
replies.insert(std::make_pair(static_cast<u32>(packet_id), std::make_pair(command, std::move(data))));
}
else
{
rpcn_log.error("Tried to forward a reply whose packet_id marks it as internal to rpcn");
2020-08-27 21:47:04 +02:00
}
}
2020-11-17 02:19:17 +01:00
break;
}
case PacketType::Notification:
{
switch (command)
{
case NotificationType::FriendNew:
case NotificationType::FriendLost:
case NotificationType::FriendQuery:
case NotificationType::FriendStatus:
{
handle_friend_notification(command, std::move(data));
break;
}
case NotificationType::MessageReceived:
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
handle_message(std::move(data));
break;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
default:
{
std::lock_guard lock(mutex_notifs);
notifications.emplace_back(std::make_pair(command, std::move(data)));
break;
}
}
break;
}
case PacketType::ServerInfo:
{
if (data.size() != 4)
return error_and_disconnect("Invalid size of ServerInfo packet");
received_version = reinterpret_cast<le_t<u32>&>(data[0]);
server_info_received = true;
break;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
default: return error_and_disconnect("Unknown packet received!");
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
return true;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
void rpcn_client::add_packet(const std::vector<u8> packet)
{
std::lock_guard lock(mutex_packets_to_send);
packets_to_send.push_back(std::move(packet));
sem_writer.release();
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::handle_output()
{
std::vector<std::vector<u8>> packets;
{
std::lock_guard lock(mutex_packets_to_send);
packets = std::move(packets_to_send);
packets_to_send.clear();
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
for (const auto& packet : packets)
{
if (!send_packet(packet))
{
return false;
}
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
// WolfSSL operations
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
std::string rpcn_client::get_wolfssl_error(WOLFSSL* wssl, int error) const
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
char error_string[WOLFSSL_MAX_ERROR_SZ]{};
const auto wssl_err = wolfSSL_get_error(wssl, error);
wolfSSL_ERR_error_string(wssl_err, &error_string[0]);
return std::string(error_string);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
rpcn_client::recvn_result rpcn_client::recvn(u8* buf, usz n)
{
u32 num_timeouts = 0;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
usz n_recv = 0;
while (n_recv != n)
{
if (!connected)
return recvn_result::recvn_noconn;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (terminate)
return recvn_result::recvn_terminate;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
int res = wolfSSL_read(read_wssl, reinterpret_cast<char*>(buf) + n_recv, n - n_recv);
if (res <= 0)
{
if (wolfSSL_want_read(read_wssl))
{
// If we received partially what we want try to wait longer
if (n_recv == 0)
return recvn_result::recvn_nodata;
num_timeouts++;
if (num_timeouts >= 1000)
{
rpcn_log.error("recvn timeout with %d bytes received", n_recv);
return recvn_result::recvn_timeout;
}
}
else
{
if (res == 0)
{
// Remote closed connection
return recvn_result::recvn_noconn;
}
rpcn_log.error("recvn failed with error: %d:%s", res, get_wolfssl_error(read_wssl, res));
return recvn_result::recvn_fatal;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
res = 0;
}
n_recv += res;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
return recvn_result::recvn_success;
}
bool rpcn_client::send_packet(const std::vector<u8>& packet)
{
u32 num_timeouts = 0;
usz n_sent = 0;
while (n_sent != packet.size())
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
if (!connected)
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
int res = wolfSSL_write(write_wssl, reinterpret_cast<const char*>(packet.data()), packet.size());
if (res <= 0)
{
if (wolfSSL_want_write(write_wssl))
{
num_timeouts++;
if (num_timeouts >= 1000)
{
rpcn_log.error("send_packet timeout with %d bytes sent", n_sent);
return error_and_disconnect("Failed to send all the bytes");
}
}
else
{
rpcn_log.error("send_packet failed with error: %s", get_wolfssl_error(write_wssl, res));
return error_and_disconnect("Failed to send all the bytes");
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
res = 0;
}
n_sent += res;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
return true;
}
// Helper functions
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::forge_send(u16 command, u64 packet_id, const std::vector<u8>& data)
{
// TODO: add a check for status?
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
const auto sent_packet = forge_request(command, packet_id, data);
add_packet(std::move(sent_packet));
return true;
}
bool rpcn_client::forge_send_reply(u16 command, u64 packet_id, const std::vector<u8>& data, std::vector<u8>& reply_data)
{
if (!forge_send(command, packet_id, data))
return false;
if (!get_reply(packet_id, reply_data))
return false;
if (is_error(static_cast<ErrorType>(reply_data[0])))
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
// disconnect();
2020-08-27 21:47:04 +02:00
return false;
}
2020-11-17 02:19:17 +01:00
return true;
}
// Connect & disconnect functions
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
void rpcn_client::disconnect()
{
if (read_wssl)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
wolfSSL_free(read_wssl);
read_wssl = nullptr;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
if (write_wssl)
{
wolfSSL_free(write_wssl);
write_wssl = nullptr;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (wssl_ctx)
{
wolfSSL_CTX_free(wssl_ctx);
wssl_ctx = nullptr;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
wolfSSL_Cleanup();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (sockfd)
{
2020-08-27 21:47:04 +02:00
#ifdef _WIN32
2020-11-17 02:19:17 +01:00
::closesocket(sockfd);
2020-08-27 21:47:04 +02:00
#else
2020-11-17 02:19:17 +01:00
::close(sockfd);
2020-08-27 21:47:04 +02:00
#endif
2020-11-17 02:19:17 +01:00
sockfd = 0;
}
connected = false;
authentified = false;
want_auth = false;
server_info_received = false;
}
bool rpcn_client::connect(const std::string& host)
{
rpcn_log.warning("Attempting to connect to RPCN!");
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
state = rpcn_state::failure_no_failure;
if (host.empty())
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
rpcn_log.error("RPCN host is empty!");
state = rpcn_state::failure_input;
2020-08-27 21:47:04 +02:00
return false;
}
2020-11-17 02:19:17 +01:00
auto splithost = fmt::split(host, {":"});
if (splithost.size() != 1 && splithost.size() != 2)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
rpcn_log.error("RPCN host is invalid!");
state = rpcn_state::failure_input;
2020-08-27 21:47:04 +02:00
return false;
}
2020-11-17 02:19:17 +01:00
u16 port = 31313;
if (splithost.size() == 2)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
port = std::stoul(splithost[1]);
if (port == 0)
{
rpcn_log.error("RPCN port is invalid!");
state = rpcn_state::failure_input;
return false;
}
2020-08-27 21:47:04 +02:00
}
{
2020-11-17 02:19:17 +01:00
// Ensures both read & write threads are in waiting state
std::lock_guard lock_read(mutex_read), lock_write(mutex_write);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
// Cleans previous data if any
disconnect();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (wolfSSL_Init() != WOLFSSL_SUCCESS)
{
rpcn_log.error("Failed to initialize wolfssl");
state = rpcn_state::failure_wolfssl;
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if ((wssl_ctx = wolfSSL_CTX_new(wolfTLSv1_2_client_method())) == nullptr)
{
rpcn_log.error("Failed to create wolfssl context");
state = rpcn_state::failure_wolfssl;
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
wolfSSL_CTX_set_verify(wssl_ctx, SSL_VERIFY_NONE, nullptr);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if ((read_wssl = wolfSSL_new(wssl_ctx)) == nullptr)
{
rpcn_log.error("Failed to create wolfssl object");
state = rpcn_state::failure_wolfssl;
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memset(&addr_rpcn, 0, sizeof(addr_rpcn));
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
addr_rpcn.sin_port = std::bit_cast<u16, be_t<u16>>(port); // htons
addr_rpcn.sin_family = AF_INET;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
hostent* host_addr = gethostbyname(splithost[0].c_str());
if (!host_addr)
{
rpcn_log.error("Failed to resolve %s", host);
state = rpcn_state::failure_resolve;
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
addr_rpcn.sin_addr.s_addr = *reinterpret_cast<u32*>(host_addr->h_addr_list[0]);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memcpy(&addr_rpcn_udp, &addr_rpcn, sizeof(addr_rpcn_udp));
addr_rpcn_udp.sin_port = std::bit_cast<u16, be_t<u16>>(3657); // htons
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
sockfd = socket(AF_INET, SOCK_STREAM, 0);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
#ifdef _WIN32
u32 timeout = 5;
#else
struct timeval timeout;
timeout.tv_sec = 0;
timeout.tv_usec = 50000; // 50ms
#endif
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, reinterpret_cast<char*>(&timeout), sizeof(timeout)) < 0)
{
rpcn_log.error("Failed to setsockopt!");
state = rpcn_state::failure_other;
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (::connect(sockfd, reinterpret_cast<struct sockaddr*>(&addr_rpcn), sizeof(addr_rpcn)) != 0)
{
rpcn_log.error("Failed to connect to RPCN server!");
state = rpcn_state::failure_connect;
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (wolfSSL_set_fd(read_wssl, sockfd) != WOLFSSL_SUCCESS)
{
rpcn_log.error("Failed to associate wolfssl to the socket");
state = rpcn_state::failure_wolfssl;
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
int ret_connect;
while ((ret_connect = wolfSSL_connect(read_wssl)) != SSL_SUCCESS)
{
if (wolfSSL_want_read(read_wssl))
continue;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
state = rpcn_state::failure_wolfssl;
rpcn_log.error("Handshake failed with RPCN Server: %s", get_wolfssl_error(read_wssl, ret_connect));
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if ((write_wssl = wolfSSL_write_dup(read_wssl)) == NULL)
{
rpcn_log.error("Failed to create write dup for SSL");
state = rpcn_state::failure_wolfssl;
return false;
}
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
connected = true;
// Wake up both read & write threads
sem_reader.release();
sem_writer.release();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
while (!server_info_received && connected && !terminate)
{
std::this_thread::sleep_for(5ms);
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!connected || terminate)
{
state = rpcn_state::failure_other;
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (received_version != RPCN_PROTOCOL_VERSION)
{
state = rpcn_state::failure_protocol;
rpcn_log.error("Server returned protocol version: %d, expected: %d", received_version, RPCN_PROTOCOL_VERSION);
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
last_ping_time = steady_clock::now() - 5s;
last_pong_time = last_ping_time;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::login(const std::string& npid, const std::string& password, const std::string& token)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
if (npid.empty() || password.empty())
{
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
std::copy(npid.begin(), npid.end(), std::back_inserter(data));
data.push_back(0);
std::copy(password.begin(), password.end(), std::back_inserter(data));
data.push_back(0);
std::copy(token.begin(), token.end(), std::back_inserter(data));
data.push_back(0);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
u64 req_id = rpcn_request_counter.fetch_add(1);
std::vector<u8> packet_data;
if (!forge_send_reply(CommandType::Login, req_id, data, packet_data))
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
vec_stream reply(packet_data);
auto error = static_cast<ErrorType>(reply.get<u8>());
online_name = reply.get_string(false);
avatar_url = reply.get_string(false);
user_id = reply.get<s64>();
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
auto get_usernames_and_status = [](vec_stream& stream, std::map<std::string, std::pair<bool, u64>>& friends)
{
u32 num_friends = stream.get<u32>();
for (u32 i = 0; i < num_friends; i++)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::string friend_name = stream.get_string(false);
bool online = !!(stream.get<u8>());
friends.insert(std::make_pair(std::move(friend_name), std::make_pair(online, 0)));
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
};
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
auto get_usernames = [](vec_stream& stream, std::set<std::string>& usernames)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
u32 num_usernames = stream.get<u32>();
for (u32 i = 0; i < num_usernames; i++)
{
2020-11-17 02:19:17 +01:00
std::string username = stream.get_string(false);
usernames.insert(std::move(username));
}
2020-11-17 02:19:17 +01:00
};
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
get_usernames_and_status(reply, friend_infos.friends);
get_usernames(reply, friend_infos.requests_sent);
get_usernames(reply, friend_infos.requests_received);
get_usernames(reply, friend_infos.blocked);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (is_error(error))
{
switch (error)
{
case LoginError: state = rpcn_state::failure_id; break;
case LoginAlreadyLoggedIn: state = rpcn_state::failure_id_already_logged_in; break;
case LoginInvalidUsername: state = rpcn_state::failure_id_username; break;
case LoginInvalidPassword: state = rpcn_state::failure_id_password; break;
case LoginInvalidToken: state = rpcn_state::failure_id_token; break;
default: state = rpcn_state::failure_id; break;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
disconnect();
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (reply.is_error())
return error_and_disconnect("Malformed reply to Login command");
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
rpcn_log.success("You are now logged in RPCN(%s | %s)!", npid, online_name);
authentified = true;
return true;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
ErrorType rpcn_client::create_user(const std::string& npid, const std::string& password, const std::string& online_name, const std::string& avatar_url, const std::string& email)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
std::copy(npid.begin(), npid.end(), std::back_inserter(data));
data.push_back(0);
std::copy(password.begin(), password.end(), std::back_inserter(data));
data.push_back(0);
std::copy(online_name.begin(), online_name.end(), std::back_inserter(data));
data.push_back(0);
std::copy(avatar_url.begin(), avatar_url.end(), std::back_inserter(data));
data.push_back(0);
std::copy(email.begin(), email.end(), std::back_inserter(data));
data.push_back(0);
u64 req_id = rpcn_request_counter.fetch_add(1);
std::vector<u8> packet_data;
if (!forge_send_reply(CommandType::Create, req_id, data, packet_data))
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
return ErrorType::CreationError;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
vec_stream reply(packet_data);
auto error = static_cast<ErrorType>(reply.get<u8>());
if (is_error(error))
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
return error;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
rpcn_log.success("You have successfully created a RPCN account(%s | %s)!", npid, online_name);
return ErrorType::NoError;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::add_friend(const std::string& friend_username)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
std::copy(friend_username.begin(), friend_username.end(), std::back_inserter(data));
data.push_back(0);
u64 req_id = rpcn_request_counter.fetch_add(1);
std::vector<u8> packet_data;
if (!forge_send_reply(CommandType::AddFriend, req_id, data, packet_data))
{
return false;
}
vec_stream reply(packet_data);
auto error = static_cast<ErrorType>(reply.get<u8>());
if (error == ErrorType::NotFound)
{
return false;
}
rpcn_log.success("You have successfully added \"%s\" as a friend", friend_username);
return true;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::remove_friend(const std::string& friend_username)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
std::copy(friend_username.begin(), friend_username.end(), std::back_inserter(data));
data.push_back(0);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
u64 req_id = rpcn_request_counter.fetch_add(1);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
std::vector<u8> packet_data;
if (!forge_send_reply(CommandType::RemoveFriend, req_id, data, packet_data))
{
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
vec_stream reply(packet_data);
auto error = static_cast<ErrorType>(reply.get<u8>());
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (is_error(error))
{
return false;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
rpcn_log.success("You have successfully removed \"%s\" from your friendlist", friend_username);
return true;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
std::vector<std::pair<u16, std::vector<u8>>> rpcn_client::get_notifications()
{
std::vector<std::pair<u16, std::vector<u8>>> notifs;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
{
std::lock_guard lock(mutex_notifs);
notifs = std::move(notifications);
notifications.clear();
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return notifs;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
std::unordered_map<u32, std::pair<u16, std::vector<u8>>> rpcn_client::get_replies()
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::unordered_map<u32, std::pair<u16, std::vector<u8>>> ret_replies;
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::lock_guard lock(mutex_replies);
ret_replies = std::move(replies);
replies.clear();
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
return ret_replies;
}
std::vector<u64> rpcn_client::get_new_messages()
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u64> ret_new_messages;
{
std::lock_guard lock(mutex_messages);
ret_new_messages = std::move(new_messages);
new_messages.clear();
}
2021-02-22 20:27:43 +01:00
2020-11-17 02:19:17 +01:00
return ret_new_messages;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::get_reply(const u64 expected_id, std::vector<u8>& data)
{
auto check_for_reply = [this, expected_id, &data]() -> bool
{
std::lock_guard lock(mutex_replies_sync);
if (auto r = replies_sync.find(expected_id); r != replies_sync.end())
{
data = std::move(r->second.second);
replies_sync.erase(r);
return true;
}
return false;
};
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
while (connected && !terminate)
{
if (check_for_reply())
return true;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
std::this_thread::sleep_for(5ms);
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (check_for_reply())
return true;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return false;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::get_server_list(u32 req_id, const SceNpCommunicationId& communication_id, std::vector<u16>& server_list)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data(COMMUNICATION_ID_SIZE), reply_data;
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send_reply(CommandType::GetServerList, req_id, data, reply_data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
vec_stream reply(reply_data, 1);
u16 num_servs = reply.get<u16>();
server_list.clear();
for (u16 i = 0; i < num_servs; i++)
{
server_list.push_back(reply.get<u16>());
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (reply.is_error())
{
server_list.clear();
return error_and_disconnect("Malformed reply to GetServerList command");
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::get_world_list(u32 req_id, const SceNpCommunicationId& communication_id, u16 server_id)
{
std::vector<u8> data(COMMUNICATION_ID_SIZE + sizeof(u16));
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u16>&>(data[COMMUNICATION_ID_SIZE]) = server_id;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::GetWorldList, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::createjoin_room(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2CreateJoinRoomRequest* req)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
extra_nps::print_createjoinroom(req);
flatbuffers::FlatBufferBuilder builder(1024);
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_binattrinternal_vec;
if (req->roomBinAttrInternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomBinAttrInternalNum; i++)
{
auto bin = CreateBinAttr(builder, req->roomBinAttrInternal[i].id, builder.CreateVector(req->roomBinAttrInternal[i].ptr.get_ptr(), req->roomBinAttrInternal[i].size));
davec.push_back(bin);
}
final_binattrinternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<IntAttr>>> final_searchintattrexternal_vec;
if (req->roomSearchableIntAttrExternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<IntAttr>> davec;
for (u32 i = 0; i < req->roomSearchableIntAttrExternalNum; i++)
{
auto bin = CreateIntAttr(builder, req->roomSearchableIntAttrExternal[i].id, req->roomSearchableIntAttrExternal[i].num);
davec.push_back(bin);
}
final_searchintattrexternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_searchbinattrexternal_vec;
if (req->roomSearchableBinAttrExternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomSearchableBinAttrExternalNum; i++)
{
auto bin = CreateBinAttr(builder, req->roomSearchableBinAttrExternal[i].id, builder.CreateVector(req->roomSearchableBinAttrExternal[i].ptr.get_ptr(), req->roomSearchableBinAttrExternal[i].size));
davec.push_back(bin);
}
final_searchbinattrexternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_binattrexternal_vec;
if (req->roomBinAttrExternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomBinAttrExternalNum; i++)
{
auto bin = CreateBinAttr(builder, req->roomBinAttrExternal[i].id, builder.CreateVector(req->roomBinAttrExternal[i].ptr.get_ptr(), req->roomBinAttrExternal[i].size));
davec.push_back(bin);
}
final_binattrexternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<u8>> final_roompassword;
if (req->roomPassword)
final_roompassword = builder.CreateVector(req->roomPassword->data, 8);
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<GroupConfig>>> final_groupconfigs_vec;
if (req->groupConfigNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<GroupConfig>> davec;
for (u32 i = 0; i < req->groupConfigNum; i++)
{
auto bin = CreateGroupConfig(builder, req->groupConfig[i].slotNum, req->groupConfig[i].withLabel, builder.CreateVector(req->groupConfig[i].label.data, 8), req->groupConfig[i].withPassword);
davec.push_back(bin);
}
final_groupconfigs_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<flatbuffers::String>>> final_allowedusers_vec;
if (req->allowedUserNum && req->allowedUser)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<flatbuffers::String>> davec;
for (u32 i = 0; i < req->allowedUserNum; i++)
{
auto bin = builder.CreateString(req->allowedUser[i].handle.data);
davec.push_back(bin);
}
final_allowedusers_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<flatbuffers::String>>> final_blockedusers_vec;
if (req->blockedUserNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<flatbuffers::String>> davec;
for (u32 i = 0; i < req->blockedUserNum; i++)
{
auto bin = builder.CreateString(req->blockedUser[i].handle.data);
davec.push_back(bin);
}
final_blockedusers_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<u8>> final_grouplabel;
if (req->joinRoomGroupLabel)
final_grouplabel = builder.CreateVector(req->joinRoomGroupLabel->data, 8);
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_memberbinattrinternal_vec;
if (req->roomMemberBinAttrInternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomMemberBinAttrInternalNum; i++)
{
auto bin = CreateBinAttr(
builder, req->roomMemberBinAttrInternal[i].id, builder.CreateVector(reinterpret_cast<const u8*>(req->roomMemberBinAttrInternal[i].ptr.get_ptr()), req->roomMemberBinAttrInternal[i].size));
davec.push_back(bin);
}
final_memberbinattrinternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<OptParam> final_optparam;
if (req->sigOptParam)
final_optparam = CreateOptParam(builder, req->sigOptParam->type, req->sigOptParam->flag, req->sigOptParam->hubMemberId);
u64 final_passwordSlotMask = 0;
if (req->passwordSlotMask)
final_passwordSlotMask = *req->passwordSlotMask;
auto req_finished = CreateCreateJoinRoomRequest(builder, req->worldId, req->lobbyId, req->maxSlot, req->flagAttr, final_binattrinternal_vec, final_searchintattrexternal_vec,
final_searchbinattrexternal_vec, final_binattrexternal_vec, final_roompassword, final_groupconfigs_vec, final_passwordSlotMask, final_allowedusers_vec, final_blockedusers_vec, final_grouplabel,
final_memberbinattrinternal_vec, req->teamId, final_optparam);
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + sizeof(u32) + bufsize);
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
if (!forge_send(CommandType::CreateRoom, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::join_room(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2JoinRoomRequest* req)
{
std::vector<u8> data;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
extra_nps::print_joinroom(req);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::FlatBufferBuilder builder(1024);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<u8>> final_roompassword;
if (req->roomPassword)
final_roompassword = builder.CreateVector(req->roomPassword->data, 8);
flatbuffers::Offset<flatbuffers::Vector<u8>> final_grouplabel;
if (req->joinRoomGroupLabel)
final_grouplabel = builder.CreateVector(req->joinRoomGroupLabel->data, 8);
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_memberbinattrinternal_vec;
if (req->roomMemberBinAttrInternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomMemberBinAttrInternalNum; i++)
{
auto bin = CreateBinAttr(builder, req->roomMemberBinAttrInternal[i].id, builder.CreateVector(req->roomMemberBinAttrInternal[i].ptr.get_ptr(), req->roomMemberBinAttrInternal[i].size));
davec.push_back(bin);
}
final_memberbinattrinternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<PresenceOptionData> final_optdata = CreatePresenceOptionData(builder, builder.CreateVector(req->optData.data, 16), req->optData.length);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
auto req_finished = CreateJoinRoomRequest(builder, req->roomId, final_roompassword, final_grouplabel, final_memberbinattrinternal_vec, final_optdata, req->teamId);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + sizeof(u32) + bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::JoinRoom, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::leave_room(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2LeaveRoomRequest* req)
{
std::vector<u8> data;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::FlatBufferBuilder builder(1024);
flatbuffers::Offset<PresenceOptionData> final_optdata = CreatePresenceOptionData(builder, builder.CreateVector(req->optData.data, 16), req->optData.length);
auto req_finished = CreateLeaveRoomRequest(builder, req->roomId, final_optdata);
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + sizeof(u32) + bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::LeaveRoom, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::search_room(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2SearchRoomRequest* req)
{
std::vector<u8> data;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
extra_nps::print_search_room(req);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::FlatBufferBuilder builder(1024);
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<IntSearchFilter>>> final_intfilter_vec;
if (req->intFilterNum)
{
std::vector<flatbuffers::Offset<IntSearchFilter>> davec{};
for (u32 i = 0; i < req->intFilterNum; i++)
{
auto int_attr = CreateIntAttr(builder, req->intFilter[i].attr.id, req->intFilter[i].attr.num);
auto bin = CreateIntSearchFilter(builder, req->intFilter[i].searchOperator, int_attr);
davec.push_back(bin);
}
final_intfilter_vec = builder.CreateVector(davec);
}
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinSearchFilter>>> final_binfilter_vec;
if (req->binFilterNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinSearchFilter>> davec;
for (u32 i = 0; i < req->binFilterNum; i++)
{
auto bin_attr = CreateBinAttr(builder, req->binFilter[i].attr.id, builder.CreateVector(req->binFilter[i].attr.ptr.get_ptr(), req->binFilter[i].attr.size));
auto bin = CreateBinSearchFilter(builder, req->binFilter[i].searchOperator, bin_attr);
davec.push_back(bin);
}
final_binfilter_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<u16>> attrid_vec;
if (req->attrIdNum)
attrid_vec = builder.CreateVector(utils::bless<const u16>(req->attrId.get_ptr()), req->attrIdNum);
SearchRoomRequestBuilder s_req(builder);
s_req.add_option(req->option);
s_req.add_worldId(req->worldId);
s_req.add_lobbyId(req->lobbyId);
s_req.add_rangeFilter_startIndex(req->rangeFilter.startIndex);
s_req.add_rangeFilter_max(req->rangeFilter.max);
s_req.add_flagFilter(req->flagFilter);
s_req.add_flagAttr(req->flagAttr);
if (req->intFilterNum)
s_req.add_intFilter(final_intfilter_vec);
if (req->binFilterNum)
s_req.add_binFilter(final_binfilter_vec);
if (req->attrIdNum)
s_req.add_attrId(attrid_vec);
auto req_finished = s_req.Finish();
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + bufsize + sizeof(u32));
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
if (!forge_send(CommandType::SearchRoom, req_id, data))
return false;
return true;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::get_roomdata_external_list(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2GetRoomDataExternalListRequest* req)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
flatbuffers::FlatBufferBuilder builder(1024);
std::vector<u64> roomIds;
for (u32 i = 0; i < req->roomIdNum; i++)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
roomIds.push_back(req->roomId[i]);
}
std::vector<u16> attrIds;
for (u32 i = 0; i < req->attrIdNum; i++)
{
attrIds.push_back(req->attrId[i]);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
auto req_finished = CreateGetRoomDataExternalListRequestDirect(builder, &roomIds, &attrIds);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + bufsize + sizeof(u32));
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
if (!forge_send(CommandType::GetRoomDataExternalList, req_id, data))
return false;
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::set_roomdata_external(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2SetRoomDataExternalRequest* req)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
flatbuffers::FlatBufferBuilder builder(1024);
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<IntAttr>>> final_searchintattrexternal_vec;
if (req->roomSearchableIntAttrExternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<IntAttr>> davec;
for (u32 i = 0; i < req->roomSearchableIntAttrExternalNum; i++)
{
auto bin = CreateIntAttr(builder, req->roomSearchableIntAttrExternal[i].id, req->roomSearchableIntAttrExternal[i].num);
davec.push_back(bin);
}
final_searchintattrexternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_searchbinattrexternal_vec;
if (req->roomSearchableBinAttrExternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomSearchableBinAttrExternalNum; i++)
{
auto bin = CreateBinAttr(builder, req->roomSearchableBinAttrExternal[i].id, builder.CreateVector(req->roomSearchableBinAttrExternal[i].ptr.get_ptr(), req->roomSearchableBinAttrExternal[i].size));
davec.push_back(bin);
}
final_searchbinattrexternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_binattrexternal_vec;
if (req->roomBinAttrExternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomBinAttrExternalNum; i++)
{
auto bin = CreateBinAttr(builder, req->roomBinAttrExternal[i].id, builder.CreateVector(req->roomBinAttrExternal[i].ptr.get_ptr(), req->roomBinAttrExternal[i].size));
davec.push_back(bin);
}
final_binattrexternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
auto req_finished = CreateSetRoomDataExternalRequest(builder, req->roomId, final_searchintattrexternal_vec, final_searchbinattrexternal_vec, final_binattrexternal_vec);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + bufsize + sizeof(u32));
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::SetRoomDataExternal, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::get_roomdata_internal(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2GetRoomDataInternalRequest* req)
{
std::vector<u8> data, reply_data;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::FlatBufferBuilder builder(1024);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<u16>> final_attr_ids_vec;
if (req->attrIdNum)
final_attr_ids_vec = builder.CreateVector(utils::bless<const u16>(req->attrId.get_ptr()), req->attrIdNum);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
auto req_finished = CreateGetRoomDataInternalRequest(builder, req->roomId, final_attr_ids_vec);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + bufsize + sizeof(u32));
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::GetRoomDataInternal, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::set_roomdata_internal(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2SetRoomDataInternalRequest* req)
{
std::vector<u8> data;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
// extra_nps::print_set_roomdata_req(req);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::FlatBufferBuilder builder(1024);
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<BinAttr>>> final_binattrinternal_vec;
if (req->roomBinAttrInternalNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<BinAttr>> davec;
for (u32 i = 0; i < req->roomBinAttrInternalNum; i++)
{
auto bin = CreateBinAttr(builder, req->roomBinAttrInternal[i].id, builder.CreateVector(req->roomBinAttrInternal[i].ptr.get_ptr(), req->roomBinAttrInternal[i].size));
davec.push_back(bin);
}
final_binattrinternal_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<flatbuffers::Offset<RoomGroupPasswordConfig>>> final_grouppasswordconfig_vec;
if (req->passwordConfigNum)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<RoomGroupPasswordConfig>> davec;
for (u32 i = 0; i < req->passwordConfigNum; i++)
{
auto rg = CreateRoomGroupPasswordConfig(builder, req->passwordConfig[i].groupId, req->passwordConfig[i].withPassword);
davec.push_back(rg);
}
final_grouppasswordconfig_vec = builder.CreateVector(davec);
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
u64 final_passwordSlotMask = 0;
if (req->passwordSlotMask)
final_passwordSlotMask = *req->passwordSlotMask;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
flatbuffers::Offset<flatbuffers::Vector<u16>> final_ownerprivilege_vec;
if (req->ownerPrivilegeRankNum)
final_ownerprivilege_vec = builder.CreateVector(utils::bless<const u16>(req->ownerPrivilegeRank.get_ptr()), req->ownerPrivilegeRankNum);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
auto req_finished =
CreateSetRoomDataInternalRequest(builder, req->roomId, req->flagFilter, req->flagAttr, final_binattrinternal_vec, final_grouppasswordconfig_vec, final_passwordSlotMask, final_ownerprivilege_vec);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + bufsize + sizeof(u32));
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::SetRoomDataInternal, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::ping_room_owner(u32 req_id, const SceNpCommunicationId& communication_id, u64 room_id)
{
std::vector<u8> data;
2020-11-10 09:55:49 +01:00
2020-11-17 02:19:17 +01:00
data.resize(COMMUNICATION_ID_SIZE + sizeof(u64));
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
*utils::bless<le_t<u64>>(&data[COMMUNICATION_ID_SIZE]) = room_id;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::PingRoomOwner, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::send_room_message(u32 req_id, const SceNpCommunicationId& communication_id, const SceNpMatching2SendRoomMessageRequest* req)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::vector<u8> data;
flatbuffers::FlatBufferBuilder builder(1024);
std::vector<u16> dst;
switch (req->castType)
{
2020-08-27 21:47:04 +02:00
case SCE_NP_MATCHING2_CASTTYPE_BROADCAST:
break;
case SCE_NP_MATCHING2_CASTTYPE_UNICAST:
dst.push_back(req->dst.unicastTarget);
break;
case SCE_NP_MATCHING2_CASTTYPE_MULTICAST:
for (u32 i = 0; i < req->dst.multicastTarget.memberIdNum; i++)
{
dst.push_back(req->dst.multicastTarget.memberId[i]);
}
break;
case SCE_NP_MATCHING2_CASTTYPE_MULTICAST_TEAM:
dst.push_back(req->dst.multicastTargetTeamId);
break;
default:
ensure(false);
2020-08-27 21:47:04 +02:00
break;
2020-11-17 02:19:17 +01:00
}
auto req_finished = CreateSendRoomMessageRequest(builder, req->roomId, req->castType, builder.CreateVector(dst.data(), dst.size()), builder.CreateVector(reinterpret_cast<const u8*>(req->msg.get_ptr()), req->msgLen), req->option);
builder.Finish(req_finished);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(COMMUNICATION_ID_SIZE + bufsize + sizeof(u32));
memcpy(data.data(), communication_id.data, COMMUNICATION_ID_SIZE);
reinterpret_cast<le_t<u32>&>(data[COMMUNICATION_ID_SIZE]) = static_cast<u32>(bufsize);
memcpy(data.data() + COMMUNICATION_ID_SIZE + sizeof(u32), buf, bufsize);
if (!forge_send(CommandType::SendRoomMessage, req_id, data))
return false;
return true;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::req_sign_infos(u32 req_id, const std::string& npid)
{
std::vector<u8> data;
std::copy(npid.begin(), npid.end(), std::back_inserter(data));
data.push_back(0);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::RequestSignalingInfos, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
bool rpcn_client::req_ticket(u32 req_id, const std::string& service_id)
{
std::vector<u8> data;
std::copy(service_id.begin(), service_id.end(), std::back_inserter(data));
data.push_back(0);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::RequestTicket, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::sendmessage(const message_data& msg_data, const std::set<std::string>& npids)
{
std::vector<u8> data;
flatbuffers::FlatBufferBuilder builder(1024);
2020-11-17 02:19:17 +01:00
flatbuffers::FlatBufferBuilder nested_builder(1024);
auto fb_message = CreateMessageDetailsDirect(nested_builder, static_cast<const char*>(msg_data.commId.data), msg_data.msgId, msg_data.mainType, msg_data.subType, msg_data.msgFeatures, msg_data.subject.c_str(), msg_data.body.c_str(), &msg_data.data);
nested_builder.Finish(fb_message);
builder.ForceVectorAlignment(nested_builder.GetSize(), sizeof(uint8_t), nested_builder.GetBufferMinAlignment());
auto nested_flatbuffer_vector = builder.CreateVector(nested_builder.GetBufferPointer(), nested_builder.GetSize());
2020-11-17 02:19:17 +01:00
std::vector<flatbuffers::Offset<flatbuffers::String>> davec;
for (const auto& npid : npids)
{
auto s_npid = builder.CreateString(npid);
davec.push_back(s_npid);
}
auto npids_vector = builder.CreateVector(davec);
2020-11-17 02:19:17 +01:00
//auto npids = builder.Create
auto fb_sendmessage = CreateSendMessageRequest(builder, nested_flatbuffer_vector, npids_vector);
2020-11-17 02:19:17 +01:00
builder.Finish(fb_sendmessage);
u8* buf = builder.GetBufferPointer();
usz bufsize = builder.GetSize();
data.resize(bufsize + sizeof(u32));
2020-11-17 02:19:17 +01:00
reinterpret_cast<le_t<u32>&>(data[0]) = static_cast<u32>(bufsize);
memcpy(data.data() + sizeof(u32), buf, bufsize);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
u64 req_id = rpcn_request_counter.fetch_add(1);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (!forge_send(CommandType::SendMessage, req_id, data))
return false;
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
return true;
}
std::vector<u8> rpcn_client::forge_request(u16 command, u64 packet_id, const std::vector<u8>& data) const
{
u16 packet_size = data.size() + RPCN_HEADER_SIZE;
std::vector<u8> packet(packet_size);
packet[0] = PacketType::Request;
reinterpret_cast<le_t<u16>&>(packet[1]) = command;
reinterpret_cast<le_t<u16>&>(packet[3]) = packet_size;
reinterpret_cast<le_t<u64>&>(packet[5]) = packet_id;
memcpy(packet.data() + RPCN_HEADER_SIZE, data.data(), data.size());
return packet;
}
bool rpcn_client::is_error(ErrorType err) const
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
if (err >= ErrorType::__error_last)
{
rpcn_log.error("Invalid error returned!");
return true;
}
switch (err)
{
case NoError: return false;
case Malformed: rpcn_log.error("Sent packet was malformed!"); break;
case Invalid: rpcn_log.error("Sent command was invalid!"); break;
case InvalidInput: rpcn_log.error("Sent data was invalid!"); break;
case TooSoon: rpcn_log.error("Request happened too soon!"); break;
case LoginError: rpcn_log.error("Unknown login error!"); break;
case LoginAlreadyLoggedIn: rpcn_log.error("User is already logged in!"); break;
case LoginInvalidUsername: rpcn_log.error("Login error: invalid username!"); break;
case LoginInvalidPassword: rpcn_log.error("Login error: invalid password!"); break;
case LoginInvalidToken: rpcn_log.error("Login error: invalid token!"); break;
case CreationError: rpcn_log.error("Error creating an account!"); break;
case CreationExistingUsername: rpcn_log.error("Error creating an account: existing username!");
case CreationBannedEmailProvider: rpcn_log.error("Error creating an account: banned email provider!");
case CreationExistingEmail: rpcn_log.error("Error creating an account: an account with that email already exist!");
case AlreadyJoined: rpcn_log.error("User has already joined!"); break;
case DbFail: rpcn_log.error("A db query failed on the server!"); break;
case EmailFail: rpcn_log.error("An email action failed on the server!"); break;
case NotFound: rpcn_log.error("A request replied not found!"); return false;
case Blocked: rpcn_log.error("You're blocked!"); break;
case AlreadyFriend: rpcn_log.error("You're already friends!"); break;
case Unsupported: rpcn_log.error("An unsupported operation was attempted!"); return false;
default: rpcn_log.fatal("Unhandled ErrorType reached the switch?"); break;
}
2020-08-27 21:47:04 +02:00
return true;
}
2020-11-17 02:19:17 +01:00
bool rpcn_client::error_and_disconnect(const std::string& error_msg)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
connected = false;
rpcn_log.error("%s", error_msg);
return false;
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
rpcn_state rpcn_client::wait_for_connection()
{
{
std::lock_guard lock(mutex_connected);
if (connected)
{
return rpcn_state::failure_no_failure;
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
if (state != rpcn_state::failure_no_failure)
{
return state;
}
}
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
sem_connected.acquire();
return state;
}
rpcn_state rpcn_client::wait_for_authentified()
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
{
std::lock_guard lock(mutex_authentified);
if (authentified)
{
return rpcn_state::failure_no_failure;
}
if (state != rpcn_state::failure_no_failure)
{
return state;
}
want_auth = true;
sem_rpcn.release();
}
sem_authentified.acquire();
return state;
}
void rpcn_client::get_friends_and_register_cb(friend_data& friend_infos, friend_cb_func cb_func, void* cb_param)
{
std::lock_guard lock(mutex_friends);
friend_infos = this->friend_infos;
friend_cbs.insert(std::make_pair(cb_func, cb_param));
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
void rpcn_client::remove_friend_cb(friend_cb_func cb_func, void* cb_param)
2020-08-27 21:47:04 +02:00
{
2020-11-17 02:19:17 +01:00
std::lock_guard lock(mutex_friends);
for (const auto& friend_cb : friend_cbs)
{
if (friend_cb.first == cb_func && friend_cb.second == cb_param)
{
friend_cbs.erase(friend_cb);
break;
}
}
2020-08-27 21:47:04 +02:00
}
2020-11-17 02:19:17 +01:00
void rpcn_client::handle_friend_notification(u16 command, std::vector<u8> data)
{
std::lock_guard lock(mutex_friends);
2020-08-27 21:47:04 +02:00
2020-11-17 02:19:17 +01:00
NotificationType ntype = static_cast<NotificationType>(command);
vec_stream vdata(data);
auto call_callbacks = [&](NotificationType ntype, std::string username, bool status)
{
for (auto& friend_cb : friend_cbs)
{
friend_cb.first(friend_cb.second, ntype, username, status);
}
};
switch (ntype)
{
case NotificationType::FriendQuery: // Other user sent a friend request
{
std::string username = vdata.get_string(false);
if (vdata.is_error())
{
rpcn_log.error("Error parsing FriendQuery notification");
break;
}
friend_infos.requests_received.insert(username);
call_callbacks(ntype, username, 0);
break;
}
case NotificationType::FriendNew: // Add a friend to the friendlist(either accepted a friend request or friend accepted it)
{
bool status = !!vdata.get<u8>();
std::string username = vdata.get_string(false);
if (vdata.is_error())
{
rpcn_log.error("Error parsing FriendNew notification");
break;
}
friend_infos.requests_received.erase(username);
friend_infos.requests_sent.erase(username);
friend_infos.friends.insert(std::make_pair(username, std::make_pair(status, 0)));
call_callbacks(ntype, username, status);
break;
}
case NotificationType::FriendLost: // Remove friend from the friendlist(user removed friend or friend removed friend)
{
std::string username = vdata.get_string(false);
if (vdata.is_error())
{
rpcn_log.error("Error parsing FriendLost notification");
break;
}
friend_infos.friends.erase(username);
call_callbacks(ntype, username, 0);
break;
}
case NotificationType::FriendStatus: // Set status of friend to Offline or Online
{
bool status = !!vdata.get<u8>();
u64 timestamp = vdata.get<u64>();
std::string username = vdata.get_string(false);
if (vdata.is_error())
{
rpcn_log.error("Error parsing FriendStatus notification");
break;
}
if (auto u = friend_infos.friends.find(username); u != friend_infos.friends.end())
{
if (timestamp < u->second.second)
{
break;
}
u->second.second = timestamp;
u->second.first = status;
call_callbacks(ntype, username, status);
}
break;
}
default:
{
rpcn_log.fatal("Unknown notification forwarded to handle_friend_notification");
break;
}
}
}
void rpcn_client::handle_message(std::vector<u8> data)
{
// Unserialize the message
vec_stream sdata(data);
std::string sender = sdata.get_string(false);
auto raw_msg_data = sdata.get_rawdata();
if (sdata.is_error())
{
return;
}
auto fb_mdata = flatbuffers::GetRoot<MessageDetails>(raw_msg_data.data());
if (!fb_mdata->communicationId() || fb_mdata->communicationId()->size() == 0 || fb_mdata->communicationId()->size() > 9 ||
!fb_mdata->subject() || !fb_mdata->body() || !fb_mdata->data())
{
rpcn_log.warning("Discarded invalid messaged!");
return;
}
message_data mdata = {
.msgId = message_counter,
.mainType = fb_mdata->mainType(),
.subType = fb_mdata->subType(),
.msgFeatures = fb_mdata->msgFeatures(),
.subject = fb_mdata->subject()->str(),
.body = fb_mdata->body()->str()};
strcpy_trunc(mdata.commId.data, fb_mdata->communicationId()->str());
mdata.data.assign(fb_mdata->data()->Data(), fb_mdata->data()->Data() + fb_mdata->data()->size());
// Save the message and call callbacks
{
std::lock_guard lock(mutex_messages);
const u64 msg_id = message_counter++;
auto id_and_msg = std::make_shared<std::pair<std::string, message_data>>(std::make_pair(std::move(sender), std::move(mdata)));
messages.emplace(msg_id, id_and_msg);
new_messages.push_back(msg_id);
active_messages.insert(msg_id);
const auto& msg = id_and_msg->second;
for (const auto& message_cb : message_cbs)
{
if (msg.mainType == message_cb.type_filter && (message_cb.inc_bootable || !(msg.msgFeatures & SCE_NP_BASIC_MESSAGE_FEATURES_BOOTABLE)))
{
message_cb.cb_func(message_cb.cb_param, id_and_msg, msg_id);
}
}
}
}
std::optional<std::shared_ptr<std::pair<std::string, message_data>>> rpcn_client::get_message(u64 id)
{
{
std::lock_guard lock(mutex_messages);
return messages.at(id);
}
}
std::vector<std::pair<u64, std::shared_ptr<std::pair<std::string, message_data>>>> rpcn_client::get_messages_and_register_cb(SceNpBasicMessageMainType type_filter, bool include_bootable, message_cb_func cb_func, void* cb_param)
{
std::vector<std::pair<u64, std::shared_ptr<std::pair<std::string, message_data>>>> vec_messages;
{
std::lock_guard lock(mutex_messages);
for (auto id : active_messages)
{
const auto& entry = messages.at(id);
const auto& msg = entry->second;
if (msg.mainType == type_filter && (include_bootable || !(msg.msgFeatures & SCE_NP_BASIC_MESSAGE_FEATURES_BOOTABLE)))
{
vec_messages.push_back(std::make_pair(id, entry));
}
}
message_cbs.insert(message_cb_t{
.cb_func = cb_func,
.cb_param = cb_param,
.type_filter = type_filter,
.inc_bootable = include_bootable,
});
}
return vec_messages;
}
void rpcn_client::remove_message_cb(message_cb_func cb_func, void* cb_param)
{
std::lock_guard lock(mutex_messages);
for (const auto& message_cb : message_cbs)
{
if (message_cb.cb_func == cb_func && message_cb.cb_param == cb_param)
{
message_cbs.erase(message_cb);
break;
}
}
}
void rpcn_client::discard_active_message(u64 id)
{
std::lock_guard lock(mutex_messages);
active_messages.erase(id);
}
u32 rpcn_client::get_num_friends() const
{
return friend_infos.friends.size();
}
u32 rpcn_client::get_num_blocks() const
{
return friend_infos.blocked.size();
}
} // namespace rpcn