Older/WebApplication/Application.cpp

207 lines
8.1 KiB
C++
Raw Normal View History

2024-12-01 15:10:25 +08:00
#include "Application.h"
#include "BoostLog.h"
#include "BulmaTheme.h"
#include "Database/Session.h"
2024-12-02 01:18:57 +08:00
#include "HomePage.h"
2024-12-01 15:10:25 +08:00
#include "LoginPage.h"
2024-12-21 13:35:12 +08:00
#include "NavigationBar.h"
2024-12-01 15:10:25 +08:00
#include "Restful.h"
#include "VisitorRecordsPage.h"
#include <Wt/Auth/AuthService.h>
#include <Wt/Auth/HashFunction.h>
#include <Wt/Auth/Identity.h>
#include <Wt/Auth/PasswordService.h>
#include <Wt/Auth/PasswordStrengthValidator.h>
#include <Wt/Auth/PasswordVerifier.h>
#include <Wt/Dbo/FixedSqlConnectionPool.h>
#include <Wt/Dbo/SqlConnectionPool.h>
#include <Wt/Dbo/backend/Sqlite3.h>
2024-12-26 23:10:41 +08:00
#include <Wt/Http/Cookie.h>
2024-12-01 15:10:25 +08:00
#include <Wt/WContainerWidget.h>
#include <Wt/WEnvironment.h>
#include <Wt/WServer.h>
#include <format>
namespace WebToolkit {
Application::Application(const Wt::WEnvironment &env, bool embedded) : Wt::WApplication(env) {
messageResourceBundle().use(appRoot() + "wt");
messageResourceBundle().use(appRoot() + "auth_strings");
messageResourceBundle().use(appRoot() + "auth_css_theme");
useStyleSheet("/resources/app.css");
LOG(info) << "app root: " << appRoot();
m_session = Database::session();
m_session->login().changed().connect(this, &Application::authEvent);
setTheme(std::make_shared<BulmaTheme>("bulma", !embedded));
if (!embedded) {
2024-12-21 13:35:12 +08:00
m_navigationBar = root()->addNew<NavigationBar>();
2024-12-24 23:46:20 +08:00
m_navigationBar->registerClicked.connect([this]() {
if (m_loginPage) {
2024-12-22 13:56:15 +08:00
m_loginPage->registerNewUser();
2024-12-24 23:46:20 +08:00
} else if (m_loginPageRef) {
2024-12-22 13:56:15 +08:00
m_loginPageRef->registerNewUser();
}
});
2024-12-21 13:35:12 +08:00
m_root = root()->addNew<Wt::WContainerWidget>();
2024-12-01 15:10:25 +08:00
} else {
std::unique_ptr<Wt::WContainerWidget> topPtr = std::make_unique<Wt::WContainerWidget>();
m_root = topPtr.get();
const std::string *div = env.getParameter("div");
if (div) {
setJavaScriptClass(*div);
bindWidget(std::move(topPtr), *div);
} else {
LOG(error) << "Missing: parameter: 'div'";
2025-01-03 04:51:08 +08:00
m_root = nullptr;
2024-12-01 15:10:25 +08:00
}
auto externalPath = env.getParameter("path");
if (externalPath != nullptr) {
m_externalPath = *externalPath;
LOG(info) << "external path: " << m_externalPath;
} else {
auto parameters = env.getParameterMap();
for (auto &p : parameters) {
LOG(info) << p.first;
}
}
}
2024-12-02 01:18:57 +08:00
LOG(info) << "url: " << url();
LOG(info) << "relative resources url: " << relativeResourcesUrl();
LOG(info) << "resources url: " << resourcesUrl();
2024-12-01 15:10:25 +08:00
LOG(info) << "internal path: " << internalPath();
2025-01-03 04:51:08 +08:00
LOG(info) << "bookmark url: " << bookmarkUrl("/");
LOG(info) << "relative url: " << resolveRelativeUrl("/");
2024-12-21 13:35:12 +08:00
auto app = Amass::Singleton<WebToolkit::Server>::instance();
m_loginPage = std::make_unique<LoginPage>(app->authService(), m_session->users(), m_session->login());
2025-01-03 04:51:08 +08:00
if (m_externalPath.empty()) {
m_loginPage->processEnvironment();
} else {
m_loginPage->processExternalEnvironment(m_externalPath, app->authService());
}
2024-12-21 23:08:38 +08:00
handlePathChange(m_externalPath.empty() ? internalPath() : m_externalPath);
internalPathChanged().connect(this, &Application::handlePathChange);
2024-12-01 15:10:25 +08:00
}
Application::~Application() {
}
void Application::authEvent() {
if (m_session->login().loggedIn()) {
const Wt::Auth::User &u = m_session->login().user();
LOG(info) << "User " << u.id() << " (" << u.identity(Wt::Auth::Identity::LoginName) << ")"
<< " logged in.";
2024-12-21 23:08:38 +08:00
if (m_loginPage) {
2025-01-02 17:16:52 +08:00
if (m_navigationBar != nullptr) {
m_loginPageRef = m_navigationBar->addLoginItem(std::move(m_loginPage));
2025-01-03 00:02:00 +08:00
m_loginPageRef->removeStyleClass("bulma-m-auto");
m_loginPageRef->removeStyleClass("bulma-container");
2025-01-02 17:16:52 +08:00
}
2024-12-21 23:08:38 +08:00
} else if (m_loginPageRef != nullptr && m_loginPageRef->parent() == m_root) {
m_loginPage = m_loginPageRef->parent()->removeWidget(m_loginPageRef);
2025-01-02 17:16:52 +08:00
if (m_navigationBar != nullptr) {
m_loginPageRef = m_navigationBar->addLoginItem(std::move(m_loginPage));
2025-01-03 00:02:00 +08:00
m_loginPageRef->removeStyleClass("bulma-m-auto");
m_loginPageRef->removeStyleClass("bulma-container");
2025-01-02 17:16:52 +08:00
}
2024-12-21 23:08:38 +08:00
}
setInternalPath("/", true);
2025-01-02 23:02:37 +08:00
2024-12-26 23:10:41 +08:00
auto app = Amass::Singleton<WebToolkit::Server>::instance();
auto &service = app->authService();
2025-01-02 23:02:37 +08:00
auto &env = environment();
auto token = env.getCookie(service.authTokenCookieName());
if (token == nullptr) {
Wt::Http::Cookie cookie(service.authTokenCookieName(), service.createAuthToken(u));
setCookie(cookie);
}
2024-12-01 15:10:25 +08:00
} else {
2024-12-21 23:08:38 +08:00
m_loginPage = m_navigationBar->removeLoginItem();
2024-12-01 15:10:25 +08:00
LOG(info) << "User logged out.";
}
}
void Application::handlePathChange(const std::string &path) {
LOG(info) << "handlePathChange: " << path;
2025-01-03 04:51:08 +08:00
if (m_root == nullptr) {
LOG(error) << "root container is null.";
return;
}
2024-12-21 23:08:38 +08:00
if (path.starts_with("/wt/login")) {
if (m_session->login().loggedIn()) {
2025-01-03 04:51:08 +08:00
LOG(info) << "already logged in.";
2024-12-21 23:08:38 +08:00
} else {
if (m_loginPage) {
m_root->clear();
m_loginPageRef = m_root->addWidget(std::move(m_loginPage));
2025-01-03 00:02:00 +08:00
m_loginPageRef->addStyleClass("bulma-m-auto bulma-container");
2024-12-21 23:08:38 +08:00
}
}
2024-12-02 01:18:57 +08:00
} else {
2024-12-21 23:08:38 +08:00
if (m_loginPageRef != nullptr && m_loginPageRef->parent() == m_root) {
m_loginPage = m_root->removeWidget(m_loginPageRef);
}
if (path.starts_with("/wt/register")) {
} else if (path.starts_with("/wt/visitor/analysis")) {
m_root->clear();
m_root->addNew<VisitorRecordsPage>(*m_session);
} else {
m_root->clear();
m_root->addNew<HomePage>();
}
2024-12-01 15:10:25 +08:00
}
}
2024-12-01 20:01:13 +08:00
Server::Server(uint16_t port, const std::string &applicationRoot, const std::string &documentRoot) {
2024-12-01 15:10:25 +08:00
try {
std::vector<std::string> args;
args.push_back(std::format("--docroot={};/resources", documentRoot));
2024-12-01 20:01:13 +08:00
args.push_back(std::format("--approot={}/resources", applicationRoot));
2024-12-01 15:10:25 +08:00
args.push_back(std::format("--http-listen=127.0.0.1:{}", port));
initializeAuthenticationService();
2024-12-01 20:01:13 +08:00
m_server = std::make_unique<Wt::WServer>(std::format("{}/resources", applicationRoot), args);
2024-12-01 15:10:25 +08:00
m_server->addEntryPoint(Wt::EntryPointType::Application,
std::bind(&Server::createApplication, this, std::placeholders::_1, false));
m_server->addEntryPoint(Wt::EntryPointType::WidgetSet,
std::bind(&Server::createApplication, this, std::placeholders::_1, true), "/wt/app.js");
2024-12-25 22:07:00 +08:00
m_server->addResource(std::make_shared<AuthenticationResource>(), "/auth/verify");
2024-12-01 15:10:25 +08:00
m_server->addResource(std::make_shared<PlaintextResource>(), "/plaintext");
m_server->start();
} catch (const std::exception &e) {
LOG(error) << e.what();
}
}
std::unique_ptr<Wt::WApplication> Server::createApplication(const Wt::WEnvironment &env, bool embedded) {
return std::make_unique<Application>(env, embedded);
}
Server::~Server() {
}
void Server::initializeAuthenticationService() {
m_authService = std::make_unique<Wt::Auth::AuthService>();
2024-12-24 23:46:20 +08:00
m_authService->setEmailVerificationEnabled(true);
m_authService->setEmailVerificationRequired(true);
2024-12-01 15:10:25 +08:00
m_authService->setAuthTokensEnabled(true, "logincookie");
m_passwordService = std::make_unique<Wt::Auth::PasswordService>(*m_authService);
auto verifier = std::make_unique<Wt::Auth::PasswordVerifier>();
verifier->addHashFunction(std::make_unique<Wt::Auth::BCryptHashFunction>(7));
m_passwordService->setVerifier(std::move(verifier));
m_passwordService->setPasswordThrottle(std::make_unique<Wt::Auth::AuthThrottle>());
m_passwordService->setStrengthValidator(std::make_unique<Wt::Auth::PasswordStrengthValidator>());
}
2024-12-26 23:10:41 +08:00
Wt::Auth::AuthService &Server::authService() {
2024-12-01 15:10:25 +08:00
return *m_authService;
}
const Wt::Auth::PasswordService &Server::passwordService() {
return *m_passwordService;
}
} // namespace WebToolkit