RetroArch/tasks/task_http.c

364 lines
8.9 KiB
C
Raw Normal View History

2015-05-05 15:36:58 +00:00
/* RetroArch - A frontend for libretro.
2017-01-22 12:40:32 +00:00
* Copyright (C) 2011-2017 - Daniel De Matteis
2015-05-05 15:36:58 +00:00
*
* 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/>.
*/
2017-05-09 18:10:58 +00:00
#include <stdlib.h>
2015-05-05 15:36:58 +00:00
#include <net/net_http.h>
2016-01-20 03:07:24 +00:00
#include <string/stdstring.h>
2015-05-05 16:34:48 +00:00
#include <compat/strl.h>
2015-06-08 09:10:12 +00:00
#include <file/file_path.h>
#include <net/net_compat.h>
2017-06-28 02:41:38 +00:00
#include <retro_timers.h>
2015-05-05 15:36:58 +00:00
2019-05-22 07:55:10 +00:00
#ifdef RARCH_INTERNAL
#include "../gfx/video_display_server.h"
2019-05-22 07:55:10 +00:00
#endif
#include "task_file_transfer.h"
#include "tasks_internal.h"
2015-05-05 15:36:58 +00:00
2015-07-08 19:35:24 +00:00
enum http_status_enum
{
2017-05-20 12:02:16 +00:00
HTTP_STATUS_CONNECTION_TRANSFER = 0,
2015-07-08 19:35:24 +00:00
HTTP_STATUS_CONNECTION_TRANSFER_PARSE,
HTTP_STATUS_TRANSFER,
HTTP_STATUS_TRANSFER_PARSE,
HTTP_STATUS_TRANSFER_PARSE_FREE
};
2017-05-13 16:10:26 +00:00
struct http_transfer_info
2016-05-18 22:04:38 +00:00
{
2016-10-27 08:19:04 +00:00
char url[255];
2016-05-18 22:04:38 +00:00
int progress;
2017-05-13 16:10:26 +00:00
};
2016-05-18 22:04:38 +00:00
2017-05-13 16:10:26 +00:00
struct http_handle
2015-07-08 19:35:24 +00:00
{
struct
{
struct http_connection_t *handle;
transfer_cb_t cb;
2016-10-27 08:19:04 +00:00
char elem1[255];
char url[255];
2015-07-08 19:35:24 +00:00
} connection;
struct http_t *handle;
transfer_cb_t cb;
unsigned status;
bool error;
2017-05-13 16:10:26 +00:00
};
typedef struct http_transfer_info http_transfer_info_t;
typedef struct http_handle http_handle_t;
2015-07-08 19:35:24 +00:00
2016-05-13 08:19:53 +00:00
static int task_http_con_iterate_transfer(http_handle_t *http)
2015-05-05 15:36:58 +00:00
{
if (!net_http_connection_iterate(http->connection.handle))
return -1;
return 0;
}
2016-05-13 08:19:53 +00:00
static int task_http_conn_iterate_transfer_parse(
2016-02-03 16:41:04 +00:00
http_handle_t *http)
2015-05-05 15:36:58 +00:00
{
if (net_http_connection_done(http->connection.handle))
{
if (http->connection.handle && http->connection.cb)
http->connection.cb(http, 0);
}
2015-05-05 15:36:58 +00:00
net_http_connection_free(http->connection.handle);
http->connection.handle = NULL;
return 0;
}
static int cb_http_conn_default(void *data_, size_t len)
{
http_handle_t *http = (http_handle_t*)data_;
if (!http)
return -1;
if (!network_init())
return -1;
2015-05-05 15:36:58 +00:00
http->handle = net_http_new(http->connection.handle);
if (!http->handle)
{
http->error = true;
2015-05-05 15:36:58 +00:00
return -1;
}
http->cb = NULL;
return 0;
}
/**
2016-05-13 08:19:53 +00:00
* task_http_iterate_transfer:
2015-05-05 15:36:58 +00:00
*
* Resumes HTTP transfer update.
*
* Returns: 0 when finished, -1 when we should continue
* with the transfer on the next frame.
**/
2016-05-13 08:19:53 +00:00
static int task_http_iterate_transfer(retro_task_t *task)
2015-05-05 15:36:58 +00:00
{
http_handle_t *http = (http_handle_t*)task->state;
2015-07-01 17:40:50 +00:00
size_t pos = 0, tot = 0;
/* FIXME: This wouldn't be needed if we could wait for a timeout */
if (task_queue_is_threaded())
retro_sleep(1);
2015-07-01 17:40:50 +00:00
if (!net_http_update(http->handle, &pos, &tot))
{
task_set_progress(task, (tot == 0) ? -1 : (signed)pos / (tot / 100));
2015-07-01 17:40:50 +00:00
return -1;
}
return 0;
2015-05-05 15:36:58 +00:00
}
2016-05-27 16:14:47 +00:00
static void task_http_transfer_handler(retro_task_t *task)
2015-05-05 15:36:58 +00:00
{
2016-05-13 08:28:44 +00:00
http_transfer_data_t *data = NULL;
http_handle_t *http = (http_handle_t*)task->state;
2015-05-05 15:36:58 +00:00
if (task_get_cancelled(task))
2016-05-05 18:32:36 +00:00
goto task_finished;
2015-05-05 15:36:58 +00:00
switch (http->status)
{
case HTTP_STATUS_CONNECTION_TRANSFER_PARSE:
2016-05-13 08:19:53 +00:00
task_http_conn_iterate_transfer_parse(http);
2015-05-05 15:36:58 +00:00
http->status = HTTP_STATUS_TRANSFER;
break;
case HTTP_STATUS_CONNECTION_TRANSFER:
2016-05-13 08:19:53 +00:00
if (!task_http_con_iterate_transfer(http))
2015-05-05 15:36:58 +00:00
http->status = HTTP_STATUS_CONNECTION_TRANSFER_PARSE;
break;
case HTTP_STATUS_TRANSFER:
2016-05-13 08:19:53 +00:00
if (!task_http_iterate_transfer(task))
2015-11-24 01:30:57 +00:00
goto task_finished;
2015-05-05 15:36:58 +00:00
break;
2015-11-24 01:30:57 +00:00
case HTTP_STATUS_TRANSFER_PARSE:
2015-11-23 14:13:26 +00:00
goto task_finished;
2015-05-05 15:36:58 +00:00
default:
break;
}
2015-07-08 19:18:34 +00:00
2016-05-05 18:32:36 +00:00
if (http->error)
goto task_finished;
2015-11-23 14:13:26 +00:00
return;
task_finished:
task_set_finished(task, true);
2015-07-08 19:35:24 +00:00
2015-11-23 14:13:26 +00:00
if (http->handle)
{
2015-11-24 01:30:57 +00:00
size_t len = 0;
char *tmp = (char*)net_http_data(http->handle, &len, false);
if (tmp && http->cb)
http->cb(tmp, len);
if (net_http_error(http->handle) || task_get_cancelled(task))
2015-11-24 01:30:57 +00:00
{
tmp = (char*)net_http_data(http->handle, &len, true);
if (tmp)
free(tmp);
if (task_get_cancelled(task))
task_set_error(task, strdup("Task cancelled."));
else if (!task->mute)
task_set_error(task, strdup("Download failed."));
2015-11-24 01:30:57 +00:00
}
else
{
data = (http_transfer_data_t*)calloc(1, sizeof(*data));
data->data = tmp;
data->len = len;
2015-07-08 19:18:34 +00:00
task_set_data(task, data);
2015-11-24 01:30:57 +00:00
}
2015-07-08 19:35:24 +00:00
2015-11-23 14:13:26 +00:00
net_http_delete(http->handle);
} else if (http->error)
task_set_error(task, strdup("Internal error."));
2015-07-08 19:35:24 +00:00
2015-11-23 14:13:26 +00:00
free(http);
2015-07-08 19:35:24 +00:00
}
2016-05-27 16:14:47 +00:00
static bool task_http_finder(retro_task_t *task, void *user_data)
{
2016-05-13 08:28:44 +00:00
http_handle_t *http = NULL;
2016-05-27 16:14:47 +00:00
if (!task || (task->handler != task_http_transfer_handler))
2016-02-08 04:58:27 +00:00
return false;
if (!user_data)
return false;
http = (http_handle_t*)task->state;
if (!http)
2016-02-08 04:58:27 +00:00
return false;
return string_is_equal(http->connection.url, (const char*)user_data);
}
2016-05-27 16:14:47 +00:00
static bool task_http_retriever(retro_task_t *task, void *data)
{
2016-05-08 02:04:50 +00:00
http_transfer_info_t *info = (http_transfer_info_t*)data;
/* Extract HTTP handle and return already if invalid */
2016-05-08 02:04:50 +00:00
http_handle_t *http = (http_handle_t *)task->state;
if (!http)
return false;
/* Fill HTTP info link */
strlcpy(info->url, http->connection.url, sizeof(info->url));
info->progress = task_get_progress(task);
return true;
}
static void http_transfer_progress_cb(retro_task_t *task)
{
2019-05-22 07:55:10 +00:00
#ifdef RARCH_INTERNAL
if (task)
video_display_server_set_window_progress(task->progress, task->finished);
#endif
}
2017-05-09 18:10:58 +00:00
static void* task_push_http_transfer_generic(
struct http_connection_t *conn,
const char *url, bool mute, const char *type,
2016-02-09 16:47:04 +00:00
retro_task_callback_t cb, void *user_data)
2015-07-08 19:35:24 +00:00
{
2016-01-28 08:57:55 +00:00
task_finder_data_t find_data;
2016-10-27 08:19:04 +00:00
char tmp[255];
size_t buf_pos = 0;
const char *s = NULL;
2019-05-22 07:55:10 +00:00
retro_task_t *t = NULL;
http_handle_t *http = NULL;
2015-07-08 19:35:24 +00:00
2019-05-22 07:55:10 +00:00
tmp[0] = '\0';
2015-07-08 19:38:22 +00:00
2019-05-22 07:55:10 +00:00
find_data.func = task_http_finder;
find_data.userdata = (void*)url;
2016-01-28 08:57:55 +00:00
/* Concurrent download of the same file is not allowed */
2017-05-14 18:43:39 +00:00
if (task_queue_find(&find_data))
2016-05-05 18:32:36 +00:00
return NULL;
2015-11-23 14:13:26 +00:00
if (!conn)
2016-05-05 18:32:36 +00:00
return NULL;
2015-11-23 14:13:26 +00:00
http = (http_handle_t*)calloc(1, sizeof(*http));
if (!http)
goto error;
2015-11-23 14:13:26 +00:00
http->connection.handle = conn;
http->connection.cb = &cb_http_conn_default;
if (type)
strlcpy(http->connection.elem1, type, sizeof(http->connection.elem1));
strlcpy(http->connection.url, url, sizeof(http->connection.url));
http->status = HTTP_STATUS_CONNECTION_TRANSFER;
t = task_init();
if (!t)
goto error;
2016-05-27 16:14:47 +00:00
t->handler = task_http_transfer_handler;
t->state = http;
2016-05-27 03:06:03 +00:00
t->mute = mute;
t->callback = cb;
t->progress_cb = http_transfer_progress_cb;
t->user_data = user_data;
t->progress = -1;
if (user_data)
s = ((file_transfer_t*)user_data)->path;
else
s = url;
buf_pos = strlcpy(tmp,
msg_hash_to_str(MSG_DOWNLOADING), sizeof(tmp));
STRLCAT_CONST_INCR(tmp, buf_pos, " ", sizeof(tmp));
if (strstr(s, ".index"))
strlcat(tmp, msg_hash_to_str(MSG_INDEX_FILE), sizeof(tmp));
else
strlcat(tmp, s, sizeof(tmp));
2016-02-03 16:41:04 +00:00
t->title = strdup(tmp);
2015-11-23 14:13:26 +00:00
2017-05-14 18:43:39 +00:00
task_queue_push(t);
2015-11-23 14:13:26 +00:00
2016-05-05 18:32:36 +00:00
return t;
error:
if (conn)
net_http_connection_free(conn);
if (http)
free(http);
2016-05-05 18:32:36 +00:00
return NULL;
2015-07-08 19:35:24 +00:00
}
2017-05-09 18:10:58 +00:00
void* task_push_http_transfer(const char *url, bool mute,
const char *type,
2018-03-16 02:16:49 +00:00
retro_task_callback_t cb, void *user_data)
{
2019-05-22 07:55:10 +00:00
if (string_is_empty(url))
return NULL;
return task_push_http_transfer_generic(
net_http_connection_new(url, "GET", NULL),
url, mute, type, cb, user_data);
2018-03-16 02:16:49 +00:00
}
2017-05-09 18:10:58 +00:00
void* task_push_http_post_transfer(const char *url,
const char *post_data, bool mute,
2017-03-02 22:36:25 +00:00
const char *type, retro_task_callback_t cb, void *user_data)
{
2019-05-22 07:55:10 +00:00
if (string_is_empty(url))
return NULL;
return task_push_http_transfer_generic(
net_http_connection_new(url, "POST", post_data),
2017-05-09 18:10:58 +00:00
url, mute, type, cb, user_data);
2017-03-02 22:36:25 +00:00
}
task_retriever_info_t *http_task_get_transfer_list(void)
{
task_retriever_data_t retrieve_data;
/* Fill retrieve data */
2016-05-27 16:14:47 +00:00
retrieve_data.handler = task_http_transfer_handler;
retrieve_data.element_size = sizeof(http_transfer_info_t);
2016-05-27 16:14:47 +00:00
retrieve_data.func = task_http_retriever;
/* Build list of current HTTP transfers and return it */
2017-05-14 18:43:39 +00:00
task_queue_retrieve(&retrieve_data);
return retrieve_data.list;
}