mirror of
https://github.com/Relintai/pandemonium_engine.git
synced 2025-01-10 12:59:40 +01:00
Relintai
f86c6440fb
- Added move_file() and is_file_moved() helper methods to WebServerRequest. - get_file_length() in WebServerRequest now returns uint64_t. - Removed parse_files() from WebServerRequest. It's doesn't work well with how the http protocol works.
431 lines
12 KiB
C++
431 lines
12 KiB
C++
/*************************************************************************/
|
|
/* simple_web_server_request.cpp */
|
|
/*************************************************************************/
|
|
/* This file is part of: */
|
|
/* PANDEMONIUM ENGINE */
|
|
/* https://github.com/Relintai/pandemonium_engine */
|
|
/*************************************************************************/
|
|
/* Copyright (c) 2022-present Péter Magyar. */
|
|
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
|
|
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
|
|
/* */
|
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
|
/* a copy of this software and associated documentation files (the */
|
|
/* "Software"), to deal in the Software without restriction, including */
|
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
|
/* the following conditions: */
|
|
/* */
|
|
/* The above copyright notice and this permission notice shall be */
|
|
/* included in all copies or substantial portions of the Software. */
|
|
/* */
|
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
|
/*************************************************************************/
|
|
|
|
#include "simple_web_server_request.h"
|
|
|
|
#include "core/object/object.h"
|
|
#include "modules/web/http/web_server.h"
|
|
#include "modules/web/http/web_server_cookie.h"
|
|
|
|
#include "modules/web/http/http_session.h"
|
|
|
|
#include "modules/web/http/http_session_manager.h"
|
|
#include "modules/web/http/web_node.h"
|
|
|
|
#include "http_server_simple.h"
|
|
#include "modules/web/http/web_permission.h"
|
|
|
|
#include "core/os/dir_access.h"
|
|
|
|
String SimpleWebServerRequest::get_cookie(const String &key) {
|
|
for (int i = 0; i < _cookies.size(); ++i) {
|
|
const CookieData &d = _cookies[i];
|
|
if (d.key == key) {
|
|
return d.value;
|
|
}
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
HTTPServerEnums::HTTPMethod SimpleWebServerRequest::get_method() const {
|
|
return _method;
|
|
}
|
|
|
|
int SimpleWebServerRequest::get_file_count() const {
|
|
return _files.size();
|
|
}
|
|
String SimpleWebServerRequest::get_file_file_name(const int index) const {
|
|
ERR_FAIL_INDEX_V(index, _files.size(), "");
|
|
|
|
return _files[index].file_name;
|
|
}
|
|
String SimpleWebServerRequest::get_file_key(const int index) const {
|
|
ERR_FAIL_INDEX_V(index, _files.size(), "");
|
|
|
|
return _files[index].file_name;
|
|
}
|
|
uint64_t SimpleWebServerRequest::get_file_length(const int index) const {
|
|
ERR_FAIL_INDEX_V(index, _files.size(), 0);
|
|
|
|
const FileEntry &e = _files[index];
|
|
|
|
if (e.type == FileEntry::FILE_ENTRY_TYPE_MEMORY) {
|
|
return e.data.size();
|
|
}
|
|
|
|
Error err;
|
|
FileAccess *fa = FileAccess::open(e.path, FileAccess::READ, &err);
|
|
|
|
ERR_FAIL_NULL_V(fa, 0);
|
|
|
|
if (err != OK) {
|
|
memdelete(fa);
|
|
return 0;
|
|
}
|
|
|
|
uint64_t len = fa->get_len();
|
|
|
|
memdelete(fa);
|
|
|
|
return len;
|
|
}
|
|
PoolByteArray SimpleWebServerRequest::get_file_data(const int index) const {
|
|
ERR_FAIL_INDEX_V(index, _files.size(), PoolByteArray());
|
|
|
|
const FileEntry &e = _files[index];
|
|
|
|
if (e.type == FileEntry::FILE_ENTRY_TYPE_MEMORY) {
|
|
return e.data;
|
|
}
|
|
|
|
Error err;
|
|
FileAccess *f = FileAccess::open(e.path, FileAccess::READ, &err);
|
|
if (!f) {
|
|
ERR_FAIL_V_MSG(PoolByteArray(), "Can't open file from path '" + String(e.path) + "'.");
|
|
}
|
|
|
|
PoolByteArray data;
|
|
data.resize(f->get_len());
|
|
PoolByteArray::Write w = data.write();
|
|
uint8_t *dptr = w.ptr();
|
|
|
|
f->get_buffer(dptr, data.size());
|
|
memdelete(f);
|
|
|
|
return data;
|
|
}
|
|
String SimpleWebServerRequest::get_file_data_str(const int index) const {
|
|
ERR_FAIL_INDEX_V(index, _files.size(), "");
|
|
|
|
const FileEntry &e = _files[index];
|
|
|
|
if (e.type == FileEntry::FILE_ENTRY_TYPE_MEMORY) {
|
|
PoolByteArray::Read r = _files[index].data.read();
|
|
|
|
String ret = reinterpret_cast<const char *>(r.ptr());
|
|
|
|
r.release();
|
|
|
|
return ret;
|
|
}
|
|
|
|
Error err;
|
|
String data = FileAccess::get_file_as_string(e.path, &err);
|
|
|
|
ERR_FAIL_COND_V_MSG(err != OK, "", vformat("Error accessing file! Error: %d", err));
|
|
|
|
return data;
|
|
}
|
|
Error SimpleWebServerRequest::move_file(const int index, const String &p_dest_file) {
|
|
ERR_FAIL_INDEX_V(index, _files.size(), ERR_INVALID_PARAMETER);
|
|
|
|
DirAccess *dir = DirAccess::create_for_path(p_dest_file.get_base_dir());
|
|
|
|
if (!dir) {
|
|
return ERR_FILE_BAD_PATH;
|
|
}
|
|
|
|
if (dir->file_exists(p_dest_file)) {
|
|
return ERR_ALREADY_IN_USE;
|
|
}
|
|
|
|
const FileEntry &e = _files[index];
|
|
|
|
if (e.type == FileEntry::FILE_ENTRY_TYPE_MEMORY) {
|
|
memdelete(dir);
|
|
|
|
Error err;
|
|
FileAccess *f = FileAccess::open(e.path, FileAccess::WRITE, &err);
|
|
if (!f) {
|
|
return ERR_FILE_BAD_PATH;
|
|
}
|
|
|
|
PoolByteArray::Read r = e.data.read();
|
|
const uint8_t *rptr = r.ptr();
|
|
|
|
f->store_buffer(rptr, e.data.size());
|
|
|
|
return OK;
|
|
}
|
|
|
|
dir->rename(e.path, p_dest_file);
|
|
memdelete(dir);
|
|
|
|
e.moved = true;
|
|
e.path = p_dest_file;
|
|
|
|
return OK;
|
|
}
|
|
bool SimpleWebServerRequest::is_file_moved(const int index) const {
|
|
ERR_FAIL_INDEX_V(index, _files.size(), true);
|
|
|
|
return _files[index].moved;
|
|
}
|
|
|
|
String SimpleWebServerRequest::get_parameter(const String &key) const {
|
|
if (!_post_parameters.has(key)) {
|
|
return get_get_parameter(key);
|
|
}
|
|
|
|
return get_post_parameter(key);
|
|
}
|
|
String SimpleWebServerRequest::get_post_parameter(const String &key) const {
|
|
if (!_post_parameters.has(key)) {
|
|
return "";
|
|
}
|
|
|
|
return _post_parameters[key];
|
|
}
|
|
String SimpleWebServerRequest::get_get_parameter(const String &key) const {
|
|
if (!_get_parameters.has(key)) {
|
|
return "";
|
|
}
|
|
|
|
return _get_parameters[key];
|
|
}
|
|
|
|
void SimpleWebServerRequest::set_post_parameter(const String &key, const String &value) {
|
|
_post_parameters[key] = value;
|
|
}
|
|
void SimpleWebServerRequest::set_get_parameter(const String &key, const String &value) {
|
|
_get_parameters[key] = value;
|
|
}
|
|
|
|
String SimpleWebServerRequest::get_header_parameter(const String &key) const {
|
|
if (!_header_parameters.has(key)) {
|
|
return "";
|
|
}
|
|
|
|
return _header_parameters[key];
|
|
}
|
|
void SimpleWebServerRequest::set_header_parameter(const String &key, const String &value) {
|
|
_header_parameters[key] = value;
|
|
}
|
|
|
|
PoolStringArray SimpleWebServerRequest::get_header_parameter_keys() const {
|
|
PoolStringArray ks;
|
|
|
|
for (const HashMap<String, String>::Element *E = _header_parameters.front(); E; E = E->next) {
|
|
ks.push_back(E->key());
|
|
}
|
|
|
|
return ks;
|
|
}
|
|
PoolStringArray SimpleWebServerRequest::get_parameter_keys() const {
|
|
PoolStringArray g = get_get_parameter_keys();
|
|
|
|
g.append_array(get_post_parameter_keys());
|
|
|
|
return g;
|
|
}
|
|
PoolStringArray SimpleWebServerRequest::get_post_parameter_keys() const {
|
|
PoolStringArray ks;
|
|
|
|
for (const HashMap<String, String>::Element *E = _post_parameters.front(); E; E = E->next) {
|
|
ks.push_back(E->key());
|
|
}
|
|
|
|
return ks;
|
|
}
|
|
PoolStringArray SimpleWebServerRequest::get_get_parameter_keys() const {
|
|
PoolStringArray ks;
|
|
|
|
for (const HashMap<String, String>::Element *E = _get_parameters.front(); E; E = E->next) {
|
|
ks.push_back(E->key());
|
|
}
|
|
|
|
return ks;
|
|
}
|
|
|
|
void SimpleWebServerRequest::send_redirect(const String &location, const HTTPServerEnums::HTTPStatusCode status_code) {
|
|
ERR_FAIL_COND(!_server);
|
|
|
|
_connection->send_redirect(Ref<WebServerRequest>(this), location, status_code);
|
|
}
|
|
|
|
void SimpleWebServerRequest::send() {
|
|
ERR_FAIL_COND(!_server);
|
|
|
|
_connection->send(Ref<WebServerRequest>(this));
|
|
|
|
// if (connection_closed) {
|
|
// SimpleWebServerRequestPool::return_request(this);
|
|
// return;
|
|
// }
|
|
|
|
// SimpleWebServerRequestPool::return_request(this);
|
|
}
|
|
|
|
void SimpleWebServerRequest::send_file(const String &p_file_path) {
|
|
ERR_FAIL_COND(!_connection.is_valid());
|
|
|
|
_connection->send_file(Ref<WebServerRequest>(this), p_file_path);
|
|
|
|
// SimpleWebServerRequestPool::return_request(this);
|
|
}
|
|
|
|
String SimpleWebServerRequest::parser_get_path() {
|
|
return _parser_path;
|
|
}
|
|
|
|
String SimpleWebServerRequest::get_host() const {
|
|
return _host;
|
|
}
|
|
|
|
void SimpleWebServerRequest::add_post_parameter(const String &key, const String &value) {
|
|
_post_parameters[key] = value;
|
|
}
|
|
|
|
void SimpleWebServerRequest::add_get_parameter(const String &key, const String &value) {
|
|
_get_parameters[key] = value;
|
|
}
|
|
|
|
void SimpleWebServerRequest::add_header_parameter(const String &key, const String &value) {
|
|
_header_parameters[key] = value;
|
|
}
|
|
|
|
void SimpleWebServerRequest::set_parser_path(const String &value) {
|
|
//https://www.rfc-editor.org/rfc/rfc3986.txt
|
|
//3.4. Query
|
|
//The query component is indicated by the first question mark ("?") ...
|
|
int qpos = value.find_char('?');
|
|
|
|
if (qpos == -1) {
|
|
_parser_path = value;
|
|
return;
|
|
}
|
|
|
|
_parser_path = value.substr_index(0, qpos);
|
|
|
|
//... and terminated by a number sign ("#") character or by the end of the URI.
|
|
int get_query_end = value.find_char('#', qpos);
|
|
|
|
// ...The characters slash ("/") and question mark ("?") may represent data within the query component.
|
|
// "key=value" pairs and one frequently used value is a reference to another URI, it is sometimes
|
|
// better for usability to avoid percent-encoding those characters.
|
|
|
|
if (get_query_end == -1) {
|
|
get_query_end = value.size() - 1;
|
|
}
|
|
|
|
String get_req_str = value.substr_index(qpos + 1, get_query_end);
|
|
|
|
Vector<String> get_req_params = get_req_str.split("&");
|
|
|
|
for (int i = 0; i < get_req_params.size(); ++i) {
|
|
String cp = get_req_params[i];
|
|
|
|
int eq = cp.find_char('=');
|
|
|
|
if (eq == -1) {
|
|
// skip if ¶m&
|
|
continue;
|
|
}
|
|
|
|
String key = cp.substr_index(0, eq);
|
|
String val = cp.substr_index(eq + 1, cp.length());
|
|
|
|
if (key.empty() || val.empty()) {
|
|
// &=v& or &p=&
|
|
continue;
|
|
}
|
|
|
|
add_get_parameter(key.percent_decode(), val.percent_decode());
|
|
}
|
|
}
|
|
|
|
void SimpleWebServerRequest::set_host(const String &value) {
|
|
_host = value;
|
|
}
|
|
|
|
void SimpleWebServerRequest::add_file_memory(const String &key, const String &file_name, const PoolByteArray &data) {
|
|
FileEntry e;
|
|
e.type = FileEntry::FILE_ENTRY_TYPE_MEMORY;
|
|
e.key = key;
|
|
e.file_name = file_name;
|
|
e.data = data;
|
|
|
|
_files.push_back(e);
|
|
}
|
|
|
|
void SimpleWebServerRequest::add_file_temp_file(const String &key, const String &file_name, const String &path) {
|
|
FileEntry e;
|
|
e.type = FileEntry::FILE_ENTRY_TYPE_TEMP_FILE;
|
|
e.key = key;
|
|
e.file_name = file_name;
|
|
e.path = path;
|
|
|
|
_files.push_back(e);
|
|
}
|
|
|
|
void SimpleWebServerRequest::add_cookie_data(const String &key, const String &value) {
|
|
CookieData d;
|
|
d.key = key;
|
|
d.value = value;
|
|
|
|
_cookies.push_back(d);
|
|
}
|
|
|
|
void SimpleWebServerRequest::set_method(const HTTPServerEnums::HTTPMethod method) {
|
|
_method = method;
|
|
}
|
|
|
|
bool SimpleWebServerRequest::sent() {
|
|
return !_sending_file_fa;
|
|
}
|
|
|
|
SimpleWebServerRequest::SimpleWebServerRequest() {
|
|
_server = nullptr;
|
|
_method = HTTPServerEnums::HTTP_METHOD_GET;
|
|
_sending_file_fa = NULL;
|
|
}
|
|
|
|
SimpleWebServerRequest::~SimpleWebServerRequest() {
|
|
for (int i = 0; i < _files.size(); ++i) {
|
|
const FileEntry &e = _files[i];
|
|
|
|
if (e.type == FileEntry::FILE_ENTRY_TYPE_TEMP_FILE) {
|
|
if (!e.moved && !e.path.empty()) {
|
|
DirAccess *d = DirAccess::create_for_path(e.path.get_base_dir());
|
|
|
|
if (d) {
|
|
d->remove(e.path);
|
|
memdelete(d);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void SimpleWebServerRequest::_bind_methods() {
|
|
}
|