RetroArch/discord/discord.c

487 lines
15 KiB
C
Raw Normal View History

2018-05-29 04:43:30 +00:00
/* RetroArch - A frontend for libretro.
* Copyright (C) 2018 - Andrés Suárez
*
* RetroArch is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with RetroArch.
* If not, see <http://www.gnu.org/licenses/>.
*/
#include <file/file_path.h>
2019-01-19 22:15:48 +00:00
#include <string/stdstring.h>
2019-01-19 22:27:59 +00:00
#include <retro_timers.h>
2018-05-29 04:43:30 +00:00
2018-08-12 16:38:05 +00:00
#include "discord.h"
#include "discord_register.h"
2018-08-12 16:38:05 +00:00
2019-01-19 22:15:48 +00:00
#include "../deps/discord-rpc/include/discord_rpc.h"
2018-08-12 16:38:05 +00:00
#include "../retroarch.h"
2018-10-13 23:17:17 +00:00
#include "../configuration.h"
2018-08-12 16:38:05 +00:00
#include "../core.h"
#include "../core_info.h"
#include "../paths.h"
#include "../playlist.h"
2019-01-19 22:15:48 +00:00
#include "../verbosity.h"
2018-08-12 16:38:05 +00:00
2018-05-29 09:14:21 +00:00
#include "../msg_hash.h"
#include "../tasks/task_file_transfer.h"
2018-05-29 09:14:21 +00:00
#ifdef HAVE_NETWORKING
#include "../../network/netplay/netplay.h"
#include "../../network/netplay/netplay_discovery.h"
2018-09-12 04:26:55 +00:00
#include "../../tasks/tasks_internal.h"
#endif
#ifdef HAVE_CHEEVOS
#include "../cheevos/cheevos.h"
#endif
#ifdef HAVE_MENU
#include "../../menu/widgets/menu_input_dialog.h"
#include "../../menu/menu_cbs.h"
#endif
#include <net/net_http.h>
#include "../network/net_http_special.h"
#include "../tasks/tasks_internal.h"
#include <streams/file_stream.h>
#include <file/file_path.h>
#include "../file_path_special.h"
2018-05-29 08:40:01 +00:00
static int64_t start_time = 0;
static int64_t pause_time = 0;
static int64_t ellapsed_time = 0;
2018-05-29 04:43:30 +00:00
2018-05-29 08:40:01 +00:00
static bool discord_ready = false;
2018-12-24 20:06:21 +00:00
static bool discord_avatar_ready = false;
2018-05-29 08:40:01 +00:00
static unsigned discord_status = 0;
2018-05-29 04:43:30 +00:00
2018-09-12 04:26:55 +00:00
struct netplay_room *room;
static char user_id[128];
static char user_name[128];
static char party_name[128];
static char client_party_name[128];
2018-12-24 20:06:21 +00:00
static char user_avatar[PATH_MAX_LENGTH];
static bool connecting = false;
static char cdn_url[] = "https://cdn.discordapp.com/avatars";
2018-05-29 04:43:30 +00:00
DiscordRichPresence discord_presence;
char* discord_get_own_username(void)
{
2019-02-03 19:10:28 +00:00
if (discord_is_ready())
return user_name;
return NULL;
}
char* discord_get_own_avatar(void)
{
2019-02-03 19:10:28 +00:00
if (discord_is_ready())
return user_avatar;
return NULL;
2018-12-24 20:06:21 +00:00
}
2019-01-06 16:04:28 +00:00
bool discord_avatar_is_ready(void)
2018-12-24 20:06:21 +00:00
{
/*To-Do: fix-me, prevent lockups in ozone due to unfinished code*/
return false;
2018-12-24 20:06:21 +00:00
}
void discord_avatar_set_ready(bool ready)
{
discord_avatar_ready = ready;
}
2019-01-06 16:04:28 +00:00
bool discord_is_ready(void)
{
return discord_ready;
}
2019-01-06 16:04:28 +00:00
#ifdef HAVE_MENU
static bool discord_download_avatar(
const char* user_id, const char* avatar_id)
{
static char url[PATH_MAX_LENGTH];
static char url_encoded[PATH_MAX_LENGTH];
2018-12-24 20:06:21 +00:00
static char full_path[PATH_MAX_LENGTH];
static char buf[PATH_MAX_LENGTH];
file_transfer_t *transf = NULL;
fill_pathname_application_special(buf,
sizeof(buf),
APPLICATION_SPECIAL_DIRECTORY_THUMBNAILS_DISCORD_AVATARS);
2018-12-24 20:06:21 +00:00
fill_pathname_join(full_path, buf, avatar_id, sizeof(full_path));
strlcpy(user_avatar, avatar_id, sizeof(user_avatar));
2018-12-24 20:06:21 +00:00
if(filestream_exists(full_path))
return true;
if (string_is_empty(avatar_id))
return false;
2019-01-06 16:04:28 +00:00
snprintf(url, sizeof(url), "%s/%s/%s.png", cdn_url, user_id, avatar_id);
net_http_urlencode_full(url_encoded, url, sizeof(url_encoded));
snprintf(buf, sizeof(buf), "%s.png", avatar_id);
2019-01-06 16:04:28 +00:00
transf = (file_transfer_t*)calloc(1, sizeof(*transf));
transf->enum_idx = MENU_ENUM_LABEL_CB_DISCORD_AVATAR;
strlcpy(transf->path, buf, sizeof(transf->path));
2019-01-06 16:04:28 +00:00
RARCH_LOG("[Discord] downloading avatar from: %s\n", url_encoded);
task_push_http_transfer(url_encoded, true, NULL, cb_generic_download, transf);
return false;
}
2019-01-06 16:04:28 +00:00
#endif
2018-05-29 04:43:30 +00:00
static void handle_discord_ready(const DiscordUser* connectedUser)
{
strlcpy(user_name, connectedUser->username, sizeof(user_name));
strlcpy(party_name, connectedUser->username, sizeof(user_name));
strlcat(party_name, "|", sizeof(party_name));
strlcat(party_name, connectedUser->discriminator, sizeof(party_name));
RARCH_LOG("[Discord] connected to user: %s#%s - avatar id: %s\n",
2018-05-29 04:43:30 +00:00
connectedUser->username,
connectedUser->discriminator,
connectedUser->userId);
2019-01-06 16:04:28 +00:00
#ifdef HAVE_MENU
discord_download_avatar(connectedUser->userId, connectedUser->avatar);
2019-01-06 16:04:28 +00:00
#endif
2018-05-29 04:43:30 +00:00
}
static void handle_discord_disconnected(int errcode, const char* message)
{
2018-05-29 08:40:01 +00:00
RARCH_LOG("[Discord] disconnected (%d: %s)\n", errcode, message);
2018-05-29 04:43:30 +00:00
}
static void handle_discord_error(int errcode, const char* message)
{
2018-05-29 08:40:01 +00:00
RARCH_LOG("[Discord] error (%d: %s)\n", errcode, message);
2018-05-29 04:43:30 +00:00
}
static void handle_discord_join_cb(retro_task_t *task, void *task_data, void *user_data, const char *err)
{
struct netplay_room *room;
char tmp_hostname[32];
http_transfer_data_t *data = (http_transfer_data_t*)task_data;
if (!data || err)
goto finish;
data->data = (char*)realloc(data->data, data->len + 1);
data->data[data->len] = '\0';
netplay_rooms_parse(data->data);
room = netplay_room_get(0);
if (room)
{
if (netplay_driver_ctl(RARCH_NETPLAY_CTL_IS_DATA_INITED, NULL))
deinit_netplay();
netplay_driver_ctl(RARCH_NETPLAY_CTL_ENABLE_CLIENT, NULL);
2019-01-07 23:59:26 +00:00
snprintf(tmp_hostname, sizeof(tmp_hostname), "%s|%d",
room->host_method == NETPLAY_HOST_METHOD_MITM ? room->mitm_address : room->address,
room->host_method == NETPLAY_HOST_METHOD_MITM ? room->mitm_port : room->port);
2019-01-07 23:59:26 +00:00
RARCH_LOG("[Discord] joining lobby at: %s\n", tmp_hostname);
task_push_netplay_crc_scan(room->gamecrc,
room->gamename, tmp_hostname, room->corename, room->subsystem_name);
connecting = true;
discord_update(DISCORD_PRESENCE_NETPLAY_CLIENT);
}
finish:
if (err)
RARCH_ERR("%s: %s\n", msg_hash_to_str(MSG_DOWNLOAD_FAILED), err);
if (data)
{
if (data->data)
free(data->data);
free(data);
}
if (user_data)
free(user_data);
}
2018-05-29 04:43:30 +00:00
static void handle_discord_join(const char* secret)
{
char url [2048] = "http://lobby.libretro.com/";
2019-01-06 16:04:28 +00:00
char tmp_hostname[32];
2018-09-12 04:26:55 +00:00
static struct string_list *list = NULL;
2019-01-06 16:04:28 +00:00
RARCH_LOG("[Discord] join secret: (%s)\n", secret);
2018-09-12 04:26:55 +00:00
list = string_split(secret, "|");
strlcpy(client_party_name, list->elems[1].data, sizeof(client_party_name));
strlcat(client_party_name, "|", sizeof(client_party_name));
strlcat(client_party_name, list->elems[2].data, sizeof(client_party_name));
2019-02-06 05:07:24 +00:00
strlcat(url, list->elems[0].data, sizeof(url));
strlcat(url, "/", sizeof(url));
RARCH_LOG("[Discord] querying lobby id: %s at %s\n", list->elems[0].data, url);
2018-09-12 04:26:55 +00:00
snprintf(tmp_hostname,
sizeof(tmp_hostname),
"%s|%s", list->elems[0].data, list->elems[1].data);
task_push_http_transfer(url, true, NULL, handle_discord_join_cb, NULL);
2018-05-29 04:43:30 +00:00
}
static void handle_discord_spectate(const char* secret)
{
2018-05-29 08:40:01 +00:00
RARCH_LOG("[Discord] spectate (%s)\n", secret);
2018-05-29 04:43:30 +00:00
}
static void handle_discord_join_response(void *ignore, const char *line)
{
/* To-Do: needs in-game widgets
if (strstr(line, "yes"))
Discord_Respond(user_id, DISCORD_REPLY_YES);
#ifdef HAVE_MENU
menu_input_dialog_end();
rarch_menu_running_finished();
#endif
*/
}
2018-05-29 04:43:30 +00:00
static void handle_discord_join_request(const DiscordUser* request)
{
static char url[PATH_MAX_LENGTH];
static char url_encoded[PATH_MAX_LENGTH];
static char filename[PATH_MAX_LENGTH];
2018-11-14 02:59:39 +00:00
char buf[PATH_MAX_LENGTH];
2019-01-06 16:04:28 +00:00
#ifdef HAVE_MENU
menu_input_ctx_line_t line;
#endif
RARCH_LOG("[Discord] join request from %s#%s - %s %s\n",
2018-05-29 04:43:30 +00:00
request->username,
request->discriminator,
request->userId,
request->avatar);
#ifdef HAVE_MENU
2019-01-06 16:04:28 +00:00
discord_download_avatar(request->userId, request->avatar);
/* To-Do: needs in-game widgets
rarch_menu_running();
2018-11-14 02:59:39 +00:00
*/
2019-01-06 16:04:28 +00:00
memset(&line, 0, sizeof(line));
snprintf(buf, sizeof(buf), "%s %s?", msg_hash_to_str(MSG_DISCORD_CONNECTION_REQUEST), request->username);
line.label = buf;
line.label_setting = "no_setting";
line.cb = handle_discord_join_response;
2019-01-06 16:04:28 +00:00
/* To-Do: needs in-game widgets
To-Do: bespoke dialog, should show while in-game and have a hotkey to accept
To-Do: show avatar of the user connecting
if (!menu_input_dialog_start(&line))
2019-01-06 16:04:28 +00:00
return;
*/
#endif
2018-05-29 04:43:30 +00:00
}
void discord_update(enum discord_presence presence)
2018-05-29 04:43:30 +00:00
{
core_info_t *core_info = NULL;
core_info_get_current_core(&core_info);
2018-05-29 08:40:01 +00:00
if (!discord_ready)
2018-05-29 04:43:30 +00:00
return;
if (presence == discord_status)
2018-05-29 08:40:01 +00:00
return;
if (!connecting && (presence == DISCORD_PRESENCE_NONE || presence == DISCORD_PRESENCE_MENU))
{
memset(&discord_presence, 0, sizeof(discord_presence));
client_party_name[0] = '\0';
}
2018-05-29 04:43:30 +00:00
switch (presence)
{
case DISCORD_PRESENCE_MENU:
2019-01-06 16:04:28 +00:00
discord_presence.details = msg_hash_to_str(MENU_ENUM_LABEL_VALUE_DISCORD_IN_MENU);
discord_presence.largeImageKey = "base";
discord_presence.largeImageText = msg_hash_to_str(MENU_ENUM_LABEL_VALUE_NO_CORE);
discord_presence.instance = 0;
2018-05-29 04:43:30 +00:00
break;
case DISCORD_PRESENCE_GAME_PAUSED:
2019-01-19 22:15:48 +00:00
discord_presence.smallImageKey = "paused";
discord_presence.smallImageText = msg_hash_to_str(
MENU_ENUM_LABEL_VALUE_DISCORD_STATUS_PAUSED);
discord_presence.details = msg_hash_to_str(
MENU_ENUM_LABEL_VALUE_DISCORD_IN_GAME_PAUSED);
pause_time = time(0);
ellapsed_time = difftime(time(0), start_time);
discord_presence.startTimestamp = pause_time;
break;
case DISCORD_PRESENCE_GAME:
if (core_info)
{
2019-02-03 23:49:35 +00:00
const char *system_id = core_info->system_id
2019-01-06 16:04:28 +00:00
? core_info->system_id : "core";
char *label = NULL;
playlist_t *current_playlist = playlist_get_cached();
if (current_playlist)
playlist_get_index_by_path(
current_playlist, path_get(RARCH_PATH_CONTENT), NULL, &label, NULL, NULL, NULL, NULL);
if (!label)
label = (char *)path_basename(path_get(RARCH_PATH_BASENAME));
#if 0
2018-08-21 14:41:16 +00:00
RARCH_LOG("[Discord] current core: %s\n", system_id);
RARCH_LOG("[Discord] current content: %s\n", label);
2018-05-29 08:40:01 +00:00
#endif
2018-08-21 14:41:16 +00:00
discord_presence.largeImageKey = system_id;
if (core_info->display_name)
discord_presence.largeImageText = core_info->display_name;
start_time = time(0);
if (pause_time != 0)
start_time = time(0) - ellapsed_time;
2019-01-19 22:15:48 +00:00
pause_time = 0;
ellapsed_time = 0;
2019-01-06 16:04:28 +00:00
discord_presence.smallImageKey = "playing";
2019-01-19 22:15:48 +00:00
discord_presence.smallImageText = msg_hash_to_str(
MENU_ENUM_LABEL_VALUE_DISCORD_STATUS_PLAYING);
discord_presence.startTimestamp = start_time;
2019-01-19 22:15:48 +00:00
discord_presence.details = msg_hash_to_str(
MENU_ENUM_LABEL_VALUE_DISCORD_IN_GAME);
2019-01-06 16:04:28 +00:00
discord_presence.state = label;
discord_presence.instance = 0;
2019-02-06 05:07:24 +00:00
if (!netplay_driver_ctl(RARCH_NETPLAY_CTL_IS_ENABLED, NULL))
{
client_party_name[0] = '\0';
discord_presence.partyId = NULL;
discord_presence.partyMax = 0;
discord_presence.partySize = 0;
connecting = false;
}
}
2018-05-29 04:43:30 +00:00
break;
case DISCORD_PRESENCE_NETPLAY_HOSTING:
2018-09-12 04:26:55 +00:00
room = netplay_get_host_room();
if (room->id == 0)
return;
RARCH_LOG("[Discord] netplay room details: id=%d, nick=%s IP=%s port=%d\n",
room->id, room->nickname,
room->host_method == NETPLAY_HOST_METHOD_MITM ? room->mitm_address : room->address,
room->host_method == NETPLAY_HOST_METHOD_MITM ? room->mitm_port : room->port);
2019-01-06 16:04:28 +00:00
{
char join_secret[128];
2019-02-06 05:07:24 +00:00
snprintf(join_secret, sizeof(join_secret), "%d|%s", room->id, party_name);
2019-01-06 16:04:28 +00:00
discord_presence.joinSecret = strdup(join_secret);
/* discord_presence.spectateSecret = "SPECSPECSPEC"; */
2019-01-19 22:15:48 +00:00
discord_presence.partyId = strdup(party_name);
2019-02-06 05:07:24 +00:00
discord_presence.partyMax = 2;
discord_presence.partySize = 1;
RARCH_LOG("[Discord] join secret: %s\n", join_secret);
RARCH_LOG("[Discord] party id: %s\n", party_name);
2019-01-06 16:04:28 +00:00
}
break;
case DISCORD_PRESENCE_NETPLAY_CLIENT:
RARCH_LOG("[Discord] party id: %s\n", client_party_name);
discord_presence.partyId = strdup(client_party_name);
2019-02-06 05:07:24 +00:00
break;
case DISCORD_PRESENCE_NETPLAY_NETPLAY_STOPPED:
{
if (!netplay_driver_ctl(RARCH_NETPLAY_CTL_IS_ENABLED, NULL) &&
!netplay_driver_ctl(RARCH_NETPLAY_CTL_IS_CONNECTED, NULL))
{
client_party_name[0] = '\0';
discord_presence.partyId = NULL;
discord_presence.partyMax = 0;
discord_presence.partySize = 0;
connecting = false;
}
}
break;
case DISCORD_PRESENCE_SHUTDOWN:
discord_presence.partyId = NULL;
discord_presence.partyMax = 0;
discord_presence.partySize = 0;
connecting = false;
default:
2018-05-29 04:43:30 +00:00
break;
}
RARCH_LOG("[Discord] updating (%d)\n", presence);
2018-05-29 04:43:30 +00:00
Discord_UpdatePresence(&discord_presence);
discord_status = presence;
2018-05-29 04:43:30 +00:00
}
2018-05-29 08:40:01 +00:00
void discord_init(void)
2018-05-29 04:43:30 +00:00
{
2019-01-06 16:04:28 +00:00
char command[PATH_MAX_LENGTH];
2018-10-13 23:17:17 +00:00
settings_t *settings = config_get_ptr();
2019-01-06 16:04:28 +00:00
2018-05-29 04:43:30 +00:00
DiscordEventHandlers handlers;
2018-05-29 08:40:01 +00:00
2018-05-29 09:14:21 +00:00
RARCH_LOG("[Discord] initializing ..\n");
2018-05-29 08:40:01 +00:00
start_time = time(0);
2018-05-29 04:43:30 +00:00
memset(&handlers, 0, sizeof(handlers));
2018-05-29 08:40:01 +00:00
handlers.ready = handle_discord_ready;
2018-05-29 04:43:30 +00:00
handlers.disconnected = handle_discord_disconnected;
2018-05-29 08:40:01 +00:00
handlers.errored = handle_discord_error;
handlers.joinGame = handle_discord_join;
2018-05-29 04:43:30 +00:00
handlers.spectateGame = handle_discord_spectate;
2018-05-29 08:40:01 +00:00
handlers.joinRequest = handle_discord_join_request;
Discord_Initialize(settings->arrays.discord_app_id, &handlers, 0, NULL);
2018-05-29 04:43:30 +00:00
#ifdef _WIN32)
strlcpy(command, get_retroarch_launch_arguments(), sizeof(command));
#else
snprintf(command, sizeof(command), "sh -c %s", get_retroarch_launch_arguments());
#endif
RARCH_LOG("[Discord] registering startup command: %s\n", command);
Discord_Register(settings->arrays.discord_app_id, command);
2018-12-24 20:06:21 +00:00
discord_ready = true;
2018-05-29 04:43:30 +00:00
}
2018-05-29 08:40:01 +00:00
void discord_shutdown(void)
2018-05-29 04:43:30 +00:00
{
2018-05-29 09:14:21 +00:00
RARCH_LOG("[Discord] shutting down ..\n");
2018-05-29 04:43:30 +00:00
Discord_ClearPresence();
Discord_Shutdown();
discord_ready = false;
2018-05-29 08:40:01 +00:00
}
2019-01-06 16:04:28 +00:00
void discord_run_callbacks(void)
{
Discord_RunCallbacks();
2019-01-06 16:04:28 +00:00
}