Feature: Implement trigger logic for host online event
This commit is contained in:
497
server/2015Remote/TriggerSettingsDlg.cpp
Normal file
497
server/2015Remote/TriggerSettingsDlg.cpp
Normal file
@@ -0,0 +1,497 @@
|
||||
#include "stdafx.h"
|
||||
#include "TriggerSettingsDlg.h"
|
||||
#include "2015RemoteDlg.h"
|
||||
#include "jsoncpp/json.h"
|
||||
#include <fstream>
|
||||
#include <sstream>
|
||||
|
||||
#ifndef _WIN64
|
||||
#ifdef _DEBUG
|
||||
#pragma comment(lib, "jsoncpp/jsoncppd.lib")
|
||||
#else
|
||||
#pragma comment(lib, "jsoncpp/jsoncpp.lib")
|
||||
#endif
|
||||
#else
|
||||
#ifdef _DEBUG
|
||||
#pragma comment(lib, "jsoncpp/jsoncpp_x64d.lib")
|
||||
#else
|
||||
#pragma comment(lib, "jsoncpp/jsoncpp_x64.lib")
|
||||
#endif
|
||||
#endif
|
||||
|
||||
// GBK (CP_ACP) -> UTF-8 编码转换
|
||||
static std::string GbkToUtf8(const std::string& gbkStr)
|
||||
{
|
||||
if (gbkStr.empty()) return "";
|
||||
|
||||
// GBK -> WideChar
|
||||
int wideLen = MultiByteToWideChar(CP_ACP, 0, gbkStr.c_str(), -1, NULL, 0);
|
||||
if (wideLen <= 0) return gbkStr;
|
||||
|
||||
std::wstring wideStr(wideLen, 0);
|
||||
MultiByteToWideChar(CP_ACP, 0, gbkStr.c_str(), -1, &wideStr[0], wideLen);
|
||||
|
||||
// WideChar -> UTF-8
|
||||
int utf8Len = WideCharToMultiByte(CP_UTF8, 0, wideStr.c_str(), -1, NULL, 0, NULL, NULL);
|
||||
if (utf8Len <= 0) return gbkStr;
|
||||
|
||||
std::string utf8Str(utf8Len, 0);
|
||||
WideCharToMultiByte(CP_UTF8, 0, wideStr.c_str(), -1, &utf8Str[0], utf8Len, NULL, NULL);
|
||||
|
||||
// 移除末尾的 null 字符
|
||||
if (!utf8Str.empty() && utf8Str.back() == '\0') {
|
||||
utf8Str.pop_back();
|
||||
}
|
||||
return utf8Str;
|
||||
}
|
||||
|
||||
// UTF-8 -> GBK (CP_ACP) 编码转换
|
||||
static std::string Utf8ToGbk(const std::string& utf8Str)
|
||||
{
|
||||
if (utf8Str.empty()) return "";
|
||||
|
||||
// UTF-8 -> WideChar
|
||||
int wideLen = MultiByteToWideChar(CP_UTF8, 0, utf8Str.c_str(), -1, NULL, 0);
|
||||
if (wideLen <= 0) return utf8Str;
|
||||
|
||||
std::wstring wideStr(wideLen, 0);
|
||||
MultiByteToWideChar(CP_UTF8, 0, utf8Str.c_str(), -1, &wideStr[0], wideLen);
|
||||
|
||||
// WideChar -> GBK
|
||||
int gbkLen = WideCharToMultiByte(CP_ACP, 0, wideStr.c_str(), -1, NULL, 0, NULL, NULL);
|
||||
if (gbkLen <= 0) return utf8Str;
|
||||
|
||||
std::string gbkStr(gbkLen, 0);
|
||||
WideCharToMultiByte(CP_ACP, 0, wideStr.c_str(), -1, &gbkStr[0], gbkLen, NULL, NULL);
|
||||
|
||||
// 移除末尾的 null 字符
|
||||
if (!gbkStr.empty() && gbkStr.back() == '\0') {
|
||||
gbkStr.pop_back();
|
||||
}
|
||||
return gbkStr;
|
||||
}
|
||||
|
||||
BEGIN_MESSAGE_MAP(CTriggerSettingsDlg, CDialogLangEx)
|
||||
ON_BN_CLICKED(IDC_BTN_SAVE, &CTriggerSettingsDlg::OnBnClickedBtnSave)
|
||||
ON_BN_CLICKED(IDC_BTN_TRIGGER_ADD, &CTriggerSettingsDlg::OnBnClickedBtnTriggerAdd)
|
||||
ON_BN_CLICKED(IDC_BTN_TRIGGER_REMOVE, &CTriggerSettingsDlg::OnBnClickedBtnTriggerRemove)
|
||||
ON_NOTIFY(LVN_ITEMCHANGED, IDC_LIST_TRIGGERS, &CTriggerSettingsDlg::OnLvnItemchangedListTriggers)
|
||||
END_MESSAGE_MAP()
|
||||
|
||||
CTriggerSettingsDlg::CTriggerSettingsDlg(std::vector<DllInfo*>& dllList, CWnd* pParent)
|
||||
: CDialogLangEx(IDD_DIALOG_TRIGGER_SETTINGS, pParent)
|
||||
, m_DllList(dllList)
|
||||
{
|
||||
}
|
||||
|
||||
CTriggerSettingsDlg::~CTriggerSettingsDlg()
|
||||
{
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::DoDataExchange(CDataExchange* pDX)
|
||||
{
|
||||
CDialogLangEx::DoDataExchange(pDX);
|
||||
DDX_Control(pDX, IDC_COMBO_TRIGGER_TYPE, m_comboTriggerType);
|
||||
DDX_Control(pDX, IDC_LIST_TRIGGER_PLUGINS, m_listPlugins);
|
||||
DDX_Control(pDX, IDC_LIST_TRIGGERS, m_listTriggers);
|
||||
}
|
||||
|
||||
BOOL CTriggerSettingsDlg::OnInitDialog()
|
||||
{
|
||||
CDialogLangEx::OnInitDialog();
|
||||
|
||||
InitControls();
|
||||
|
||||
// 加载配置
|
||||
m_Configs = LoadTriggerConfigs();
|
||||
|
||||
// 加载插件列表
|
||||
LoadPluginsToList();
|
||||
|
||||
// 加载已配置的触发器
|
||||
LoadTriggersToList();
|
||||
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::InitControls()
|
||||
{
|
||||
// 初始化触发类型下拉框
|
||||
m_comboTriggerType.InsertString(TRIGGER_HOST_ONLINE, _TR("主机上线"));
|
||||
m_comboTriggerType.SetCurSel(TRIGGER_HOST_ONLINE);
|
||||
|
||||
// 初始化插件列表
|
||||
m_listPlugins.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_GRIDLINES | LVS_EX_CHECKBOXES);
|
||||
m_listPlugins.InsertColumn(0, _TR("插件名称"), LVCFMT_LEFT, 160);
|
||||
|
||||
// 初始化已配置触发器列表
|
||||
m_listTriggers.SetExtendedStyle(LVS_EX_FULLROWSELECT | LVS_EX_GRIDLINES);
|
||||
m_listTriggers.InsertColumn(0, _TR("触发器"), LVCFMT_LEFT, 160);
|
||||
|
||||
// 设置静态文本
|
||||
SetWindowText(_TR("触发器设置"));
|
||||
GetDlgItem(IDC_STATIC_TRIGGER_TYPE)->SetWindowText(_TR("触发类型:"));
|
||||
GetDlgItem(IDC_STATIC_TRIGGER_ACTION)->SetWindowText(_TR("执行动作:"));
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::LoadPluginsToList()
|
||||
{
|
||||
m_listPlugins.DeleteAllItems();
|
||||
|
||||
int index = 0;
|
||||
for (const auto& dll : m_DllList) {
|
||||
if (!dll || !dll->Data) continue;
|
||||
|
||||
m_listPlugins.InsertItem(index, CString(dll->Name.c_str()));
|
||||
m_listPlugins.SetItemData(index, (DWORD_PTR)dll);
|
||||
index++;
|
||||
}
|
||||
|
||||
// 如果有已配置的主机上线触发器,勾选对应的插件
|
||||
TriggerConfig* onlineTrigger = GetOnlineTrigger(m_Configs);
|
||||
if (onlineTrigger) {
|
||||
for (int i = 0; i < m_listPlugins.GetItemCount(); i++) {
|
||||
DllInfo* dll = reinterpret_cast<DllInfo*>(m_listPlugins.GetItemData(i));
|
||||
if (!dll) continue;
|
||||
for (const auto& pluginName : onlineTrigger->PluginNames) {
|
||||
if (pluginName == dll->Name) {
|
||||
m_listPlugins.SetCheck(i, TRUE);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::LoadTriggersToList()
|
||||
{
|
||||
m_listTriggers.DeleteAllItems();
|
||||
|
||||
const char* typeNames[] = { "主机上线" };
|
||||
|
||||
int index = 0;
|
||||
for (const auto& cfg : m_Configs) {
|
||||
if (!cfg.PluginNames.empty()) {
|
||||
// 显示触发器类型和插件数量
|
||||
CString text;
|
||||
text.Format(_T("%s (%d)"), _TR(typeNames[cfg.Type]), (int)cfg.PluginNames.size());
|
||||
m_listTriggers.InsertItem(index, text);
|
||||
m_listTriggers.SetItemData(index, cfg.Type);
|
||||
index++;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::OnBnClickedBtnTriggerAdd()
|
||||
{
|
||||
// 获取选中的触发类型
|
||||
int triggerType = m_comboTriggerType.GetCurSel();
|
||||
if (triggerType < 0) return;
|
||||
|
||||
// 获取勾选的插件(直接从 DllInfo 获取名称,避免编码转换问题)
|
||||
std::vector<std::string> selectedPlugins;
|
||||
for (int i = 0; i < m_listPlugins.GetItemCount(); i++) {
|
||||
if (m_listPlugins.GetCheck(i)) {
|
||||
DllInfo* dll = reinterpret_cast<DllInfo*>(m_listPlugins.GetItemData(i));
|
||||
if (dll) {
|
||||
selectedPlugins.push_back(dll->Name);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (selectedPlugins.empty()) {
|
||||
MessageBoxL(_TR("请先选择至少一个插件"), _TR("提示"), MB_ICONINFORMATION);
|
||||
return;
|
||||
}
|
||||
|
||||
// 查找或创建该类型的触发器
|
||||
TriggerConfig* cfg = nullptr;
|
||||
for (auto& c : m_Configs) {
|
||||
if (c.Type == (TriggerType)triggerType) {
|
||||
cfg = &c;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!cfg) {
|
||||
TriggerConfig newCfg;
|
||||
newCfg.Type = (TriggerType)triggerType;
|
||||
m_Configs.push_back(newCfg);
|
||||
cfg = &m_Configs.back();
|
||||
}
|
||||
|
||||
cfg->PluginNames = selectedPlugins;
|
||||
|
||||
// 刷新显示
|
||||
LoadTriggersToList();
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::OnBnClickedBtnTriggerRemove()
|
||||
{
|
||||
POSITION pos = m_listTriggers.GetFirstSelectedItemPosition();
|
||||
if (!pos) return;
|
||||
|
||||
int nItem = m_listTriggers.GetNextSelectedItem(pos);
|
||||
TriggerType type = (TriggerType)m_listTriggers.GetItemData(nItem);
|
||||
|
||||
// 从配置中移除
|
||||
for (auto it = m_Configs.begin(); it != m_Configs.end(); ++it) {
|
||||
if (it->Type == type) {
|
||||
m_Configs.erase(it);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// 取消插件列表的勾选
|
||||
for (int i = 0; i < m_listPlugins.GetItemCount(); i++) {
|
||||
m_listPlugins.SetCheck(i, FALSE);
|
||||
}
|
||||
|
||||
// 刷新显示
|
||||
LoadTriggersToList();
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::OnLvnItemchangedListTriggers(NMHDR* pNMHDR, LRESULT* pResult)
|
||||
{
|
||||
LPNMLISTVIEW pNMLV = reinterpret_cast<LPNMLISTVIEW>(pNMHDR);
|
||||
*pResult = 0;
|
||||
|
||||
// 只处理选中状态变化
|
||||
if (!(pNMLV->uNewState & LVIS_SELECTED)) return;
|
||||
|
||||
int nItem = pNMLV->iItem;
|
||||
if (nItem < 0) return;
|
||||
|
||||
TriggerType type = (TriggerType)m_listTriggers.GetItemData(nItem);
|
||||
|
||||
// 查找对应的触发器配置
|
||||
TriggerConfig* cfg = nullptr;
|
||||
for (auto& c : m_Configs) {
|
||||
if (c.Type == type) {
|
||||
cfg = &c;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// 先取消所有勾选
|
||||
for (int i = 0; i < m_listPlugins.GetItemCount(); i++) {
|
||||
m_listPlugins.SetCheck(i, FALSE);
|
||||
}
|
||||
|
||||
// 勾选该触发器配置的插件
|
||||
if (cfg) {
|
||||
for (int i = 0; i < m_listPlugins.GetItemCount(); i++) {
|
||||
DllInfo* dll = reinterpret_cast<DllInfo*>(m_listPlugins.GetItemData(i));
|
||||
if (!dll) continue;
|
||||
for (const auto& pluginName : cfg->PluginNames) {
|
||||
if (pluginName == dll->Name) {
|
||||
m_listPlugins.SetCheck(i, TRUE);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 同步下拉框选择
|
||||
m_comboTriggerType.SetCurSel(type);
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::OnBnClickedBtnSave()
|
||||
{
|
||||
// 先从界面收集当前选择
|
||||
int triggerType = m_comboTriggerType.GetCurSel();
|
||||
if (triggerType >= 0) {
|
||||
std::vector<std::string> selectedPlugins;
|
||||
for (int i = 0; i < m_listPlugins.GetItemCount(); i++) {
|
||||
if (m_listPlugins.GetCheck(i)) {
|
||||
DllInfo* dll = reinterpret_cast<DllInfo*>(m_listPlugins.GetItemData(i));
|
||||
if (dll) {
|
||||
selectedPlugins.push_back(dll->Name);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 更新或创建触发器配置
|
||||
TriggerConfig* cfg = nullptr;
|
||||
for (auto& c : m_Configs) {
|
||||
if (c.Type == (TriggerType)triggerType) {
|
||||
cfg = &c;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (!selectedPlugins.empty()) {
|
||||
if (!cfg) {
|
||||
TriggerConfig newCfg;
|
||||
newCfg.Type = (TriggerType)triggerType;
|
||||
m_Configs.push_back(newCfg);
|
||||
cfg = &m_Configs.back();
|
||||
}
|
||||
cfg->PluginNames = selectedPlugins;
|
||||
} else if (cfg) {
|
||||
// 如果没有选中插件,删除该触发器
|
||||
for (auto it = m_Configs.begin(); it != m_Configs.end(); ++it) {
|
||||
if (it->Type == (TriggerType)triggerType) {
|
||||
m_Configs.erase(it);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
SaveTriggerConfigs(m_Configs);
|
||||
LoadTriggersToList();
|
||||
|
||||
MessageBoxL(_TR("配置已保存"), _TR("提示"), MB_ICONINFORMATION);
|
||||
}
|
||||
|
||||
std::string CTriggerSettingsDlg::GetTriggerConfigPath()
|
||||
{
|
||||
std::string dbPath = GetDbPath();
|
||||
size_t pos = dbPath.find_last_of("\\/");
|
||||
std::string dir = (pos != std::string::npos) ? dbPath.substr(0, pos + 1) : "";
|
||||
return dir + "triggers.json";
|
||||
}
|
||||
|
||||
std::vector<TriggerConfig> CTriggerSettingsDlg::LoadTriggerConfigs()
|
||||
{
|
||||
std::vector<TriggerConfig> configs;
|
||||
std::string path = GetTriggerConfigPath();
|
||||
|
||||
std::ifstream file(path);
|
||||
if (!file.is_open()) {
|
||||
return configs;
|
||||
}
|
||||
|
||||
Json::Value root;
|
||||
Json::CharReaderBuilder builder;
|
||||
std::string errors;
|
||||
|
||||
if (!Json::parseFromStream(builder, file, &root, &errors)) {
|
||||
return configs;
|
||||
}
|
||||
|
||||
if (!root.isArray()) {
|
||||
return configs;
|
||||
}
|
||||
|
||||
for (const auto& item : root) {
|
||||
TriggerConfig cfg;
|
||||
cfg.Type = (TriggerType)item.get("type", TRIGGER_HOST_ONLINE).asInt();
|
||||
|
||||
const Json::Value& plugins = item["plugins"];
|
||||
if (plugins.isArray()) {
|
||||
for (const auto& p : plugins) {
|
||||
cfg.PluginNames.push_back(Utf8ToGbk(p.asString())); // UTF-8 -> GBK
|
||||
}
|
||||
}
|
||||
|
||||
if (!cfg.PluginNames.empty()) {
|
||||
configs.push_back(cfg);
|
||||
}
|
||||
}
|
||||
|
||||
return configs;
|
||||
}
|
||||
|
||||
void CTriggerSettingsDlg::SaveTriggerConfigs(const std::vector<TriggerConfig>& configs)
|
||||
{
|
||||
std::string path = GetTriggerConfigPath();
|
||||
|
||||
Json::Value root(Json::arrayValue);
|
||||
|
||||
for (const auto& cfg : configs) {
|
||||
Json::Value item;
|
||||
item["type"] = cfg.Type;
|
||||
|
||||
Json::Value plugins(Json::arrayValue);
|
||||
for (const auto& name : cfg.PluginNames) {
|
||||
plugins.append(GbkToUtf8(name)); // GBK -> UTF-8
|
||||
}
|
||||
item["plugins"] = plugins;
|
||||
|
||||
root.append(item);
|
||||
}
|
||||
|
||||
std::ofstream file(path);
|
||||
if (file.is_open()) {
|
||||
Json::StreamWriterBuilder builder;
|
||||
builder["indentation"] = " ";
|
||||
builder["emitUTF8"] = true; // 输出可读的 UTF-8 字符,而非 \uXXXX
|
||||
std::unique_ptr<Json::StreamWriter> writer(builder.newStreamWriter());
|
||||
writer->write(root, &file);
|
||||
file.close(); // 确保文件完全写入并关闭
|
||||
}
|
||||
|
||||
// 通知 TriggerManager 重新加载缓存
|
||||
TriggerManager::Instance().Reload();
|
||||
}
|
||||
|
||||
TriggerConfig* CTriggerSettingsDlg::GetOnlineTrigger(std::vector<TriggerConfig>& configs)
|
||||
{
|
||||
for (auto& cfg : configs) {
|
||||
if (cfg.Type == TRIGGER_HOST_ONLINE && !cfg.PluginNames.empty()) {
|
||||
return &cfg;
|
||||
}
|
||||
}
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
// ============================================
|
||||
// TriggerManager 实现
|
||||
// ============================================
|
||||
|
||||
TriggerManager::TriggerManager() : m_bLoaded(false)
|
||||
{
|
||||
InitializeCriticalSection(&m_cs);
|
||||
}
|
||||
|
||||
TriggerManager::~TriggerManager()
|
||||
{
|
||||
DeleteCriticalSection(&m_cs);
|
||||
}
|
||||
|
||||
void TriggerManager::LoadFromDisk()
|
||||
{
|
||||
// 不加锁,由调用者保证线程安全
|
||||
m_OnlinePlugins.clear();
|
||||
|
||||
auto configs = CTriggerSettingsDlg::LoadTriggerConfigs();
|
||||
for (const auto& cfg : configs) {
|
||||
if (cfg.Type == TRIGGER_HOST_ONLINE) {
|
||||
for (const auto& name : cfg.PluginNames) {
|
||||
m_OnlinePlugins.insert(name);
|
||||
}
|
||||
}
|
||||
}
|
||||
m_bLoaded = true;
|
||||
}
|
||||
|
||||
void TriggerManager::Reload()
|
||||
{
|
||||
EnterCriticalSection(&m_cs);
|
||||
LoadFromDisk();
|
||||
LeaveCriticalSection(&m_cs);
|
||||
}
|
||||
|
||||
bool TriggerManager::HasOnlineTrigger()
|
||||
{
|
||||
EnterCriticalSection(&m_cs);
|
||||
if (!m_bLoaded) {
|
||||
LoadFromDisk();
|
||||
}
|
||||
bool has = !m_OnlinePlugins.empty();
|
||||
LeaveCriticalSection(&m_cs);
|
||||
return has;
|
||||
}
|
||||
|
||||
std::set<std::string> TriggerManager::GetOnlinePlugins()
|
||||
{
|
||||
EnterCriticalSection(&m_cs);
|
||||
if (!m_bLoaded) {
|
||||
LoadFromDisk();
|
||||
}
|
||||
std::set<std::string> result = m_OnlinePlugins; // 复制一份返回
|
||||
LeaveCriticalSection(&m_cs);
|
||||
return result;
|
||||
}
|
||||
Reference in New Issue
Block a user