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 ;
2025-01-16 04:48:31 +08:00
task - > remark = std : : string ( root . at ( " comment " ) . as_string ( ) ) ;
2024-11-26 22:58:54 +08:00
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
}