mirror of https://github.com/kiwix/libkiwix.git
168 lines
4.7 KiB
C++
168 lines
4.7 KiB
C++
|
|
#include "../include/manager.h"
|
|
#include "../include/server.h"
|
|
#include "../include/name_mapper.h"
|
|
#include "../include/tools.h"
|
|
|
|
// Output generated via mustache templates sometimes contains end-of-line
|
|
// whitespace. This complicates representing the expected output of a unit-test
|
|
// as C++ raw strings in editors that are configured to delete EOL whitespace.
|
|
// A workaround is to put special markers (//EOLWHITESPACEMARKER) at the end
|
|
// of such lines in the expected output string and remove them at runtime.
|
|
// This is exactly what this function is for.
|
|
std::string removeEOLWhitespaceMarkers(const std::string& s)
|
|
{
|
|
const std::regex pattern("//EOLWHITESPACEMARKER");
|
|
return std::regex_replace(s, pattern, "");
|
|
}
|
|
|
|
std::string replace(std::string s, std::string pattern, std::string replacement)
|
|
{
|
|
return std::regex_replace(s, std::regex(pattern), replacement);
|
|
}
|
|
|
|
using TestContextImpl = std::vector<std::pair<std::string, std::string> >;
|
|
struct TestContext : TestContextImpl {
|
|
TestContext(const std::initializer_list<value_type>& il)
|
|
: TestContextImpl(il)
|
|
{}
|
|
};
|
|
|
|
std::ostream& operator<<(std::ostream& out, const TestContext& ctx)
|
|
{
|
|
out << "Test context:\n";
|
|
for ( const auto& kv : ctx )
|
|
out << "\t" << kv.first << ": " << kv.second << "\n";
|
|
out << std::endl;
|
|
return out;
|
|
}
|
|
|
|
|
|
typedef httplib::Headers Headers;
|
|
|
|
Headers invariantHeaders(Headers headers)
|
|
{
|
|
headers.erase("Date");
|
|
return headers;
|
|
}
|
|
|
|
class ZimFileServer
|
|
{
|
|
public: // types
|
|
typedef std::shared_ptr<httplib::Response> Response;
|
|
typedef std::vector<std::string> FilePathCollection;
|
|
|
|
public: // functions
|
|
ZimFileServer(int serverPort, std::string libraryFilePath);
|
|
ZimFileServer(int serverPort,
|
|
bool withTaskbar,
|
|
const FilePathCollection& zimpaths,
|
|
std::string indexTemplateString = "");
|
|
~ZimFileServer();
|
|
|
|
Response GET(const char* path, const Headers& headers = Headers())
|
|
{
|
|
return client->Get(path, headers);
|
|
}
|
|
|
|
Response HEAD(const char* path, const Headers& headers = Headers())
|
|
{
|
|
return client->Head(path, headers);
|
|
}
|
|
|
|
private:
|
|
void run(int serverPort, std::string indexTemplateString = "");
|
|
|
|
private: // data
|
|
kiwix::Library library;
|
|
kiwix::Manager manager;
|
|
std::unique_ptr<kiwix::HumanReadableNameMapper> nameMapper;
|
|
std::unique_ptr<kiwix::Server> server;
|
|
std::unique_ptr<httplib::Client> client;
|
|
const bool withTaskbar = true;
|
|
};
|
|
|
|
ZimFileServer::ZimFileServer(int serverPort, std::string libraryFilePath)
|
|
: manager(&this->library)
|
|
{
|
|
if ( kiwix::isRelativePath(libraryFilePath) )
|
|
libraryFilePath = kiwix::computeAbsolutePath(kiwix::getCurrentDirectory(), libraryFilePath);
|
|
manager.readFile(libraryFilePath, true, true);
|
|
run(serverPort);
|
|
}
|
|
|
|
ZimFileServer::ZimFileServer(int serverPort,
|
|
bool _withTaskbar,
|
|
const FilePathCollection& zimpaths,
|
|
std::string indexTemplateString)
|
|
: manager(&this->library)
|
|
, withTaskbar(_withTaskbar)
|
|
{
|
|
for ( const auto& zimpath : zimpaths ) {
|
|
if (!manager.addBookFromPath(zimpath, zimpath, "", false))
|
|
throw std::runtime_error("Unable to add the ZIM file '" + zimpath + "'");
|
|
}
|
|
run(serverPort, indexTemplateString);
|
|
}
|
|
|
|
void ZimFileServer::run(int serverPort, std::string indexTemplateString)
|
|
{
|
|
const std::string address = "127.0.0.1";
|
|
nameMapper.reset(new kiwix::HumanReadableNameMapper(library, false));
|
|
server.reset(new kiwix::Server(&library, nameMapper.get()));
|
|
server->setRoot("ROOT");
|
|
server->setAddress(address);
|
|
server->setPort(serverPort);
|
|
server->setNbThreads(2);
|
|
server->setVerbose(false);
|
|
server->setTaskbar(withTaskbar, withTaskbar);
|
|
server->setMultiZimSearchLimit(3);
|
|
if (!indexTemplateString.empty()) {
|
|
server->setIndexTemplateString(indexTemplateString);
|
|
}
|
|
|
|
if ( !server->start() )
|
|
throw std::runtime_error("ZimFileServer failed to start");
|
|
|
|
client.reset(new httplib::Client(address, serverPort));
|
|
}
|
|
|
|
ZimFileServer::~ZimFileServer()
|
|
{
|
|
server->stop();
|
|
}
|
|
|
|
class ServerTest : public ::testing::Test
|
|
{
|
|
private:
|
|
std::unique_ptr<ZimFileServer> taskbarlessZfs_;
|
|
|
|
protected:
|
|
std::unique_ptr<ZimFileServer> zfs1_;
|
|
|
|
const ZimFileServer::FilePathCollection ZIMFILES {
|
|
"./test/zimfile.zim",
|
|
"./test/example.zim",
|
|
"./test/poor.zim",
|
|
"./test/corner_cases.zim"
|
|
};
|
|
|
|
protected:
|
|
void SetUp() override {
|
|
zfs1_.reset(new ZimFileServer(SERVER_PORT, /*withTaskbar=*/true, ZIMFILES));
|
|
}
|
|
|
|
ZimFileServer& taskbarlessZimFileServer()
|
|
{
|
|
if ( ! taskbarlessZfs_ ) {
|
|
taskbarlessZfs_.reset(new ZimFileServer(SERVER_PORT+1, /*withTaskbar=*/false, ZIMFILES));
|
|
}
|
|
return *taskbarlessZfs_;
|
|
}
|
|
|
|
void TearDown() override {
|
|
zfs1_.reset();
|
|
taskbarlessZfs_.reset();
|
|
}
|
|
};
|