mirror of https://github.com/kiwix/libkiwix.git
219 lines
6.3 KiB
C++
219 lines
6.3 KiB
C++
/*
|
|
* Copyright 2019 Matthieu Gautier<mgautier@kymeria.fr>
|
|
*
|
|
* 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
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
* MA 02110-1301, USA.
|
|
*/
|
|
|
|
|
|
#ifndef KIWIXLIB_SERVER_RESPONSE_H
|
|
#define KIWIXLIB_SERVER_RESPONSE_H
|
|
|
|
#include <string>
|
|
#include <map>
|
|
|
|
#include <mustache.hpp>
|
|
#include "byte_range.h"
|
|
#include "etag.h"
|
|
#include "i18n.h"
|
|
|
|
#include <zim/item.h>
|
|
|
|
extern "C" {
|
|
#include "microhttpd_wrapper.h"
|
|
}
|
|
|
|
namespace zim {
|
|
class Archive;
|
|
} // namespace zim
|
|
|
|
namespace kiwix {
|
|
|
|
class InternalServer;
|
|
class RequestContext;
|
|
|
|
class Response {
|
|
public:
|
|
Response(bool verbose);
|
|
virtual ~Response() = default;
|
|
|
|
static std::unique_ptr<Response> build(const InternalServer& server);
|
|
static std::unique_ptr<Response> build_304(const InternalServer& server, const ETag& etag);
|
|
static std::unique_ptr<Response> build_416(const InternalServer& server, size_t resourceLength);
|
|
static std::unique_ptr<Response> build_redirect(const InternalServer& server, const std::string& redirectUrl);
|
|
|
|
MHD_Result send(const RequestContext& request, MHD_Connection* connection);
|
|
|
|
void set_code(int code) { m_returnCode = code; }
|
|
void set_cacheable() { m_etag.set_option(ETag::CACHEABLE_ENTITY); }
|
|
void set_server_id(const std::string& id) { m_etag.set_server_id(id); }
|
|
void add_header(const std::string& name, const std::string& value) { m_customHeaders[name] = value; }
|
|
|
|
int getReturnCode() const { return m_returnCode; }
|
|
|
|
private: // functions
|
|
virtual MHD_Response* create_mhd_response(const RequestContext& request);
|
|
MHD_Response* create_error_response(const RequestContext& request) const;
|
|
|
|
protected: // data
|
|
bool m_verbose;
|
|
int m_returnCode;
|
|
ByteRange m_byteRange;
|
|
ETag m_etag;
|
|
std::map<std::string, std::string> m_customHeaders;
|
|
|
|
friend class ItemResponse;
|
|
};
|
|
|
|
|
|
class ContentResponse : public Response {
|
|
public:
|
|
ContentResponse(
|
|
const std::string& root,
|
|
bool verbose,
|
|
const std::string& content,
|
|
const std::string& mimetype);
|
|
|
|
static std::unique_ptr<ContentResponse> build(
|
|
const InternalServer& server,
|
|
const std::string& content,
|
|
const std::string& mimetype);
|
|
|
|
static std::unique_ptr<ContentResponse> build(
|
|
const InternalServer& server,
|
|
const std::string& template_str,
|
|
kainjow::mustache::data data,
|
|
const std::string& mimetype);
|
|
|
|
private:
|
|
MHD_Response* create_mhd_response(const RequestContext& request);
|
|
|
|
bool can_compress(const RequestContext& request) const;
|
|
|
|
|
|
private:
|
|
std::string m_root;
|
|
std::string m_content;
|
|
std::string m_mimeType;
|
|
};
|
|
|
|
class ContentResponseBlueprint
|
|
{
|
|
public: // functions
|
|
ContentResponseBlueprint(const InternalServer* server,
|
|
const RequestContext* request,
|
|
int httpStatusCode,
|
|
const std::string& mimeType,
|
|
const std::string& templateStr)
|
|
: m_server(*server)
|
|
, m_request(*request)
|
|
, m_httpStatusCode(httpStatusCode)
|
|
, m_mimeType(mimeType)
|
|
, m_template(templateStr)
|
|
{}
|
|
|
|
virtual ~ContentResponseBlueprint() = default;
|
|
|
|
operator std::unique_ptr<ContentResponse>() const
|
|
{
|
|
return generateResponseObject();
|
|
}
|
|
|
|
operator std::unique_ptr<Response>() const
|
|
{
|
|
return operator std::unique_ptr<ContentResponse>();
|
|
}
|
|
|
|
|
|
protected: // functions
|
|
std::string getMessage(const std::string& msgId) const;
|
|
virtual std::unique_ptr<ContentResponse> generateResponseObject() const;
|
|
|
|
public: //data
|
|
const InternalServer& m_server;
|
|
const RequestContext& m_request;
|
|
const int m_httpStatusCode;
|
|
const std::string m_mimeType;
|
|
const std::string m_template;
|
|
kainjow::mustache::data m_data;
|
|
};
|
|
|
|
struct HTTPErrorResponse : ContentResponseBlueprint
|
|
{
|
|
HTTPErrorResponse(const InternalServer& server,
|
|
const RequestContext& request,
|
|
int httpStatusCode,
|
|
const std::string& pageTitleMsgId,
|
|
const std::string& headingMsgId,
|
|
const std::string& cssUrl = "");
|
|
|
|
HTTPErrorResponse& operator+(const std::string& msg);
|
|
HTTPErrorResponse& operator+(const ParameterizedMessage& errorDetails);
|
|
HTTPErrorResponse& operator+=(const ParameterizedMessage& errorDetails);
|
|
};
|
|
|
|
class UrlNotFoundMsg {};
|
|
|
|
extern const UrlNotFoundMsg urlNotFoundMsg;
|
|
|
|
struct HTTP404Response : HTTPErrorResponse
|
|
{
|
|
HTTP404Response(const InternalServer& server,
|
|
const RequestContext& request);
|
|
|
|
using HTTPErrorResponse::operator+;
|
|
HTTPErrorResponse& operator+(UrlNotFoundMsg /*unused*/);
|
|
};
|
|
|
|
class InvalidUrlMsg {};
|
|
|
|
extern const InvalidUrlMsg invalidUrlMsg;
|
|
|
|
struct HTTP400Response : HTTPErrorResponse
|
|
{
|
|
HTTP400Response(const InternalServer& server,
|
|
const RequestContext& request);
|
|
|
|
using HTTPErrorResponse::operator+;
|
|
HTTPErrorResponse& operator+(InvalidUrlMsg /*unused*/);
|
|
};
|
|
|
|
struct HTTP500Response : HTTPErrorResponse
|
|
{
|
|
HTTP500Response(const InternalServer& server,
|
|
const RequestContext& request);
|
|
|
|
private: // overrides
|
|
// generateResponseObject() is overriden in order to produce a minimal
|
|
// response without any need for additional resources from the server
|
|
std::unique_ptr<ContentResponse> generateResponseObject() const override;
|
|
};
|
|
|
|
class ItemResponse : public Response {
|
|
public:
|
|
ItemResponse(bool verbose, const zim::Item& item, const std::string& mimetype, const ByteRange& byterange);
|
|
static std::unique_ptr<Response> build(const InternalServer& server, const RequestContext& request, const zim::Item& item);
|
|
|
|
private:
|
|
MHD_Response* create_mhd_response(const RequestContext& request);
|
|
|
|
zim::Item m_item;
|
|
std::string m_mimeType;
|
|
};
|
|
|
|
}
|
|
|
|
#endif //KIWIXLIB_SERVER_RESPONSE_H
|