mirror of
https://gitlab.com/Theopse/fbi-i18n-zh.git
synced 2025-04-06 03:58:02 +08:00
Fix reporting cancellation in xfer info callback, hide HTTPC functions.
This commit is contained in:
parent
4d29abe43e
commit
82b82a752c
@ -16,10 +16,11 @@
|
|||||||
#define MAKE_HTTP_USER_AGENT(major, minor, micro) MAKE_HTTP_USER_AGENT_(major, minor, micro)
|
#define MAKE_HTTP_USER_AGENT(major, minor, micro) MAKE_HTTP_USER_AGENT_(major, minor, micro)
|
||||||
#define HTTP_USER_AGENT MAKE_HTTP_USER_AGENT(VERSION_MAJOR, VERSION_MINOR, VERSION_MICRO)
|
#define HTTP_USER_AGENT MAKE_HTTP_USER_AGENT(VERSION_MAJOR, VERSION_MINOR, VERSION_MICRO)
|
||||||
|
|
||||||
|
#define HTTP_MAX_REDIRECTS 50
|
||||||
#define HTTP_TIMEOUT_SEC 15
|
#define HTTP_TIMEOUT_SEC 15
|
||||||
#define HTTP_TIMEOUT_NS ((u64) HTTP_TIMEOUT_SEC * 1000000000)
|
#define HTTP_TIMEOUT_NS ((u64) HTTP_TIMEOUT_SEC * 1000000000)
|
||||||
|
|
||||||
struct http_context_s {
|
struct httpc_context_s {
|
||||||
httpcContext httpc;
|
httpcContext httpc;
|
||||||
|
|
||||||
bool compressed;
|
bool compressed;
|
||||||
@ -28,7 +29,9 @@ struct http_context_s {
|
|||||||
u32 bufferSize;
|
u32 bufferSize;
|
||||||
};
|
};
|
||||||
|
|
||||||
void http_resolve_redirect(char* oldUrl, const char* redirectTo, size_t size) {
|
typedef struct httpc_context_s* httpc_context;
|
||||||
|
|
||||||
|
static void httpc_resolve_redirect(char* oldUrl, const char* redirectTo, size_t size) {
|
||||||
if(size > 0) {
|
if(size > 0) {
|
||||||
if(redirectTo[0] == '/') {
|
if(redirectTo[0] == '/') {
|
||||||
char* baseEnd = oldUrl;
|
char* baseEnd = oldUrl;
|
||||||
@ -57,37 +60,25 @@ void http_resolve_redirect(char* oldUrl, const char* redirectTo, size_t size) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Result http_open(http_context* context, const char* url, bool userAgent) {
|
static Result httpc_open(httpc_context* context, const char* url, bool userAgent) {
|
||||||
return http_open_ranged(context, url, userAgent, 0, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
Result http_open_ranged(http_context* context, const char* url, bool userAgent, u32 rangeStart, u32 rangeEnd) {
|
|
||||||
if(url == NULL) {
|
if(url == NULL) {
|
||||||
return R_APP_INVALID_ARGUMENT;
|
return R_APP_INVALID_ARGUMENT;
|
||||||
}
|
}
|
||||||
|
|
||||||
Result res = 0;
|
Result res = 0;
|
||||||
|
|
||||||
http_context ctx = (http_context) calloc(1, sizeof(struct http_context_s));
|
httpc_context ctx = (httpc_context) calloc(1, sizeof(struct httpc_context_s));
|
||||||
if(ctx != NULL) {
|
if(ctx != NULL) {
|
||||||
char currUrl[1024];
|
char currUrl[1024];
|
||||||
string_copy(currUrl, url, sizeof(currUrl));
|
string_copy(currUrl, url, sizeof(currUrl));
|
||||||
|
|
||||||
char range[64];
|
|
||||||
if(rangeEnd > rangeStart) {
|
|
||||||
snprintf(range, sizeof(range), "%lu-%lu", rangeStart, rangeEnd);
|
|
||||||
} else {
|
|
||||||
snprintf(range, sizeof(range), "%lu-", rangeStart);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool resolved = false;
|
bool resolved = false;
|
||||||
u32 redirectCount = 0;
|
u32 redirectCount = 0;
|
||||||
while(R_SUCCEEDED(res) && !resolved && redirectCount < 32) {
|
while(R_SUCCEEDED(res) && !resolved && redirectCount < HTTP_MAX_REDIRECTS) {
|
||||||
if(R_SUCCEEDED(res = httpcOpenContext(&ctx->httpc, HTTPC_METHOD_GET, currUrl, 1))) {
|
if(R_SUCCEEDED(res = httpcOpenContext(&ctx->httpc, HTTPC_METHOD_GET, currUrl, 1))) {
|
||||||
u32 response = 0;
|
u32 response = 0;
|
||||||
if(R_SUCCEEDED(res = httpcSetSSLOpt(&ctx->httpc, SSLCOPT_DisableVerify))
|
if(R_SUCCEEDED(res = httpcSetSSLOpt(&ctx->httpc, SSLCOPT_DisableVerify))
|
||||||
&& (!userAgent || R_SUCCEEDED(res = httpcAddRequestHeaderField(&ctx->httpc, "User-Agent", HTTP_USER_AGENT)))
|
&& (!userAgent || R_SUCCEEDED(res = httpcAddRequestHeaderField(&ctx->httpc, "User-Agent", HTTP_USER_AGENT)))
|
||||||
&& (rangeStart == 0 || R_SUCCEEDED(res = httpcAddRequestHeaderField(&ctx->httpc, "Range", range)))
|
|
||||||
&& R_SUCCEEDED(res = httpcAddRequestHeaderField(&ctx->httpc, "Accept-Encoding", "gzip, deflate"))
|
&& R_SUCCEEDED(res = httpcAddRequestHeaderField(&ctx->httpc, "Accept-Encoding", "gzip, deflate"))
|
||||||
&& R_SUCCEEDED(res = httpcSetKeepAlive(&ctx->httpc, HTTPC_KEEPALIVE_ENABLED))
|
&& R_SUCCEEDED(res = httpcSetKeepAlive(&ctx->httpc, HTTPC_KEEPALIVE_ENABLED))
|
||||||
&& R_SUCCEEDED(res = httpcBeginRequest(&ctx->httpc))
|
&& R_SUCCEEDED(res = httpcBeginRequest(&ctx->httpc))
|
||||||
@ -100,7 +91,7 @@ Result http_open_ranged(http_context* context, const char* url, bool userAgent,
|
|||||||
if(R_SUCCEEDED(res = httpcGetResponseHeader(&ctx->httpc, "Location", redirectTo, sizeof(redirectTo)))) {
|
if(R_SUCCEEDED(res = httpcGetResponseHeader(&ctx->httpc, "Location", redirectTo, sizeof(redirectTo)))) {
|
||||||
httpcCloseContext(&ctx->httpc);
|
httpcCloseContext(&ctx->httpc);
|
||||||
|
|
||||||
http_resolve_redirect(currUrl, redirectTo, sizeof(currUrl));
|
httpc_resolve_redirect(currUrl, redirectTo, sizeof(currUrl));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
resolved = true;
|
resolved = true;
|
||||||
@ -152,7 +143,7 @@ Result http_open_ranged(http_context* context, const char* url, bool userAgent,
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
Result http_close(http_context context) {
|
static Result httpc_close(httpc_context context) {
|
||||||
if(context == NULL) {
|
if(context == NULL) {
|
||||||
return R_APP_INVALID_ARGUMENT;
|
return R_APP_INVALID_ARGUMENT;
|
||||||
}
|
}
|
||||||
@ -166,7 +157,7 @@ Result http_close(http_context context) {
|
|||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
Result http_get_size(http_context context, u32* size) {
|
static Result httpc_get_size(httpc_context context, u32* size) {
|
||||||
if(context == NULL || size == NULL) {
|
if(context == NULL || size == NULL) {
|
||||||
return R_APP_INVALID_ARGUMENT;
|
return R_APP_INVALID_ARGUMENT;
|
||||||
}
|
}
|
||||||
@ -174,37 +165,7 @@ Result http_get_size(http_context context, u32* size) {
|
|||||||
return httpcGetDownloadSizeState(&context->httpc, NULL, size);
|
return httpcGetDownloadSizeState(&context->httpc, NULL, size);
|
||||||
}
|
}
|
||||||
|
|
||||||
Result http_get_file_name(http_context context, char* out, u32 size) {
|
static Result httpc_read(httpc_context context, u32* bytesRead, void* buffer, u32 size) {
|
||||||
if(context == NULL || out == NULL) {
|
|
||||||
return R_APP_INVALID_ARGUMENT;
|
|
||||||
}
|
|
||||||
|
|
||||||
Result res = 0;
|
|
||||||
|
|
||||||
char* header = (char*) calloc(1, size + 64);
|
|
||||||
if(header != NULL) {
|
|
||||||
if(R_SUCCEEDED(res = httpcGetResponseHeader(&context->httpc, "Content-Disposition", header, size + 64))) {
|
|
||||||
char* start = strstr(header, "filename=");
|
|
||||||
if(start != NULL) {
|
|
||||||
char format[32];
|
|
||||||
snprintf(format, sizeof(format), "filename=\"%%%lu[^\"]\"", size);
|
|
||||||
if(sscanf(start, format, out) != 1) {
|
|
||||||
res = R_APP_BAD_DATA;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
res = R_APP_BAD_DATA;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
free(header);
|
|
||||||
} else {
|
|
||||||
res = R_APP_OUT_OF_MEMORY;
|
|
||||||
}
|
|
||||||
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
Result http_read(http_context context, u32* bytesRead, void* buffer, u32 size) {
|
|
||||||
if(context == NULL || buffer == NULL) {
|
if(context == NULL || buffer == NULL) {
|
||||||
return R_APP_INVALID_ARGUMENT;
|
return R_APP_INVALID_ARGUMENT;
|
||||||
}
|
}
|
||||||
@ -312,7 +273,7 @@ int http_curl_xfer_info_callback(void* clientp, curl_off_t dltotal, curl_off_t d
|
|||||||
http_curl_data* curlData = (http_curl_data*) clientp;
|
http_curl_data* curlData = (http_curl_data*) clientp;
|
||||||
|
|
||||||
if(curlData->checkRunning != NULL && R_FAILED(curlData->res = curlData->checkRunning(curlData->userData))) {
|
if(curlData->checkRunning != NULL && R_FAILED(curlData->res = curlData->checkRunning(curlData->userData))) {
|
||||||
return 0;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(curlData->progress != NULL) {
|
if(curlData->progress != NULL) {
|
||||||
@ -329,10 +290,10 @@ Result http_download_callback(const char* url, u32 bufferSize, void* userData, R
|
|||||||
|
|
||||||
void* buf = malloc(bufferSize);
|
void* buf = malloc(bufferSize);
|
||||||
if(buf != NULL) {
|
if(buf != NULL) {
|
||||||
http_context context = NULL;
|
httpc_context context = NULL;
|
||||||
if(R_SUCCEEDED(res = http_open(&context, url, true))) {
|
if(R_SUCCEEDED(res = httpc_open(&context, url, true))) {
|
||||||
u32 dlSize = 0;
|
u32 dlSize = 0;
|
||||||
if(R_SUCCEEDED(res = http_get_size(context, &dlSize))) {
|
if(R_SUCCEEDED(res = httpc_get_size(context, &dlSize))) {
|
||||||
if(progress != NULL) {
|
if(progress != NULL) {
|
||||||
progress(userData, dlSize, 0);
|
progress(userData, dlSize, 0);
|
||||||
}
|
}
|
||||||
@ -341,7 +302,7 @@ Result http_download_callback(const char* url, u32 bufferSize, void* userData, R
|
|||||||
u32 currSize = 0;
|
u32 currSize = 0;
|
||||||
while(total < dlSize
|
while(total < dlSize
|
||||||
&& (checkRunning == NULL || R_SUCCEEDED(res = checkRunning(userData)))
|
&& (checkRunning == NULL || R_SUCCEEDED(res = checkRunning(userData)))
|
||||||
&& R_SUCCEEDED(res = http_read(context, &currSize, buf, bufferSize))
|
&& R_SUCCEEDED(res = httpc_read(context, &currSize, buf, bufferSize))
|
||||||
&& R_SUCCEEDED(res = callback(userData, buf, currSize))) {
|
&& R_SUCCEEDED(res = callback(userData, buf, currSize))) {
|
||||||
if(progress != NULL) {
|
if(progress != NULL) {
|
||||||
progress(userData, dlSize, total);
|
progress(userData, dlSize, total);
|
||||||
@ -350,7 +311,7 @@ Result http_download_callback(const char* url, u32 bufferSize, void* userData, R
|
|||||||
total += currSize;
|
total += currSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
Result closeRes = http_close(context);
|
Result closeRes = httpc_close(context);
|
||||||
if(R_SUCCEEDED(res)) {
|
if(R_SUCCEEDED(res)) {
|
||||||
res = closeRes;
|
res = closeRes;
|
||||||
}
|
}
|
||||||
@ -363,18 +324,18 @@ Result http_download_callback(const char* url, u32 bufferSize, void* userData, R
|
|||||||
http_curl_data curlData = {bufferSize, userData, callback, checkRunning, progress, buf, 0, 0};
|
http_curl_data curlData = {bufferSize, userData, callback, checkRunning, progress, buf, 0, 0};
|
||||||
|
|
||||||
curl_easy_setopt(curl, CURLOPT_URL, url);
|
curl_easy_setopt(curl, CURLOPT_URL, url);
|
||||||
curl_easy_setopt(curl, CURLOPT_USERAGENT, HTTP_USER_AGENT);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_ACCEPT_ENCODING, "gzip");
|
|
||||||
curl_easy_setopt(curl, CURLOPT_BUFFERSIZE, bufferSize);
|
curl_easy_setopt(curl, CURLOPT_BUFFERSIZE, bufferSize);
|
||||||
curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, HTTP_TIMEOUT_SEC);
|
curl_easy_setopt(curl, CURLOPT_ACCEPT_ENCODING, "");
|
||||||
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1);
|
curl_easy_setopt(curl, CURLOPT_USERAGENT, HTTP_USER_AGENT);
|
||||||
curl_easy_setopt(curl, CURLOPT_MAXREDIRS, 50);
|
curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, (long) HTTP_TIMEOUT_SEC);
|
||||||
curl_easy_setopt(curl, CURLOPT_FAILONERROR, 1);
|
curl_easy_setopt(curl, CURLOPT_MAXREDIRS, (long) HTTP_MAX_REDIRECTS);
|
||||||
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0);
|
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1L);
|
||||||
curl_easy_setopt(curl, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_2TLS);
|
curl_easy_setopt(curl, CURLOPT_FAILONERROR, 1L);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_HTTP_VERSION, (long) CURL_HTTP_VERSION_2TLS);
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, http_curl_write_callback);
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, http_curl_write_callback);
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void*) &curlData);
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, (void*) &curlData);
|
||||||
curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0);
|
curl_easy_setopt(curl, CURLOPT_NOPROGRESS, 0L);
|
||||||
curl_easy_setopt(curl, CURLOPT_XFERINFOFUNCTION, http_curl_xfer_info_callback);
|
curl_easy_setopt(curl, CURLOPT_XFERINFOFUNCTION, http_curl_xfer_info_callback);
|
||||||
curl_easy_setopt(curl, CURLOPT_XFERINFODATA, (void*) &curlData);
|
curl_easy_setopt(curl, CURLOPT_XFERINFODATA, (void*) &curlData);
|
||||||
|
|
||||||
@ -390,7 +351,7 @@ Result http_download_callback(const char* url, u32 bufferSize, void* userData, R
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(ret != CURLE_OK) {
|
if(ret != CURLE_OK) {
|
||||||
if(ret == CURLE_WRITE_ERROR) {
|
if(ret == CURLE_WRITE_ERROR || ret == CURLE_ABORTED_BY_CALLBACK) {
|
||||||
res = curlData.res;
|
res = curlData.res;
|
||||||
} else if(ret == CURLE_HTTP_RETURNED_ERROR) {
|
} else if(ret == CURLE_HTTP_RETURNED_ERROR) {
|
||||||
long responseCode = 0;
|
long responseCode = 0;
|
||||||
|
@ -1,14 +1,5 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
typedef struct http_context_s* http_context;
|
|
||||||
|
|
||||||
Result http_open(http_context* context, const char* url, bool userAgent);
|
|
||||||
Result http_open_ranged(http_context* context, const char* url, bool userAgent, u32 rangeStart, u32 rangeEnd);
|
|
||||||
Result http_close(http_context context);
|
|
||||||
Result http_get_size(http_context context, u32* size);
|
|
||||||
Result http_get_file_name(http_context context, char* out, u32 size);
|
|
||||||
Result http_read(http_context context, u32* bytesRead, void* buffer, u32 size);
|
|
||||||
|
|
||||||
Result http_download_callback(const char* url, u32 bufferSize, void* userData, Result (*callback)(void* userData, void* buffer, size_t size),
|
Result http_download_callback(const char* url, u32 bufferSize, void* userData, Result (*callback)(void* userData, void* buffer, size_t size),
|
||||||
Result (*checkRunning)(void* userData),
|
Result (*checkRunning)(void* userData),
|
||||||
Result (*progress)(void* userData, u64 total, u64 curr));
|
Result (*progress)(void* userData, u64 total, u64 curr));
|
||||||
|
@ -151,16 +151,13 @@ static Result action_install_url_open_dst(void* data, u32 index, void* initialRe
|
|||||||
string_copy(installData->currPath, installData->paths[index], FILE_PATH_MAX);
|
string_copy(installData->currPath, installData->paths[index], FILE_PATH_MAX);
|
||||||
} else {
|
} else {
|
||||||
char filename[FILE_NAME_MAX];
|
char filename[FILE_NAME_MAX];
|
||||||
// TODO
|
string_get_path_file(filename, installData->urls[index], FILE_NAME_MAX);
|
||||||
//if(R_FAILED(http_get_file_name(installData->currContext, filename, FILE_NAME_MAX))) {
|
|
||||||
string_get_path_file(filename, installData->urls[index], FILE_NAME_MAX);
|
|
||||||
//}
|
|
||||||
|
|
||||||
char name[FILE_NAME_MAX];
|
char name[FILE_NAME_MAX];
|
||||||
string_get_file_name(name, filename, FILE_NAME_MAX);
|
string_get_file_name(name, filename, FILE_NAME_MAX);
|
||||||
|
|
||||||
snprintf(dir, FILE_PATH_MAX, "/3ds/%s/", name);
|
snprintf(dir, FILE_PATH_MAX, "/3ds/%s/", name);
|
||||||
snprintf(installData->currPath, FILE_PATH_MAX, "/3ds/%s/%s", name, filename);
|
snprintf(installData->currPath, FILE_PATH_MAX, "/3ds/%s/%s.3dsx", name, name);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(R_SUCCEEDED(res = fs_ensure_dir(sdmcArchive, "/3ds/")) && R_SUCCEEDED(res = fs_ensure_dir(sdmcArchive, dir))) {
|
if(R_SUCCEEDED(res = fs_ensure_dir(sdmcArchive, "/3ds/")) && R_SUCCEEDED(res = fs_ensure_dir(sdmcArchive, dir))) {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user