Files
scummvm-cursorfix/backends/networking/http/android/networkreadstream-android.cpp
2026-02-02 04:50:13 +01:00

409 lines
14 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program 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 Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
// Allow use of stuff in <time.h> and abort()
#define FORBIDDEN_SYMBOL_EXCEPTION_time_h
#define FORBIDDEN_SYMBOL_EXCEPTION_abort
// Disable printf override in common/forbidden.h to avoid
// clashes with log.h from the Android SDK.
// That header file uses
// __attribute__ ((format(printf, 3, 4)))
// which gets messed up by our override mechanism; this could
// be avoided by either changing the Android SDK to use the equally
// legal and valid
// __attribute__ ((format(__printf__, 3, 4)))
// or by refining our printf override to use a varadic macro
// (which then wouldn't be portable, though).
// Anyway, for now we just disable the printf override globally
// for the Android port
#define FORBIDDEN_SYMBOL_EXCEPTION_printf
#include "backends/platform/android/android.h"
#include "backends/platform/android/jni-android.h"
#include "backends/networking/basic/android/jni.h"
#include "backends/networking/http/android/networkreadstream-android.h"
#include "backends/networking/http/android/connectionmanager-android.h"
#include "common/debug.h"
namespace Networking {
NetworkReadStream *NetworkReadStream::make(const char *url, RequestHeaders *headersList, const Common::String &postFields, bool uploading, bool usingPatch, bool keepAlive, long keepAliveIdle, long keepAliveInterval) {
return new NetworkReadStreamAndroid(url, headersList, postFields, uploading, usingPatch, keepAlive, keepAliveIdle, keepAliveInterval);
}
NetworkReadStream *NetworkReadStream::make(const char *url, RequestHeaders *headersList, const Common::HashMap<Common::String, Common::String> &formFields, const Common::HashMap<Common::String, Common::Path> &formFiles, bool keepAlive, long keepAliveIdle, long keepAliveInterval) {
return new NetworkReadStreamAndroid(url, headersList, formFields, formFiles, keepAlive, keepAliveIdle, keepAliveInterval);
}
NetworkReadStream *NetworkReadStream::make(const char *url, RequestHeaders *headersList, const byte *buffer, uint32 bufferSize, bool uploading, bool usingPatch, bool post, bool keepAlive, long keepAliveIdle, long keepAliveInterval) {
return new NetworkReadStreamAndroid(url, headersList, buffer, bufferSize, uploading, usingPatch, post, keepAlive, keepAliveIdle, keepAliveInterval);
}
void NetworkReadStreamAndroid::gotHeaders(JNIEnv *env, jobject obj, jlong nativePointer, jobjectArray headers) {
NetworkReadStreamAndroid *stream = (NetworkReadStreamAndroid *)nativePointer;
if (!stream) {
return;
}
jsize size = env->GetArrayLength(headers);
assert((size & 1) == 0);
stream->_responseHeadersMap.clear();
for (jsize i = 0; i < size; i += 2) {
jstring key_obj = (jstring)env->GetObjectArrayElement(headers, i);
jstring value_obj = (jstring)env->GetObjectArrayElement(headers, i + 1);
const char *key = env->GetStringUTFChars(key_obj, 0);
const char *value = env->GetStringUTFChars(value_obj, 0);
if (key != nullptr && value != nullptr) {
stream->_responseHeadersMap[key] = value;
}
env->ReleaseStringUTFChars(key_obj, key);
env->ReleaseStringUTFChars(value_obj, value);
env->DeleteLocalRef(key_obj);
env->DeleteLocalRef(value_obj);
}
}
void NetworkReadStreamAndroid::gotData(JNIEnv *env, jobject obj, jlong nativePointer, jbyteArray data, jint size, jint totalSize) {
NetworkReadStreamAndroid *stream = (NetworkReadStreamAndroid *)nativePointer;
if (!stream) {
return;
}
jsize arrSize = env->GetArrayLength(data);
assert(size >= 0 && (jsize)size <= arrSize);
if (size > 0) {
jbyte *dataP = (jbyte *)env->GetPrimitiveArrayCritical(data, 0);
assert(dataP);
stream->_backingStream.write(dataP, size);
stream->_downloaded += size;
env->ReleasePrimitiveArrayCritical(data, dataP, JNI_ABORT);
}
stream->setProgress(stream->_downloaded, totalSize);
}
void NetworkReadStreamAndroid::finished_(JNIEnv *env, jobject obj, jlong nativePointer, jint errorCode, jstring errorMsg) {
NetworkReadStreamAndroid *stream = (NetworkReadStreamAndroid *)nativePointer;
if (!stream) {
return;
}
Common::String errorMsgStr;
if (errorMsg) {
const char *errorMsgP = env->GetStringUTFChars(errorMsg, 0);
if (errorMsgP != 0) {
errorMsgStr = Common::String(errorMsgP);
env->ReleaseStringUTFChars(errorMsg, errorMsgP);
}
}
stream->finished(errorCode, errorMsgStr);
}
static jobjectArray getHeaders(JNIEnv *env, RequestHeaders *headersList) {
if (!headersList) {
return nullptr;
}
jclass stringClass = env->FindClass("java/lang/String");
jobjectArray array = env->NewObjectArray(headersList->size() * 2, stringClass, nullptr);
env->DeleteLocalRef(stringClass);
int i = 0;
for (const Common::String &header : *headersList) {
// Find the colon separator
uint colonPos = header.findFirstOf(':');
if (colonPos == Common::String::npos) {
warning("NetworkReadStreamAndroid: Malformed header (no colon): %s", header.c_str());
continue;
}
// Split into key and value parts
Common::String key = header.substr(0, colonPos);
Common::String value = header.substr(colonPos + 1);
// Trim whitespace from key and value
key.trim();
value.trim();
jobject key_obj = env->NewStringUTF(key.c_str());
jobject value_obj = env->NewStringUTF(value.c_str());
// Store key and value as separate strings
env->SetObjectArrayElement(array, i, key_obj);
env->SetObjectArrayElement(array, i + 1, value_obj);
env->DeleteLocalRef(key_obj);
env->DeleteLocalRef(value_obj);
i += 2;
}
return array;
}
static jobjectArray getFormFields(JNIEnv *env, const Common::HashMap<Common::String, Common::String> &map) {
jclass stringClass = env->FindClass("java/lang/String");
jobjectArray array = env->NewObjectArray(map.size() * 2, stringClass, nullptr);
env->DeleteLocalRef(stringClass);
int i = 0;
for (const Common::HashMap<Common::String, Common::String>::Node &entry : map) {
jobject key_obj = env->NewStringUTF(entry._key.c_str());
jobject value_obj = env->NewStringUTF(entry._value.c_str());
// Store key and value as separate strings
env->SetObjectArrayElement(array, i, key_obj);
env->SetObjectArrayElement(array, i + 1, value_obj);
env->DeleteLocalRef(key_obj);
env->DeleteLocalRef(value_obj);
}
return array;
}
static jobjectArray getFormFiles(JNIEnv *env, const Common::HashMap<Common::String, Common::Path> &map) {
jclass stringClass = env->FindClass("java/lang/String");
jobjectArray array = env->NewObjectArray(map.size() * 2, stringClass, nullptr);
env->DeleteLocalRef(stringClass);
int i = 0;
for (const Common::HashMap<Common::String, Common::Path>::Node &entry : map) {
jobject key_obj = env->NewStringUTF(entry._key.c_str());
jobject value_obj = env->NewStringUTF(entry._value.toString('/').c_str());
// Store key and value as separate strings
env->SetObjectArrayElement(array, i, key_obj);
env->SetObjectArrayElement(array, i + 1, value_obj);
env->DeleteLocalRef(key_obj);
env->DeleteLocalRef(value_obj);
}
return array;
}
NetworkReadStreamAndroid::NetworkReadStreamAndroid(const char *url, RequestHeaders *headersList, const Common::String &postFields,
bool uploading, bool usingPatch, bool keepAlive, long keepAliveIdle, long keepAliveInterval) :
NetworkReadStream(keepAlive, keepAliveIdle, keepAliveInterval), _request(nullptr) {
if (!reuse(url, headersList, postFields, uploading, usingPatch)) {
abort();
}
}
NetworkReadStreamAndroid::NetworkReadStreamAndroid(const char *url, RequestHeaders *headersList, const Common::HashMap<Common::String, Common::String> &formFields,
const Common::HashMap<Common::String, Common::Path> &formFiles, bool keepAlive, long keepAliveIdle, long keepAliveInterval) :
NetworkReadStream(keepAlive, keepAliveIdle, keepAliveInterval), _request(nullptr) {
if (!reuse(url, headersList, formFields, formFiles)) {
abort();
}
}
NetworkReadStreamAndroid::NetworkReadStreamAndroid(const char *url, RequestHeaders *headersList, const byte *buffer, uint32 bufferSize,
bool uploading, bool usingPatch, bool post, bool keepAlive, long keepAliveIdle, long keepAliveInterval) :
NetworkReadStream(keepAlive, keepAliveIdle, keepAliveInterval), _request(nullptr) {
if (!reuse(url, headersList, buffer, bufferSize, uploading, usingPatch, post)) {
abort();
}
}
bool NetworkReadStreamAndroid::reuse(const char *url, RequestHeaders *headersList, const Common::String &postFields, bool uploading, bool usingPatch) {
JNIEnv *env = JNI::getEnv();
resetStream(env);
jstring url_obj = env->NewStringUTF(url);
jobjectArray headers_obj = getHeaders(env, headersList);
jbyteArray uploadBuffer_obj = env->NewByteArray(postFields.size());
env->SetByteArrayRegion(uploadBuffer_obj, 0, postFields.size(), (const jbyte *)postFields.c_str());
jobject obj = env->NewObject(NetJNI::_CLS_HTTPRequest, NetJNI::_MID_request_bufinit, (jlong)this, url_obj, headers_obj, uploadBuffer_obj, uploading, usingPatch, false);
env->DeleteLocalRef(uploadBuffer_obj);
env->DeleteLocalRef(headers_obj);
env->DeleteLocalRef(url_obj);
if (env->ExceptionCheck()) {
LOGE("HTTPRequest::<init> failed");
env->ExceptionDescribe();
env->ExceptionClear();
return false;
}
_request = env->NewGlobalRef(obj);
env->DeleteLocalRef(obj);
dynamic_cast<ConnectionManagerAndroid &>(ConnMan).registerRequest(env, _request);
return true;
}
bool NetworkReadStreamAndroid::reuse(const char *url, RequestHeaders *headersList, const Common::HashMap<Common::String, Common::String> &formFields, const Common::HashMap<Common::String, Common::Path> &formFiles) {
JNIEnv *env = JNI::getEnv();
resetStream(env);
jstring url_obj = env->NewStringUTF(url);
jobjectArray headers_obj = getHeaders(env, headersList);
jobjectArray formFields_obj = getFormFields(env, formFields);
jobjectArray formFiles_obj = getFormFiles(env, formFiles);
jobject obj = env->NewObject(NetJNI::_CLS_HTTPRequest, NetJNI::_MID_request_forminit, (jlong)this, url_obj, headers_obj, formFields_obj, formFiles_obj);
env->DeleteLocalRef(formFiles_obj);
env->DeleteLocalRef(formFields_obj);
env->DeleteLocalRef(headers_obj);
env->DeleteLocalRef(url_obj);
if (env->ExceptionCheck()) {
LOGE("HTTPRequest::<init> failed");
env->ExceptionDescribe();
env->ExceptionClear();
return false;
}
_request = env->NewGlobalRef(obj);
env->DeleteLocalRef(obj);
dynamic_cast<ConnectionManagerAndroid &>(ConnMan).registerRequest(env, _request);
return true;
}
bool NetworkReadStreamAndroid::reuse(const char *url, RequestHeaders *headersList, const byte *buffer, uint32 bufferSize, bool uploading, bool usingPatch, bool post) {
JNIEnv *env = JNI::getEnv();
resetStream(env);
jstring url_obj = env->NewStringUTF(url);
jobjectArray headers_obj = getHeaders(env, headersList);
jbyteArray uploadBuffer_obj = env->NewByteArray(bufferSize);
env->SetByteArrayRegion(uploadBuffer_obj, 0, bufferSize, (const jbyte *)buffer);
jobject obj = env->NewObject(NetJNI::_CLS_HTTPRequest, NetJNI::_MID_request_bufinit, (jlong)this, url_obj, headers_obj, uploadBuffer_obj, uploading, usingPatch, false);
env->DeleteLocalRef(uploadBuffer_obj);
env->DeleteLocalRef(headers_obj);
env->DeleteLocalRef(url_obj);
if (env->ExceptionCheck()) {
LOGE("HTTPRequest::<init> failed");
env->ExceptionDescribe();
env->ExceptionClear();
return false;
}
_request = env->NewGlobalRef(obj);
env->DeleteLocalRef(obj);
dynamic_cast<ConnectionManagerAndroid &>(ConnMan).registerRequest(env, _request);
return true;
}
NetworkReadStreamAndroid::~NetworkReadStreamAndroid() {
JNIEnv *env = JNI::getEnv();
env->CallVoidMethod(_request, NetJNI::_MID_request_cancel);
if (env->ExceptionCheck()) {
LOGE("HTTPRequest::cancel failed");
env->ExceptionDescribe();
env->ExceptionClear();
}
env->DeleteGlobalRef(_request);
}
void NetworkReadStreamAndroid::finished(int errorCode, const Common::String &errorMsg) {
_requestComplete = true;
_errorCode = errorCode;
_errorMsg = errorMsg;
if (_errorCode >= 200 && _errorCode < 300) {
debug(9, "NetworkReadStreamAndroid: %s - Request succeeded", currentLocation().c_str());
} else {
warning("NetworkReadStreamAndroid: %s - Request failed (%d - %s)", currentLocation().c_str(), _errorCode, _errorMsg.c_str());
}
}
void NetworkReadStreamAndroid::resetStream(JNIEnv *env) {
_eos = _requestComplete = false;
_progressDownloaded = _progressTotal = 0;
if (_request) {
env->CallVoidMethod(_request, NetJNI::_MID_request_cancel);
if (env->ExceptionCheck()) {
LOGE("HTTPRequest::cancel failed");
env->ExceptionDescribe();
env->ExceptionClear();
}
env->DeleteGlobalRef(_request);
_request = nullptr;
}
_responseHeadersMap.clear();
_downloaded = 0;
_errorCode = 0;
_errorMsg.clear();
}
Common::String NetworkReadStreamAndroid::currentLocation() const {
if (!_request) {
return Common::String();
}
JNIEnv *env = JNI::getEnv();
jstring location_obj = (jstring)env->CallObjectMethod(_request, NetJNI::_MID_request_getURL);
if (env->ExceptionCheck()) {
LOGE("HTTPRequest::getURL failed");
env->ExceptionDescribe();
env->ExceptionClear();
return Common::String();
}
uint length = env->GetStringLength(location_obj);
if (!length) {
env->DeleteLocalRef(location_obj);
return Common::String();
}
const char *location_ptr = env->GetStringUTFChars(location_obj, 0);
if (!location_ptr) {
env->DeleteLocalRef(location_obj);
return Common::String();
}
Common::String result(location_ptr, length);
env->ReleaseStringUTFChars(location_obj, location_ptr);
env->DeleteLocalRef(location_obj);
return result;
}
} // End of namespace Networking