Some thoughts about poll
This commit is contained in:
parent
4b87b560ac
commit
59c1ffd027
@ -55,6 +55,8 @@ add_subdirectory(database)
|
|||||||
add_subdirectory(utils)
|
add_subdirectory(utils)
|
||||||
|
|
||||||
configure_file(config.h.in config.h @ONLY)
|
configure_file(config.h.in config.h @ONLY)
|
||||||
|
configure_file(run.sh.in run.sh @ONLY)
|
||||||
|
execute_process(COMMAND chmod +x ${CMAKE_CURRENT_BINARY_DIR}/run.sh)
|
||||||
|
|
||||||
target_link_libraries(${PROJECT_NAME} PRIVATE
|
target_link_libraries(${PROJECT_NAME} PRIVATE
|
||||||
FCGI::FCGI
|
FCGI::FCGI
|
||||||
|
@ -37,7 +37,7 @@ CREATE TABLE IF NOT EXISTS sessions (
|
|||||||
`owner` INTEGER UNSIGNED NOT NULL,
|
`owner` INTEGER UNSIGNED NOT NULL,
|
||||||
`started` TIMESTAMP DEFAULT UTC_TIMESTAMP(),
|
`started` TIMESTAMP DEFAULT UTC_TIMESTAMP(),
|
||||||
`latest` TIMESTAMP DEFAULT UTC_TIMESTAMP(),
|
`latest` TIMESTAMP DEFAULT UTC_TIMESTAMP(),
|
||||||
`access` CHAR(32),
|
`access` CHAR(32) NOT NULL UNIQUE,
|
||||||
`renew` CHAR(32),
|
`renew` CHAR(32),
|
||||||
`persist` BOOLEAN NOT NULL,
|
`persist` BOOLEAN NOT NULL,
|
||||||
`device` TEXT,
|
`device` TEXT,
|
||||||
|
@ -34,7 +34,7 @@ void Handler::Login::handle(Request& request) {
|
|||||||
success = server->validatePassword(login, password);
|
success = server->validatePassword(login, password);
|
||||||
} catch (const DBInterface::NoLogin& e) {
|
} catch (const DBInterface::NoLogin& e) {
|
||||||
std::cerr << "Exception on logging in:\n\t" << e.what() << std::endl;
|
std::cerr << "Exception on logging in:\n\t" << e.what() << std::endl;
|
||||||
return error(request, Result::noLogin, Response::Status::badRequest); //can send unauthed instead, to exclude login spoofing
|
return error(request, Result::wrongCredentials, Response::Status::badRequest);
|
||||||
} catch (const std::exception& e) {
|
} catch (const std::exception& e) {
|
||||||
std::cerr << "Exception on logging in:\n\t" << e.what() << std::endl;
|
std::cerr << "Exception on logging in:\n\t" << e.what() << std::endl;
|
||||||
return error(request, Result::unknownError, Response::Status::internalError);
|
return error(request, Result::unknownError, Response::Status::internalError);
|
||||||
@ -43,7 +43,7 @@ void Handler::Login::handle(Request& request) {
|
|||||||
return error(request, Result::unknownError, Response::Status::internalError);
|
return error(request, Result::unknownError, Response::Status::internalError);
|
||||||
}
|
}
|
||||||
if (!success)
|
if (!success)
|
||||||
return error(request, Result::noLogin, Response::Status::badRequest);
|
return error(request, Result::wrongCredentials, Response::Status::badRequest);
|
||||||
|
|
||||||
nlohmann::json body = nlohmann::json::object();
|
nlohmann::json body = nlohmann::json::object();
|
||||||
body["result"] = Result::success;
|
body["result"] = Result::success;
|
||||||
|
@ -19,6 +19,7 @@ public:
|
|||||||
emptyLogin,
|
emptyLogin,
|
||||||
noPassword,
|
noPassword,
|
||||||
emptyPassword,
|
emptyPassword,
|
||||||
|
wrongCredentials,
|
||||||
unknownError
|
unknownError
|
||||||
};
|
};
|
||||||
|
|
||||||
|
44
handler/poll.cpp
Normal file
44
handler/poll.cpp
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Yury Gubich <blue@macaw.me>
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#include "handler/poll.h"
|
||||||
|
|
||||||
|
#include "response/response.h"
|
||||||
|
#include "server/server.h"
|
||||||
|
#include "request/redirect.h"
|
||||||
|
|
||||||
|
Handler::Poll::Poll (Server* server):
|
||||||
|
Handler("login", Request::Method::get),
|
||||||
|
server(server)
|
||||||
|
{}
|
||||||
|
|
||||||
|
void Handler::Poll::handle (Request& request) {
|
||||||
|
std::string access = request.getAuthorizationToken();
|
||||||
|
if (access.empty())
|
||||||
|
return error(request, Result::tokenProblem, Response::Status::unauthorized);
|
||||||
|
|
||||||
|
if (access.size() != 32)
|
||||||
|
return error(request, Result::tokenProblem, Response::Status::badRequest);
|
||||||
|
|
||||||
|
try {
|
||||||
|
Session& session = server->getSession(access);
|
||||||
|
throw Redirect(&session);
|
||||||
|
} catch (const Redirect& r) {
|
||||||
|
throw r;
|
||||||
|
} catch (const std::exception& e) {
|
||||||
|
std::cerr << "Exception on poll:\n\t" << e.what() << std::endl;
|
||||||
|
return error(request, Result::unknownError, Response::Status::internalError);
|
||||||
|
} catch (...) {
|
||||||
|
std::cerr << "Unknown exception on poll" << std::endl;
|
||||||
|
return error(request, Result::unknownError, Response::Status::internalError);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Handler::Poll::error(Request& request, Result result, Response::Status status) const {
|
||||||
|
Response& res = request.createResponse(status);
|
||||||
|
nlohmann::json body = nlohmann::json::object();
|
||||||
|
body["result"] = result;
|
||||||
|
|
||||||
|
res.setBody(body);
|
||||||
|
res.send();
|
||||||
|
}
|
32
handler/poll.h
Normal file
32
handler/poll.h
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Yury Gubich <blue@macaw.me>
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "handler.h"
|
||||||
|
#include "request/request.h"
|
||||||
|
#include "response/response.h"
|
||||||
|
|
||||||
|
class Server;
|
||||||
|
namespace Handler {
|
||||||
|
|
||||||
|
class Poll : public Handler {
|
||||||
|
public:
|
||||||
|
Poll(Server* server);
|
||||||
|
void handle(Request& request) override;
|
||||||
|
|
||||||
|
enum class Result {
|
||||||
|
success,
|
||||||
|
tokenProblem,
|
||||||
|
replace,
|
||||||
|
timeout,
|
||||||
|
unknownError
|
||||||
|
};
|
||||||
|
|
||||||
|
void error(Request& request, Result result, Response::Status status) const;
|
||||||
|
private:
|
||||||
|
Server* server;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
@ -1,5 +1,6 @@
|
|||||||
set(HEADERS
|
set(HEADERS
|
||||||
request.h
|
request.h
|
||||||
|
redirectable.h
|
||||||
)
|
)
|
||||||
|
|
||||||
set(SOURCES
|
set(SOURCES
|
||||||
|
13
request/accepting.h
Normal file
13
request/accepting.h
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Yury Gubich <blue@macaw.me>
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include "request/request.h"
|
||||||
|
|
||||||
|
class Accepting {
|
||||||
|
public:
|
||||||
|
virtual void accept(std::unique_ptr<Request> request) = 0;
|
||||||
|
};
|
12
request/redirect.cpp
Normal file
12
request/redirect.cpp
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Yury Gubich <blue@macaw.me>
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#include "redirect.h"
|
||||||
|
|
||||||
|
Redirect::Redirect(Accepting* destination):
|
||||||
|
destination(destination)
|
||||||
|
{}
|
||||||
|
|
||||||
|
const char* Redirect::what() const noexcept {
|
||||||
|
return "This is a redirect, should have beeh handled in router, but if you see it - something went terrebly wrong";
|
||||||
|
}
|
16
request/redirect.h
Normal file
16
request/redirect.h
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
// SPDX-FileCopyrightText: 2023 Yury Gubich <blue@macaw.me>
|
||||||
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <exception>
|
||||||
|
|
||||||
|
#include "accepting.h"
|
||||||
|
|
||||||
|
class Redirect : std::exception {
|
||||||
|
public:
|
||||||
|
Redirect(Accepting* destination);
|
||||||
|
|
||||||
|
Accepting* destination;
|
||||||
|
const char* what() const noexcept override;
|
||||||
|
};
|
@ -10,6 +10,7 @@ constexpr static const char* SCRIPT_FILENAME("SCRIPT_FILENAME");
|
|||||||
constexpr static const char* SERVER_NAME("SERVER_NAME");
|
constexpr static const char* SERVER_NAME("SERVER_NAME");
|
||||||
constexpr static const char* CONTENT_TYPE("CONTENT_TYPE");
|
constexpr static const char* CONTENT_TYPE("CONTENT_TYPE");
|
||||||
constexpr static const char* CONTENT_LENGTH("CONTENT_LENGTH");
|
constexpr static const char* CONTENT_LENGTH("CONTENT_LENGTH");
|
||||||
|
constexpr static const char* AUTHORIZATION("HTTP_AUTHORIZATION");
|
||||||
|
|
||||||
constexpr static const char* urlEncoded("application/x-www-form-urlencoded");
|
constexpr static const char* urlEncoded("application/x-www-form-urlencoded");
|
||||||
|
|
||||||
@ -31,7 +32,8 @@ constexpr std::array<
|
|||||||
Request::Request ():
|
Request::Request ():
|
||||||
state(State::initial),
|
state(State::initial),
|
||||||
raw(),
|
raw(),
|
||||||
response(nullptr)
|
response(nullptr),
|
||||||
|
path()
|
||||||
{}
|
{}
|
||||||
|
|
||||||
Request::~Request() {
|
Request::~Request() {
|
||||||
@ -39,25 +41,34 @@ Request::~Request() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Request::terminate() {
|
void Request::terminate() {
|
||||||
switch (state) {
|
switch (state) {
|
||||||
|
case State::terminated:
|
||||||
|
case State::initial:
|
||||||
|
break;
|
||||||
case State::accepted:
|
case State::accepted:
|
||||||
|
std::cout << "A termination of accepted request that was not responded to, it's probably an error" << std::endl;
|
||||||
|
FCGX_Finish_r(&raw);
|
||||||
|
break;
|
||||||
|
case State::responding:
|
||||||
|
std::cout << "A termination of responding request that was not actually sent, it's probably an error" << std::endl;
|
||||||
|
FCGX_Finish_r(&raw);
|
||||||
|
break;
|
||||||
case State::responded:
|
case State::responded:
|
||||||
FCGX_Finish_r(&raw);
|
FCGX_Finish_r(&raw);
|
||||||
break;
|
break;
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
|
state = State::terminated;
|
||||||
|
}
|
||||||
std::string_view Request::methodName() const {
|
std::string_view Request::methodName() const {
|
||||||
if (state == State::initial)
|
if (state == State::initial || state == State::terminated)
|
||||||
throw std::runtime_error("An attempt to read request method on not accepted request");
|
throw std::runtime_error("An attempt to read request method on not accepted request");
|
||||||
|
|
||||||
return FCGX_GetParam(REQUEST_METHOD, raw.envp);
|
return FCGX_GetParam(REQUEST_METHOD, raw.envp);
|
||||||
}
|
}
|
||||||
|
|
||||||
Request::Method Request::method() const {
|
Request::Method Request::method() const {
|
||||||
std::string_view method = methodName();
|
std::string_view method = methodName();
|
||||||
for (const auto& pair : methods) {
|
for (const auto& pair : methods) {
|
||||||
if (pair.first == method)
|
if (pair.first == method)
|
||||||
return pair.second;
|
return pair.second;
|
||||||
@ -89,6 +100,10 @@ OStream Request::getErrorStream() {
|
|||||||
return OStream(raw.err);
|
return OStream(raw.err);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool Request::active() const {
|
||||||
|
return state != State::initial && state != State::terminated;
|
||||||
|
}
|
||||||
|
|
||||||
Response& Request::createResponse() {
|
Response& Request::createResponse() {
|
||||||
if (state != State::accepted)
|
if (state != State::accepted)
|
||||||
throw std::runtime_error("An attempt create response to the request in the wrong state");
|
throw std::runtime_error("An attempt create response to the request in the wrong state");
|
||||||
@ -125,11 +140,15 @@ void Request::responseIsComplete() {
|
|||||||
throw std::runtime_error("An attempt to mark the request as complete, but it wasn't responded");
|
throw std::runtime_error("An attempt to mark the request as complete, but it wasn't responded");
|
||||||
break;
|
break;
|
||||||
case State::responding:
|
case State::responding:
|
||||||
|
std::cout << responseCode() << '\t' << methodName() << '\t' << path << std::endl;
|
||||||
state = State::responded;
|
state = State::responded;
|
||||||
break;
|
break;
|
||||||
case State::responded:
|
case State::responded:
|
||||||
throw std::runtime_error("An attempt to mark the request as a complete for the second time");
|
throw std::runtime_error("An attempt to mark the request as a complete for the second time");
|
||||||
break;
|
break;
|
||||||
|
case State::terminated:
|
||||||
|
throw std::runtime_error("An attempt to mark the request as a complete on a terminated request");
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,25 +156,28 @@ Request::State Request::currentState() const {
|
|||||||
return state;
|
return state;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string Request::getPath(const std::string& serverName) const {
|
void Request::readPath(const std::string& serverName) {
|
||||||
if (state != State::accepted)
|
if (state != State::accepted)
|
||||||
throw std::runtime_error("An attempt to request path on a wrong request state");
|
throw std::runtime_error("An attempt to request path on a wrong request state");
|
||||||
|
|
||||||
std::string path;
|
if (!path.empty())
|
||||||
|
std::cout << "Request already has path \"" + path + "\", but it's being read again, probably an error";
|
||||||
|
|
||||||
std::string_view scriptFileName(FCGX_GetParam(SCRIPT_FILENAME, raw.envp));
|
std::string_view scriptFileName(FCGX_GetParam(SCRIPT_FILENAME, raw.envp));
|
||||||
std::string::size_type snLocation = scriptFileName.find(serverName);
|
std::string::size_type snLocation = scriptFileName.find(serverName);
|
||||||
|
|
||||||
if (snLocation != std::string::npos) {
|
if (snLocation != std::string::npos) {
|
||||||
if (snLocation + serverName.size() < scriptFileName.size())
|
if (snLocation + serverName.size() < scriptFileName.size())
|
||||||
path = scriptFileName.substr(snLocation + serverName.size() + 1);
|
path = scriptFileName.substr(snLocation + serverName.size() + 1);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!path.empty()) {
|
if (!path.empty()) {
|
||||||
while (path.back() == '/')
|
while (path.back() == '/')
|
||||||
path.erase(path.end() - 1);
|
path.erase(path.end() - 1);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string Request::getPath() const {
|
||||||
return path;
|
return path;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -167,6 +189,9 @@ std::string Request::getServerName() const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void Request::printEnvironment(std::ostream& out) {
|
void Request::printEnvironment(std::ostream& out) {
|
||||||
|
if (!active())
|
||||||
|
throw std::runtime_error("An attempt to print environment of a request in a wrong state");
|
||||||
|
|
||||||
char **envp = raw.envp;
|
char **envp = raw.envp;
|
||||||
for (int i = 0; envp[i] != nullptr; ++i) {
|
for (int i = 0; envp[i] != nullptr; ++i) {
|
||||||
out << envp[i] << "\n";
|
out << envp[i] << "\n";
|
||||||
@ -187,8 +212,8 @@ void Request::printEnvironment(nlohmann::json& out) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool Request::isFormUrlEncoded() const {
|
bool Request::isFormUrlEncoded() const {
|
||||||
if (state == State::initial)
|
if (!active())
|
||||||
throw std::runtime_error("An attempt to read request content type on not accepted request");
|
throw std::runtime_error("An attempt to read content type of a request in a wrong state");
|
||||||
|
|
||||||
std::string_view contentType(FCGX_GetParam(CONTENT_TYPE, raw.envp));
|
std::string_view contentType(FCGX_GetParam(CONTENT_TYPE, raw.envp));
|
||||||
if (!contentType.empty() && contentType.find(urlEncoded) != std::string_view::npos) {
|
if (!contentType.empty() && contentType.find(urlEncoded) != std::string_view::npos) {
|
||||||
@ -199,15 +224,15 @@ bool Request::isFormUrlEncoded() const {
|
|||||||
}
|
}
|
||||||
|
|
||||||
unsigned int Request::contentLength() const {
|
unsigned int Request::contentLength() const {
|
||||||
if (state == State::initial)
|
if (!active())
|
||||||
throw std::runtime_error("An attempt to read request content length on not accepted request");
|
throw std::runtime_error("An attempt to read content length of a request in a wrong state");
|
||||||
|
|
||||||
return atoi(FCGX_GetParam(CONTENT_LENGTH, raw.envp));
|
return atoi(FCGX_GetParam(CONTENT_LENGTH, raw.envp));
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<std::string, std::string> Request::getForm() const {
|
std::map<std::string, std::string> Request::getForm() const {
|
||||||
if (state == State::initial)
|
if (!active())
|
||||||
throw std::runtime_error("An attempt to read form on not accepted request");
|
throw std::runtime_error("An attempt to read form of a request in a wrong state");
|
||||||
|
|
||||||
std::map<std::string, std::string> result;
|
std::map<std::string, std::string> result;
|
||||||
std::string_view contentType(FCGX_GetParam(CONTENT_TYPE, raw.envp));
|
std::string_view contentType(FCGX_GetParam(CONTENT_TYPE, raw.envp));
|
||||||
@ -223,3 +248,21 @@ std::map<std::string, std::string> Request::getForm() const {
|
|||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string Request::getAuthorizationToken() const {
|
||||||
|
if (!active())
|
||||||
|
throw std::runtime_error("An attempt to read authorization token of a request in a wrong state");
|
||||||
|
|
||||||
|
const char* auth = FCGX_GetParam(AUTHORIZATION, raw.envp);
|
||||||
|
if (auth == nullptr)
|
||||||
|
return std::string();
|
||||||
|
|
||||||
|
std::string result(auth);
|
||||||
|
if (result.find("Bearer") != 0)
|
||||||
|
return std::string();
|
||||||
|
|
||||||
|
result.erase(0, 6);
|
||||||
|
trim(result);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
@ -16,6 +16,7 @@
|
|||||||
|
|
||||||
#include "stream/ostream.h"
|
#include "stream/ostream.h"
|
||||||
#include "utils/formdecode.h"
|
#include "utils/formdecode.h"
|
||||||
|
#include "utils/helpers.h"
|
||||||
#include "response/response.h"
|
#include "response/response.h"
|
||||||
|
|
||||||
class Request {
|
class Request {
|
||||||
@ -25,7 +26,8 @@ public:
|
|||||||
initial,
|
initial,
|
||||||
accepted,
|
accepted,
|
||||||
responding,
|
responding,
|
||||||
responded
|
responded,
|
||||||
|
terminated
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class Method {
|
enum class Method {
|
||||||
@ -42,7 +44,7 @@ public:
|
|||||||
Request& operator = (Request&& other) = delete;
|
Request& operator = (Request&& other) = delete;
|
||||||
|
|
||||||
bool wait(int socketDescriptor);
|
bool wait(int socketDescriptor);
|
||||||
void terminate();
|
bool active() const;
|
||||||
|
|
||||||
Response& createResponse();
|
Response& createResponse();
|
||||||
Response& createResponse(Response::Status status);
|
Response& createResponse(Response::Status status);
|
||||||
@ -55,8 +57,10 @@ public:
|
|||||||
unsigned int contentLength() const;
|
unsigned int contentLength() const;
|
||||||
std::map<std::string, std::string> getForm() const;
|
std::map<std::string, std::string> getForm() const;
|
||||||
|
|
||||||
std::string getPath(const std::string& serverName) const;
|
void readPath(const std::string& serverName);
|
||||||
|
std::string getPath() const;
|
||||||
std::string getServerName() const;
|
std::string getServerName() const;
|
||||||
|
std::string getAuthorizationToken() const;
|
||||||
void printEnvironment(std::ostream& out);
|
void printEnvironment(std::ostream& out);
|
||||||
void printEnvironment(nlohmann::json& out);
|
void printEnvironment(nlohmann::json& out);
|
||||||
|
|
||||||
@ -64,9 +68,11 @@ private:
|
|||||||
OStream getOutputStream();
|
OStream getOutputStream();
|
||||||
OStream getErrorStream();
|
OStream getErrorStream();
|
||||||
void responseIsComplete();
|
void responseIsComplete();
|
||||||
|
void terminate();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
State state;
|
State state;
|
||||||
FCGX_Request raw;
|
FCGX_Request raw;
|
||||||
std::unique_ptr<Response> response;
|
std::unique_ptr<Response> response;
|
||||||
|
std::string path;
|
||||||
};
|
};
|
||||||
|
21
run.sh.in
Normal file
21
run.sh.in
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
start_service() {
|
||||||
|
if systemctl is-active --quiet $1
|
||||||
|
then
|
||||||
|
echo "$1 is already running"
|
||||||
|
else
|
||||||
|
sudo systemctl start $1
|
||||||
|
echo "$1 started"
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
if [ ! -d "/run/pica" ]; then
|
||||||
|
sudo mkdir /run/pica
|
||||||
|
sudo chown $USER:$USER /run/pica
|
||||||
|
fi
|
||||||
|
|
||||||
|
start_service "mariadb"
|
||||||
|
start_service "httpd"
|
||||||
|
|
||||||
|
./@PROJECT_NAME@
|
@ -3,6 +3,8 @@
|
|||||||
|
|
||||||
#include "router.h"
|
#include "router.h"
|
||||||
|
|
||||||
|
#include "request/redirect.h"
|
||||||
|
|
||||||
Router::Router():
|
Router::Router():
|
||||||
get(),
|
get(),
|
||||||
post()
|
post()
|
||||||
@ -25,54 +27,54 @@ void Router::addRoute(Handler handler) {
|
|||||||
throw std::runtime_error("could'not add route " + handler->path + " to the routing table");
|
throw std::runtime_error("could'not add route " + handler->path + " to the routing table");
|
||||||
}
|
}
|
||||||
|
|
||||||
void Router::route(const std::string& path, std::unique_ptr<Request> request) {
|
void Router::route(std::unique_ptr<Request> request) {
|
||||||
std::map<std::string, Handler>::const_iterator itr, end;
|
std::map<std::string, Handler>::const_iterator itr, end;
|
||||||
switch (request->method()) {
|
switch (request->method()) {
|
||||||
case Request::Method::get:
|
case Request::Method::get:
|
||||||
itr = get.find(path);
|
itr = get.find(request->getPath());
|
||||||
end = get.end();
|
end = get.end();
|
||||||
break;
|
break;
|
||||||
case Request::Method::post:
|
case Request::Method::post:
|
||||||
itr = post.find(path);
|
itr = post.find(request->getPath());
|
||||||
end = post.end();
|
end = post.end();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
return handleMethodNotAllowed(path, std::move(request));
|
return handleMethodNotAllowed(std::move(request));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (itr == end)
|
if (itr == end)
|
||||||
return handleNotFound(path, std::move(request));
|
return handleNotFound(std::move(request));
|
||||||
|
|
||||||
try {
|
try {
|
||||||
std::cout << "Handling " << path << "..." << std::endl;
|
|
||||||
itr->second->handle(*request.get());
|
itr->second->handle(*request.get());
|
||||||
|
|
||||||
if (request->currentState() != Request::State::responded)
|
if (request->currentState() != Request::State::responded)
|
||||||
handleInternalError(path, std::runtime_error("handler failed to handle the request"), std::move(request));
|
handleInternalError(std::runtime_error("handler failed to handle the request"), std::move(request));
|
||||||
else
|
} catch (const Redirect& redirect) {
|
||||||
std::cout << request->responseCode() << '\t' << request->methodName() << '\t' << path << std::endl;
|
redirect.destination->accept(std::move(request));
|
||||||
} catch (const std::exception& e) {
|
} catch (const std::exception& e) {
|
||||||
handleInternalError(path, e, std::move(request));
|
handleInternalError(e, std::move(request));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Router::handleNotFound(const std::string& path, std::unique_ptr<Request> request) {
|
void Router::handleNotFound(std::unique_ptr<Request> request) {
|
||||||
Response& notFound = request->createResponse(Response::Status::notFound);
|
Response& notFound = request->createResponse(Response::Status::notFound);
|
||||||
|
std::string path = request->getPath();
|
||||||
notFound.setBody(std::string("Path \"") + path + "\" was not found");
|
notFound.setBody(std::string("Path \"") + path + "\" was not found");
|
||||||
notFound.send();
|
notFound.send();
|
||||||
std::cerr << notFound.statusCode() << '\t' << request->methodName() << '\t' << path << std::endl;
|
std::cerr << notFound.statusCode() << '\t' << request->methodName() << '\t' << path << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Router::handleInternalError(const std::string& path, const std::exception& exception, std::unique_ptr<Request> request) {
|
void Router::handleInternalError(const std::exception& exception, std::unique_ptr<Request> request) {
|
||||||
Response& error = request->createResponse(Response::Status::internalError);
|
Response& error = request->createResponse(Response::Status::internalError);
|
||||||
error.setBody(std::string(exception.what()));
|
error.setBody(std::string(exception.what()));
|
||||||
error.send();
|
error.send();
|
||||||
std::cerr << error.statusCode() << '\t' << request->methodName() << '\t' << path << std::endl;
|
std::cerr << error.statusCode() << '\t' << request->methodName() << '\t' << request->getPath() << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Router::handleMethodNotAllowed(const std::string& path, std::unique_ptr<Request> request) {
|
void Router::handleMethodNotAllowed(std::unique_ptr<Request> request) {
|
||||||
Response& error = request->createResponse(Response::Status::methodNotAllowed);
|
Response& error = request->createResponse(Response::Status::methodNotAllowed);
|
||||||
error.setBody(std::string("Method not allowed"));
|
error.setBody(std::string("Method not allowed"));
|
||||||
error.send();
|
error.send();
|
||||||
std::cerr << error.statusCode() << '\t' << request->methodName() << '\t' << path << std::endl;
|
std::cerr << error.statusCode() << '\t' << request->methodName() << '\t' << request->getPath() << std::endl;
|
||||||
}
|
}
|
||||||
|
@ -20,12 +20,12 @@ public:
|
|||||||
Router();
|
Router();
|
||||||
|
|
||||||
void addRoute(Handler handler);
|
void addRoute(Handler handler);
|
||||||
void route(const std::string& path, std::unique_ptr<Request> request);
|
void route(std::unique_ptr<Request> request);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void handleNotFound(const std::string& path, std::unique_ptr<Request> request);
|
void handleNotFound(std::unique_ptr<Request> request);
|
||||||
void handleInternalError(const std::string& path, const std::exception& exception, std::unique_ptr<Request> request);
|
void handleInternalError(const std::exception& exception, std::unique_ptr<Request> request);
|
||||||
void handleMethodNotAllowed(const std::string& path, std::unique_ptr<Request> request);
|
void handleMethodNotAllowed(std::unique_ptr<Request> request);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::map<std::string, Handler> get;
|
std::map<std::string, Handler> get;
|
||||||
|
@ -5,13 +5,15 @@
|
|||||||
|
|
||||||
#include <random>
|
#include <random>
|
||||||
|
|
||||||
|
#include "database/exceptions.h"
|
||||||
|
|
||||||
#include "handler/info.h"
|
#include "handler/info.h"
|
||||||
#include "handler/env.h"
|
#include "handler/env.h"
|
||||||
#include "handler/register.h"
|
#include "handler/register.h"
|
||||||
#include "handler/login.h"
|
#include "handler/login.h"
|
||||||
|
|
||||||
constexpr const char* pepper = "well, not much of a secret, huh?";
|
constexpr const char* pepper = "well, not much of a secret, huh?";
|
||||||
constexpr uint8_t currentDbVesion = 1;
|
|
||||||
constexpr const char* randomChars = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
constexpr const char* randomChars = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
||||||
constexpr uint8_t saltSize = 16;
|
constexpr uint8_t saltSize = 16;
|
||||||
constexpr uint8_t hashSize = 32;
|
constexpr uint8_t hashSize = 32;
|
||||||
@ -19,6 +21,8 @@ constexpr uint8_t hashParallel = 1;
|
|||||||
constexpr uint8_t hashIterations = 2;
|
constexpr uint8_t hashIterations = 2;
|
||||||
constexpr uint32_t hashMemoryCost = 65536;
|
constexpr uint32_t hashMemoryCost = 65536;
|
||||||
|
|
||||||
|
constexpr uint8_t currentDbVesion = 1;
|
||||||
|
|
||||||
Server::Server():
|
Server::Server():
|
||||||
terminating(false),
|
terminating(false),
|
||||||
requestCount(0),
|
requestCount(0),
|
||||||
@ -72,8 +76,8 @@ void Server::handleRequest(std::unique_ptr<Request> request) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string path = request->getPath(serverName.value());
|
request->readPath(serverName.value());
|
||||||
router.route(path.data(), std::move(request));
|
router.route(std::move(request));
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string Server::generateRandomString(std::size_t length) {
|
std::string Server::generateRandomString(std::size_t length) {
|
||||||
@ -128,9 +132,22 @@ bool Server::validatePassword(const std::string& login, const std::string& passw
|
|||||||
}
|
}
|
||||||
|
|
||||||
Session& Server::openSession(const std::string& login) {
|
Session& Server::openSession(const std::string& login) {
|
||||||
std::string accessToken = generateRandomString(32);
|
std::string accessToken, renewToken;
|
||||||
std::string renewToken = generateRandomString(32);
|
unsigned int sessionId = 0;
|
||||||
unsigned int sessionId = db->createSession(login, accessToken, renewToken);
|
int counter = 10;
|
||||||
|
do {
|
||||||
|
try {
|
||||||
|
accessToken = generateRandomString(32);
|
||||||
|
renewToken = generateRandomString(32);
|
||||||
|
sessionId = db->createSession(login, accessToken, renewToken);
|
||||||
|
break;
|
||||||
|
} catch (const DBInterface::Duplicate& e) {
|
||||||
|
std::cout << "Duplicate on creating session, trying again with different tokens";
|
||||||
|
}
|
||||||
|
} while (--counter != 0);
|
||||||
|
|
||||||
|
if (sessionId == 0)
|
||||||
|
throw std::runtime_error("Couldn't create session, ran out of attempts");
|
||||||
|
|
||||||
std::unique_ptr<Session>& session = sessions[accessToken] = std::make_unique<Session>(sessionId, accessToken, renewToken);
|
std::unique_ptr<Session>& session = sessions[accessToken] = std::make_unique<Session>(sessionId, accessToken, renewToken);
|
||||||
return *session.get();
|
return *session.get();
|
||||||
|
@ -36,6 +36,7 @@ public:
|
|||||||
unsigned int registerAccount(const std::string& login, const std::string& password);
|
unsigned int registerAccount(const std::string& login, const std::string& password);
|
||||||
bool validatePassword(const std::string& login, const std::string& password);
|
bool validatePassword(const std::string& login, const std::string& password);
|
||||||
Session& openSession(const std::string& login);
|
Session& openSession(const std::string& login);
|
||||||
|
Session& getSession(const std::string& accessToken);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void handleRequest(std::unique_ptr<Request> request);
|
void handleRequest(std::unique_ptr<Request> request);
|
||||||
|
@ -3,10 +3,13 @@
|
|||||||
|
|
||||||
#include "session.h"
|
#include "session.h"
|
||||||
|
|
||||||
|
#include "handler/poll.h"
|
||||||
|
|
||||||
Session::Session(unsigned int id, const std::string& access, const std::string& renew):
|
Session::Session(unsigned int id, const std::string& access, const std::string& renew):
|
||||||
id(id),
|
id(id),
|
||||||
access(access),
|
access(access),
|
||||||
renew(renew)
|
renew(renew),
|
||||||
|
polling(nullptr)
|
||||||
{}
|
{}
|
||||||
|
|
||||||
std::string Session::getAccessToken() const {
|
std::string Session::getAccessToken() const {
|
||||||
@ -16,3 +19,16 @@ std::string Session::getAccessToken() const {
|
|||||||
std::string Session::getRenewToken() const {
|
std::string Session::getRenewToken() const {
|
||||||
return renew;
|
return renew;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Session::accept(std::unique_ptr<Request> request) {
|
||||||
|
if (polling) {
|
||||||
|
Response& res = request->createResponse(Response::Status::ok);
|
||||||
|
nlohmann::json body = nlohmann::json::object();
|
||||||
|
body["result"] = Handler::Poll::Result::replace;
|
||||||
|
|
||||||
|
res.setBody(body);
|
||||||
|
res.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
polling = std::move(request);
|
||||||
|
}
|
@ -5,15 +5,19 @@
|
|||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
class Session {
|
#include "request/accepting.h"
|
||||||
|
|
||||||
|
class Session : public Accepting {
|
||||||
public:
|
public:
|
||||||
Session(unsigned int id, const std::string& access, const std::string& renew);
|
Session(unsigned int id, const std::string& access, const std::string& renew);
|
||||||
|
|
||||||
std::string getAccessToken() const;
|
std::string getAccessToken() const;
|
||||||
std::string getRenewToken() const;
|
std::string getRenewToken() const;
|
||||||
|
void accept(std::unique_ptr<Request> request) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
unsigned int id;
|
unsigned int id;
|
||||||
std::string access;
|
std::string access;
|
||||||
std::string renew;
|
std::string renew;
|
||||||
|
std::unique_ptr<Request> polling;
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user