#include "Application.h" #include "BoostLog.h" #include "BulmaTheme.h" #include "Database/Session.h" #include "HomePage.h" #include "LoginPage.h" #include "NavigationBar.h" #include "Restful.h" #include "VisitorRecordsPage.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include 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("bulma", !embedded)); if (!embedded) { m_navigationBar = root()->addNew(); m_navigationBar->registerClicked.connect([this]() { if (m_loginPage) { m_loginPage->registerNewUser(); } else if (m_loginPageRef) { m_loginPageRef->registerNewUser(); } }); m_root = root()->addNew(); } else { std::unique_ptr topPtr = std::make_unique(); 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'"; } 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; } } } LOG(info) << "url: " << url(); LOG(info) << "relative resources url: " << relativeResourcesUrl(); LOG(info) << "resources url: " << resourcesUrl(); LOG(info) << "internal path: " << internalPath(); auto app = Amass::Singleton::instance(); m_loginPage = std::make_unique(app->authService(), m_session->users(), m_session->login()); m_loginPage->processEnvironment(); handlePathChange(m_externalPath.empty() ? internalPath() : m_externalPath); internalPathChanged().connect(this, &Application::handlePathChange); } 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."; if (m_loginPage) { if (m_navigationBar != nullptr) { m_loginPageRef = m_navigationBar->addLoginItem(std::move(m_loginPage)); } } else if (m_loginPageRef != nullptr && m_loginPageRef->parent() == m_root) { m_loginPage = m_loginPageRef->parent()->removeWidget(m_loginPageRef); if (m_navigationBar != nullptr) { m_loginPageRef = m_navigationBar->addLoginItem(std::move(m_loginPage)); } } setInternalPath("/", true); auto app = Amass::Singleton::instance(); auto &service = app->authService(); auto &env = environment(); auto token = env.getCookie(service.authTokenCookieName()); if (token == nullptr) { Wt::Http::Cookie cookie(service.authTokenCookieName(), service.createAuthToken(u)); setCookie(cookie); } } else { m_loginPage = m_navigationBar->removeLoginItem(); LOG(info) << "User logged out."; } } void Application::handlePathChange(const std::string &path) { LOG(info) << "handlePathChange: " << path; if (path.starts_with("/wt/login")) { if (m_session->login().loggedIn()) { std::exit(0); } else { if (m_loginPage) { m_root->clear(); m_loginPageRef = m_root->addWidget(std::move(m_loginPage)); } } } else { 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(*m_session); } else { m_root->clear(); m_root->addNew(); } } } Server::Server(uint16_t port, const std::string &applicationRoot, const std::string &documentRoot) { try { std::vector args; args.push_back(std::format("--docroot={};/resources", documentRoot)); args.push_back(std::format("--approot={}/resources", applicationRoot)); args.push_back(std::format("--http-listen=127.0.0.1:{}", port)); initializeAuthenticationService(); m_server = std::make_unique(std::format("{}/resources", applicationRoot), args); 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"); m_server->addResource(std::make_shared(), "/auth/verify"); m_server->addResource(std::make_shared(), "/plaintext"); m_server->start(); } catch (const std::exception &e) { LOG(error) << e.what(); } } std::unique_ptr Server::createApplication(const Wt::WEnvironment &env, bool embedded) { return std::make_unique(env, embedded); } Server::~Server() { } void Server::initializeAuthenticationService() { m_authService = std::make_unique(); m_authService->setEmailVerificationEnabled(true); m_authService->setEmailVerificationRequired(true); m_authService->setAuthTokensEnabled(true, "logincookie"); m_passwordService = std::make_unique(*m_authService); auto verifier = std::make_unique(); verifier->addHashFunction(std::make_unique(7)); m_passwordService->setVerifier(std::move(verifier)); m_passwordService->setPasswordThrottle(std::make_unique()); m_passwordService->setStrengthValidator(std::make_unique()); } Wt::Auth::AuthService &Server::authService() { return *m_authService; } const Wt::Auth::PasswordService &Server::passwordService() { return *m_passwordService; } } // namespace WebToolkit