Older/Server/Application.cpp

485 lines
21 KiB
C++
Raw Normal View History

2024-01-24 23:19:53 +08:00
#include "Application.h"
2024-11-26 22:58:54 +08:00
#include "Database/Session.h"
2024-01-24 23:19:53 +08:00
#include "DateTime.h"
#include "HttpSession.h"
#include "IoContext.h"
2024-11-10 18:33:39 +08:00
#include "Nng/SocketAisoWrapper.h"
2024-01-24 23:19:53 +08:00
#include "ServiceLogic.h"
#include "ServiceManager.h"
2024-05-03 22:30:46 +08:00
#include "SystemUsage.h"
2024-01-24 23:19:53 +08:00
#include "WeChatContext/CorporationContext.h"
2025-01-12 00:46:14 +08:00
#include "WebRTC/SignalServer.h"
2024-11-26 22:58:54 +08:00
#include <Wt/Dbo/Json.h>
2025-01-10 21:49:22 +08:00
#include <boost/asio/strand.hpp>
2024-11-10 18:33:39 +08:00
#include <boost/json/object.hpp>
#include <boost/json/serialize.hpp>
2025-01-07 16:54:07 +08:00
#include <boost/process/v2/process.hpp>
2024-05-05 22:00:15 +08:00
#include <boost/stacktrace.hpp>
2024-01-24 23:19:53 +08:00
2024-11-10 18:33:39 +08:00
constexpr auto IpcUrl = "ipc:///tmp/nng_ipc_server";
2024-11-27 20:23:02 +08:00
static std::vector<std::string> urlFilter = {
2025-01-10 21:49:22 +08:00
"/", "/search", "/LoginPage", "/MessageBoard", "/我的笔记", "/我的笔记/", "/我的博客",
2024-11-27 20:23:02 +08:00
};
2025-01-10 21:49:22 +08:00
class ApplicationPrivate {
public:
std::shared_ptr<boost::urls::router<Application::RequestHandler>> router;
std::shared_ptr<boost::asio::ip::tcp::acceptor> acceptor;
};
Application::Application(const std::string &path) : ApplicationSettings(path), m_d{new ApplicationPrivate()} {
using namespace boost::urls;
m_d->router = std::make_shared<router<RequestHandler>>();
2024-01-24 23:19:53 +08:00
// clang-format off
2025-01-10 21:49:22 +08:00
m_d->router->insert("/{path*}",[this](HttpSession &session, const Request &request, const matches &matches) {
2024-01-24 23:19:53 +08:00
using namespace boost::beast;
2025-01-10 21:49:22 +08:00
url_view view(request.target());
2024-01-24 23:19:53 +08:00
auto target = view.path();
LOG(info) << target;
if (target.find("..") != boost::beast::string_view::npos) {
session.reply(ServiceLogic::badRequest(request, "Illegal request-target"));
return;
}
std::string path = ResponseUtility::pathCat(getDocumentRoot(), target);
if (target.back() == '/') path.append("index.html");
if (std::filesystem::is_directory(path)) path.append("/index.html");
boost::beast::error_code ec;
http::file_body::value_type body;
body.open(path.c_str(), boost::beast::file_mode::scan, ec);
if (ec == boost::beast::errc::no_such_file_or_directory) {
std::ostringstream oss;
oss << "The resource '" << target << "' was not found.";
LOG(error) << oss.str();
session.errorReply(request, http::status::not_found, oss.str());
return;
} else if (ec) {
session.reply(ServiceLogic::serverError(request, ec.message()));
return;
}
auto const size = body.size();
http::response<http::file_body> res{std::piecewise_construct, std::make_tuple(std::move(body)),
std::make_tuple(http::status::ok, request.version())};
res.set(http::field::server, BOOST_BEAST_VERSION_STRING);
res.set(http::field::content_type, ResponseUtility::mimeType(path));
res.content_length(size);
res.keep_alive(request.keep_alive());
session.reply(std::move(res));
});
2025-01-10 21:49:22 +08:00
m_d->router->insert("/wechat/{session*}",[this](HttpSession &session, const Request &request, const matches &matches) {
2024-01-24 23:19:53 +08:00
ServiceLogic::onWechat(shared_from_this(), request,
[&session](auto &&response) { session.reply(std::move(response)); });
});
2025-01-10 21:49:22 +08:00
m_d->router->insert("/api/v1/tasklist", [this](HttpSession &session, const Request &request, const matches &matches) {
2024-01-24 23:19:53 +08:00
using namespace boost::beast;
2024-11-27 19:23:06 +08:00
auto database = Database::session();
2024-11-26 22:58:54 +08:00
Tasks tasks = database->find<Task>();
std::ostringstream oss;
Wt::Dbo::jsonSerialize(tasks, oss);
2024-01-24 23:19:53 +08:00
http::response<boost::beast::http::string_body> s{boost::beast::http::status::ok, request.version()};
s.set(http::field::server, BOOST_BEAST_VERSION_STRING);
s.set(http::field::content_type, "application/json;charset=UTF-8");
s.keep_alive(request.keep_alive());
2024-11-26 22:58:54 +08:00
s.body() = oss.str();
2024-01-24 23:19:53 +08:00
s.prepare_payload();
session.reply(std::move(s));
});
2025-01-10 21:49:22 +08:00
m_d->router->insert("/api/v1/task/add", [this](HttpSession &session, const Request &request, const matches &matches) mutable {
2024-01-24 23:19:53 +08:00
using namespace boost::beast;
2024-11-27 00:08:24 +08:00
using namespace std::chrono;
2024-01-24 23:19:53 +08:00
LOG(info) << "add task: " << request.body();
auto rootJson = boost::json::parse(request.body());
auto &root = rootJson.as_object();
std::string content;
if (root.contains("content")) {
content = root.at("content").as_string();
}
2024-11-27 19:23:06 +08:00
auto database = Database::session();;
2024-11-26 22:58:54 +08:00
auto task = std::make_unique<Task>();
2024-11-27 00:08:24 +08:00
task->createTime = system_clock::time_point(seconds(root.at("createTime").as_int64()));
2024-11-26 22:58:54 +08:00
task->content = content;
task->comment = std::string(root.at("comment").as_string());
auto t = database->add(std::move(task));
Wt::Dbo::ptr<Task> parent = database->find<Task>("where id=?").bind(root.at("parentId").as_int64());
if (parent) {
parent.modify()->children.insert(t);
}
2024-01-24 23:19:53 +08:00
boost::json::object reply;
2024-11-26 22:58:54 +08:00
reply["status"] = 0;
2024-01-24 23:19:53 +08:00
http::response<boost::beast::http::string_body> s{boost::beast::http::status::ok, request.version()};
s.set(http::field::server, BOOST_BEAST_VERSION_STRING);
s.set(http::field::content_type, "application/json;charset=UTF-8");
s.keep_alive(request.keep_alive());
s.body() = boost::json::serialize(reply);
s.prepare_payload();
session.reply(std::move(s));
});
2025-01-10 21:49:22 +08:00
m_d->router->insert("/api/v1/task/delete/{id}", [this](HttpSession &session, const Request &request,const matches &matches) {
2024-01-24 23:19:53 +08:00
using namespace boost::beast;
LOG(info) << "delete task: " << matches.at("id");
2024-11-27 19:23:06 +08:00
auto database = Database::session();;
2024-11-26 22:58:54 +08:00
Wt::Dbo::ptr<Task> joe = database->find<Task>().where("id = ?").bind(std::stoi(matches.at("id")));
int status = -1;
if (joe) {
joe.remove();
status = 0;
}
2024-01-24 23:19:53 +08:00
boost::json::object reply;
2024-11-26 22:58:54 +08:00
reply["status"] = status;
2024-01-24 23:19:53 +08:00
http::response<boost::beast::http::string_body> s{boost::beast::http::status::ok, request.version()};
s.set(http::field::server, BOOST_BEAST_VERSION_STRING);
s.set(http::field::content_type, "application/json;charset=UTF-8");
s.keep_alive(request.keep_alive());
s.body() = boost::json::serialize(reply);
s.prepare_payload();
session.reply(std::move(s));
});
2025-01-10 21:49:22 +08:00
m_d->router->insert("/notify", [this](HttpSession &session, const Request &request, const matches &matches) {
2024-01-24 23:19:53 +08:00
auto corp = Amass::Singleton<CorporationContext>::instance();
corp->notify(request);
session.reply(
ServiceLogic::make_200<boost::beast::http::string_body>(request, "notify successed.\n", "text/html"));
});
2024-11-27 19:23:06 +08:00
2025-01-10 21:49:22 +08:00
m_d->router->insert("/api/v1/visit_analysis", [this](HttpSession &session, const Request &request, const matches &matches) {
2024-07-30 22:17:55 +08:00
using namespace boost::beast;
auto rootJson = boost::json::parse(request.body());
auto &root = rootJson.as_object();
std::string url;
if (root.contains("url")) {
url = root["url"].as_string();
}
2024-11-27 19:23:06 +08:00
auto database = Database::session();
2024-12-18 23:35:26 +08:00
if (std::filesystem::exists("amass_blog" + url) && (url.find("/我的博客/page") != 0) && (url.find("/wt") != 0)) {
2024-11-27 23:50:04 +08:00
if (url.size() > 1 && url.back() == '/') {
url.pop_back();
}
2024-11-27 19:23:06 +08:00
Wt::Dbo::Transaction transaction(*database);
auto record = std::make_unique<VisitorRecord>();
record->time = std::chrono::system_clock::now();
record->url = url;
2024-08-06 22:24:26 +08:00
if (root.contains("visitor_uuid")) {
2024-11-27 19:23:06 +08:00
record->visitorUuid = root["visitor_uuid"].as_string();
2024-08-06 22:24:26 +08:00
}
std::string userAgent;
if (root.contains("user_agent")) {
2024-11-27 19:23:06 +08:00
record->userAgent = root["user_agent"].as_string();
2024-08-06 22:24:26 +08:00
}
2024-11-27 19:23:06 +08:00
database->add(std::move(record));
2024-08-06 22:24:26 +08:00
}
2024-11-27 20:00:57 +08:00
Wt::Dbo::Transaction transaction(*database);
2024-07-30 22:17:55 +08:00
boost::json::object reply;
2024-11-27 19:23:06 +08:00
reply["page_view_count"] = database->query<int>("SELECT COUNT(*) FROM visitor_record WHERE url = ?").bind(std::string(url));
reply["unique_visitor_count"] = database->query<int>("SELECT COUNT(DISTINCT visitor_uuid) FROM visitor_record WHERE url = ?").bind(std::string(url));
reply["site_page_view_count"] = database->query<int>("SELECT COUNT(*) FROM visitor_record");
reply["site_unique_visitor_count"] = database->query<int>("SELECT COUNT(DISTINCT visitor_uuid) FROM visitor_record");
2024-07-30 23:17:42 +08:00
2024-07-30 22:17:55 +08:00
http::response<boost::beast::http::string_body> s{boost::beast::http::status::ok, request.version()};
s.set(http::field::server, BOOST_BEAST_VERSION_STRING);
s.set(http::field::content_type, "application/json;charset=UTF-8");
s.keep_alive(request.keep_alive());
s.body() = boost::json::serialize(reply);
s.prepare_payload();
session.reply(std::move(s));
});
2024-07-31 22:28:05 +08:00
2025-01-10 21:49:22 +08:00
m_d->router->insert("/api/v1/most_viewed_urls", [this](HttpSession &session, const Request &request, const matches &matches) {
2024-08-05 23:01:19 +08:00
using namespace boost::beast;
int size = 5;
std::error_code error;
if (!request.body().empty()) {
auto rootJson = boost::json::parse(request.body(), error);
if (error) {
LOG(info) << "<" << request.body() << "> parse json error: " << error.message();
} else {
auto &root = rootJson.as_object();
if (root.contains("size")) {
size = root.at("size").as_int64();
}
}
}
2024-11-27 19:23:06 +08:00
auto database = Database::session();
2024-11-27 20:00:57 +08:00
Wt::Dbo::Transaction transaction(*database);
2024-11-27 20:23:02 +08:00
Wt::Dbo::collection<std::tuple<std::string, int>> query = database->query<std::tuple<std::string, int>>("SELECT url, COUNT(*) as count FROM visitor_record GROUP BY url ORDER BY count DESC LIMIT ?").bind(static_cast<int> (size+urlFilter.size()));
2024-08-05 23:01:19 +08:00
boost::json::array reply;
2024-11-27 20:23:02 +08:00
int index = 0;
2024-11-27 19:23:06 +08:00
for (auto &[url, count] : query) {
2024-11-27 20:23:02 +08:00
if (std::find(urlFilter.cbegin(), urlFilter.cend(),url) != urlFilter.cend()) continue;
2024-08-05 23:01:19 +08:00
boost::json::object object;
2024-11-27 19:23:06 +08:00
object["url"] = url;
object["count"] = count;
2024-08-05 23:01:19 +08:00
reply.push_back(std::move(object));
2024-11-27 20:23:02 +08:00
index++;
if(index>=size)break;
2024-08-05 23:01:19 +08:00
}
http::response<boost::beast::http::string_body> s{boost::beast::http::status::ok, request.version()};
s.set(http::field::server, BOOST_BEAST_VERSION_STRING);
2024-08-06 09:48:25 +08:00
s.set(http::field::content_type, "application/json;charset=UTF-8");
s.keep_alive(request.keep_alive());
s.body() = boost::json::serialize(reply);
s.prepare_payload();
session.reply(std::move(s));
});
2025-01-10 21:49:22 +08:00
m_d->router->insert("/api/v1/latest_viewed_urls", [this](HttpSession &session, const Request &request, const matches &matches) {
2024-08-06 09:48:25 +08:00
using namespace boost::beast;
2024-11-27 19:23:06 +08:00
using namespace std::chrono;
2024-08-06 09:48:25 +08:00
int size = 5;
std::error_code error;
if (!request.body().empty()) {
auto rootJson = boost::json::parse(request.body(), error);
if (error) {
LOG(info) << "<" << request.body() << "> parse json error: " << error.message();
} else {
auto &root = rootJson.as_object();
if (root.contains("size")) {
size = root.at("size").as_int64();
}
}
}
2024-11-27 19:23:06 +08:00
auto database = Database::session();
2024-11-27 20:00:57 +08:00
Wt::Dbo::Transaction transaction(*database);
2024-11-27 19:23:06 +08:00
using Reslut = std::tuple<std::string, system_clock::time_point>;
2024-11-27 20:23:02 +08:00
Wt::Dbo::collection<Reslut> query = database->query<Reslut>("SELECT url, MAX(time) FROM visitor_record GROUP BY url ORDER BY MAX(time) DESC LIMIT ?").bind(static_cast<int> (size+urlFilter.size()));
2024-08-06 09:48:25 +08:00
boost::json::array reply;
2024-11-27 20:23:02 +08:00
int index=0;
2024-11-27 19:23:06 +08:00
for (auto &[url, time] : query) {
2024-11-27 20:23:02 +08:00
if (std::find(urlFilter.cbegin(), urlFilter.cend(),url) != urlFilter.cend()) continue;
2024-08-06 09:48:25 +08:00
boost::json::object object;
2024-11-27 19:23:06 +08:00
object["url"] = url;
object["time"] = duration_cast<seconds>(time.time_since_epoch()).count();
2024-08-06 09:48:25 +08:00
reply.push_back(std::move(object));
2024-11-27 20:23:02 +08:00
index++;
if(index>=size)break;
2024-08-06 09:48:25 +08:00
}
http::response<boost::beast::http::string_body> s{boost::beast::http::status::ok, request.version()};
s.set(http::field::server, BOOST_BEAST_VERSION_STRING);
2024-08-05 23:01:19 +08:00
s.set(http::field::content_type, "application/json;charset=UTF-8");
s.keep_alive(request.keep_alive());
s.body() = boost::json::serialize(reply);
s.prepare_payload();
session.reply(std::move(s));
2025-01-07 22:05:44 +08:00
});
2025-01-10 21:49:22 +08:00
m_d->router->insert("/api/v1/search/reindex", [this](HttpSession &session, const Request &request, const matches &matches) {
2025-01-07 16:54:07 +08:00
using namespace boost::beast;
2025-01-07 22:05:44 +08:00
std::string authorizationHeader;
if (request.count(http::field::authorization)) {
authorizationHeader = request[http::field::authorization];
}
http::response<boost::beast::http::string_body> s{http::status::ok, request.version()};
if (!authorizationHeader.empty() && authorizationHeader.substr(0, 7) == "Bearer ") {
std::string bearerToken = authorizationHeader.substr(7);
auto key = getMeiliSearchApiKey();
auto config = getMeiliSearchConfig();
boost::json::object reply;
if (!key.empty() && !config.empty()) {
if (key == bearerToken) {
config = std::filesystem::absolute(config);
LOG(info) << "config path: " << config;
boost::process::process process(session.executor(), "/usr/bin/docker", {"run", "-t", "--rm", "--network=host",
"--env=MEILISEARCH_HOST_URL=http://localhost:7700",
std::format("--env=MEILISEARCH_API_KEY={}", key),
std::format("--volume={}:/docs-scraper/config.json", config),
"getmeili/docs-scraper:latest",
"pipenv", "run", "./docs_scraper", "config.json"
});
boost::process::error_code error;
int code = process.wait(error);
reply["status"] = code;
reply["message"] = error ? error.message() : "succeed.";
} else {
s.result(http::status::unauthorized);
reply["status"] = static_cast<int>(http::status::unauthorized);
reply["message"] = "Unauthorized";
}
} else {
reply["status"] = 404;
reply["message"] = "please fill MeiliSearchApiKey and MeiliSearchConfig.";
}
s.set(http::field::content_type, "application/json;charset=UTF-8");
s.body() = boost::json::serialize(reply);
2025-01-07 17:18:20 +08:00
} else {
2025-01-07 22:05:44 +08:00
s.result(http::status::unauthorized);
s.set(http::field::content_type, "text/plain");
s.body() = "Unauthorized";
2025-01-07 16:54:07 +08:00
}
s.set(http::field::server, BOOST_BEAST_VERSION_STRING);
s.keep_alive(request.keep_alive());
s.prepare_payload();
session.reply(std::move(s));
2024-08-05 23:01:19 +08:00
});
2024-11-27 19:23:06 +08:00
// clang-format on
2024-01-24 23:19:53 +08:00
m_ioContext = Amass::Singleton<IoContext>::instance<Amass::Construct>(getThreads());
m_timer = std::make_shared<boost::asio::system_timer>(*m_ioContext->ioContext());
2024-11-10 18:33:39 +08:00
m_replyer = std::make_shared<Nng::Asio::Socket>(*m_ioContext->ioContext(), Nng::Reply);
m_replyer->listen(IpcUrl);
2024-05-03 22:30:46 +08:00
m_systemUsage = std::make_shared<SystemUsage>(*m_ioContext->ioContext(), getHomeAssistantAccessToken());
m_systemUsage->start();
2025-01-12 00:46:14 +08:00
m_signalServer = std::make_shared<SignalServer>(*this);
2024-01-24 23:19:53 +08:00
alarmTask();
}
2025-01-10 21:49:22 +08:00
Application::~Application() {
if (m_d != nullptr) {
delete m_d;
}
}
2024-01-24 23:19:53 +08:00
boost::asio::io_context &Application::ioContext() {
return *m_ioContext->ioContext();
}
2025-01-10 21:49:22 +08:00
void Application::insertUrl(std::string_view url, RequestHandler &&handler) {
m_d->router->insert(url, std::move(handler));
}
void Application::startAcceptHttpConnections(const std::string &address, uint16_t port) {
m_d->acceptor = std::make_shared<boost::asio::ip::tcp::acceptor>(*m_ioContext->ioContext());
boost::beast::error_code error;
boost::asio::ip::tcp::endpoint endpoint(boost::asio::ip::make_address(address), port);
m_d->acceptor->open(endpoint.protocol(), error);
if (error) {
LOG(error) << error.message();
return;
2024-05-05 22:00:15 +08:00
}
2025-01-10 21:49:22 +08:00
m_d->acceptor->set_option(boost::asio::socket_base::reuse_address(true), error);
if (error) {
LOG(error) << error.message();
return;
}
m_d->acceptor->bind(endpoint, error);
if (error) {
LOG(error) << error.message();
return;
}
m_d->acceptor->listen(boost::asio::socket_base::max_listen_connections, error);
if (error) {
LOG(error) << error.message();
return;
}
asyncAcceptHttpConnections();
2024-01-24 23:19:53 +08:00
}
2025-01-10 21:49:22 +08:00
void Application::asyncAcceptHttpConnections() {
auto socket = std::make_shared<boost::asio::ip::tcp::socket>(boost::asio::make_strand(*m_ioContext->ioContext()));
m_d->acceptor->async_accept(*socket, [self{shared_from_this()}, socket](const boost::system::error_code &error) {
if (error) {
if (error == boost::asio::error::operation_aborted) return;
LOG(error) << error.message();
} else {
auto session = std::make_shared<HttpSession>(std::move(*socket), self->m_d->router);
session->run();
}
self->asyncAcceptHttpConnections();
});
2024-10-23 19:53:51 +08:00
}
2024-01-24 23:19:53 +08:00
int Application::exec() {
2024-11-10 18:33:39 +08:00
startAcceptRequest();
2024-01-24 23:19:53 +08:00
LOG(info) << "application start successful ...";
startCheckInterval(*m_ioContext->ioContext(), 2);
2024-11-10 18:33:39 +08:00
m_ioContext->run();
2024-01-24 23:19:53 +08:00
LOG(info) << "application exit successful ...";
return m_status;
}
void Application::alarmTask() {
int hour = 10;
int minute = 30;
auto alarmTime = DateTime::currentDateTime();
alarmTime.setHour(hour);
alarmTime.setMinute(minute);
if (std::chrono::system_clock::now() > alarmTime()) {
alarmTime = alarmTime.tomorrow();
}
m_timer->expires_at(alarmTime());
m_timer->async_wait([this](const boost::system::error_code &error) mutable {
if (error) {
LOG(error) << error.message();
return;
}
2024-11-27 19:23:06 +08:00
auto session = Database::session();
2024-11-26 22:58:54 +08:00
Tasks tasks = session->find<Task>();
2024-01-24 23:19:53 +08:00
bool founded = false;
std::string content;
for (auto &task : tasks) {
if (founded) break;
2024-11-26 22:58:54 +08:00
for (auto child : task->children) {
if (!child->finished) {
content = child->content;
2024-01-24 23:19:53 +08:00
founded = true;
break;
}
}
2024-11-26 22:58:54 +08:00
if (!founded && !task->finished) {
content = task->content;
2024-01-24 23:19:53 +08:00
founded = true;
}
}
if (founded) {
std::ostringstream oss;
oss << "待完成事项:" << std::endl;
oss << "==========" << std::endl;
oss << content << std::endl;
oss << "==========" << std::endl;
oss << "每天都要过得充实开心哦~";
auto manager = Amass::Singleton<ServiceManager>::instance();
if (manager) manager->sendMessage(NotifyServerChan, oss.str());
}
alarmTask();
});
2024-11-10 18:33:39 +08:00
}
void Application::startAcceptRequest() {
m_replyer->asyncReceive([ptr{weak_from_this()}](const boost::system::error_code &error, const Nng::Buffer &buffer) {
if (error) {
LOG(error) << error.message();
}
LOG(info) << buffer.data();
if (ptr.expired()) return;
auto self = ptr.lock();
auto value = boost::json::parse(buffer.data());
auto &request = value.as_object();
if (request.at("command").as_string() == "exit") {
boost::json::object reply;
reply["status"] = 0;
reply["message"] = "will exit.";
auto txt = boost::json::serialize(reply);
self->m_replyer->send(txt.data(), txt.size());
std::raise(SIGUSR1); // 发送自定义信号
}
self->startAcceptRequest();
});
}
void Application::requetExit() {
LOG(info) << "send exit request to program.";
Nng::Socket request(Nng::Request);
2024-11-10 20:23:00 +08:00
request.setOption(Nng::RecvTimeout, std::chrono::milliseconds(2000));
std::error_code error;
request.dial(IpcUrl, error);
if (error) {
LOG(error) << error.message();
return;
}
2024-11-10 18:33:39 +08:00
boost::json::object object;
object["command"] = "exit";
auto text = boost::json::serialize(object);
request.send(text.data(), text.size());
auto buffer = request.recv();
LOG(info) << buffer.data<char>();
2024-01-24 23:19:53 +08:00
}