446 lines
17 KiB
C++
446 lines
17 KiB
C++
#include "pch.h"
|
|
|
|
#include "tunnel.h"
|
|
#include "protocol.h"
|
|
|
|
#include "globalcfg.h"
|
|
#include "httplib.h"
|
|
#include "misc.h"
|
|
#include "usrerr.h"
|
|
|
|
#include <strsafe.h>
|
|
#include <spdlog/fmt/bin_to_hex.h>
|
|
|
|
#define HTTP_JSON_CONTENT TEXT("application/json")
|
|
|
|
static httplib::Client *g_httpCtx = nullptr;
|
|
static httplib::Client *g_tunnelHttpCtx = nullptr;
|
|
|
|
int InitControlServer(const TCHAR *pUserSvrUrl) {
|
|
|
|
if (g_tunnelHttpCtx) {
|
|
delete g_tunnelHttpCtx;
|
|
g_tunnelHttpCtx = nullptr;
|
|
}
|
|
|
|
if (UsedSCGProxy()) {
|
|
TCHAR scgProxyUrl[MAX_PATH];
|
|
StringCbPrintf(scgProxyUrl, MAX_PATH, TEXT("http://127.0.0.1:%d"), GetGlobalCfgInfo()->scgProxy.scgGwPort);
|
|
|
|
SPDLOG_DEBUG(TEXT("Control Server Used Proxy: {0} --> {1}"), pUserSvrUrl, scgProxyUrl);
|
|
g_tunnelHttpCtx = new httplib::Client(scgProxyUrl);
|
|
} else {
|
|
g_tunnelHttpCtx = new httplib::Client(pUserSvrUrl);
|
|
SPDLOG_DEBUG(TEXT("Control Server Unused Proxy: {0}"), pUserSvrUrl);
|
|
}
|
|
|
|
if (g_tunnelHttpCtx) {
|
|
g_tunnelHttpCtx->set_connection_timeout(0, 1000000); // 1 second
|
|
g_tunnelHttpCtx->set_read_timeout(5, 0); // 5 seconds
|
|
g_tunnelHttpCtx->set_write_timeout(5, 0); // 5 seconds
|
|
g_tunnelHttpCtx->set_keep_alive(true);
|
|
g_tunnelHttpCtx->set_post_connect_cb([](socket_t sock) {
|
|
if (UsedSCGProxy()) {
|
|
int ret;
|
|
unsigned char vmid[4];
|
|
unsigned char *p;
|
|
const unsigned int id = htonl(GetGlobalCfgInfo()->curConnVmId);
|
|
const auto svrId = static_cast<UINT8>(GetGlobalCfgInfo()->userCfg.cliConfig.scgCtrlAppId);
|
|
unsigned char scgProxy[] = {0x01, // VERSION
|
|
0x09, // Length
|
|
0xF0, // ++++++ INFO[0] TYPE
|
|
0x04, // INFO[0] LENGTH
|
|
0, // INFO[0] VMID[0]
|
|
0, // INFO[0] VMID[1]
|
|
0, // INFO[0] VMID[2]
|
|
0, // INFO[0] VMID[3]
|
|
0xF1, // INFO[1] TYPE
|
|
0x01, // INFO[1] LENGTH
|
|
svrId}; // ------ INFO[1] SCG Service ID
|
|
|
|
p = scgProxy;
|
|
memcpy(vmid, &id, 4);
|
|
scgProxy[4] = vmid[0];
|
|
scgProxy[5] = vmid[1];
|
|
scgProxy[6] = vmid[2];
|
|
scgProxy[7] = vmid[3];
|
|
|
|
if (GetGlobalCfgInfo()->logLevel == spdlog::level::trace) {
|
|
std::array<unsigned char, sizeof(scgProxy)> arr;
|
|
std::copy(std::begin(scgProxy), std::end(scgProxy), std::begin(arr));
|
|
SPDLOG_DEBUG(TEXT("TCP Proxy SCG Payload: {0:Xa}"), spdlog::to_hex(arr));
|
|
}
|
|
|
|
ret = send(sock, reinterpret_cast<const char *>(p), sizeof(scgProxy), 0);
|
|
|
|
while (ret < static_cast<int>(sizeof(scgProxy))) {
|
|
p += ret;
|
|
ret += send(sock, reinterpret_cast<const char *>(p), sizeof(scgProxy), 0);
|
|
}
|
|
|
|
SPDLOG_DEBUG(TEXT("Service Connected To SCG Server({1}/{2}): {0}"),
|
|
sock,
|
|
GetGlobalCfgInfo()->curConnVmId,
|
|
svrId);
|
|
}
|
|
});
|
|
}
|
|
|
|
SPDLOG_DEBUG(TEXT("Connect to Tunnel Control Service: {0}"), pUserSvrUrl);
|
|
|
|
return ERR_SUCCESS;
|
|
}
|
|
|
|
template<class T> int CreateProtocolRequest(T *pReqParams, TCHAR **pOutJson) {
|
|
std::string json;
|
|
|
|
if (!g_httpCtx && lstrlen(GetGlobalCfgInfo()->platformServerUrl) > 0) {
|
|
g_httpCtx = new httplib::Client(GetGlobalCfgInfo()->platformServerUrl);
|
|
if (g_httpCtx) {
|
|
g_httpCtx->set_connection_timeout(0, 300000); // 300 milliseconds
|
|
g_httpCtx->set_read_timeout(5, 0); // 5 seconds
|
|
g_httpCtx->set_write_timeout(5, 0); // 5 seconds
|
|
g_httpCtx->set_keep_alive(true);
|
|
g_httpCtx->enable_server_certificate_verification(false);
|
|
}
|
|
}
|
|
|
|
if (aigc::JsonHelper::ObjectToJson(*pReqParams, json)) {
|
|
*pOutJson = _strdup(json.c_str());
|
|
return ERR_SUCCESS;
|
|
}
|
|
|
|
return -ERR_JSON_CREATE;
|
|
}
|
|
|
|
template<class T> int DecodeProtocolResponse(T *pResponse, const TCHAR *pJson) {
|
|
if (aigc::JsonHelper::JsonToObject(*pResponse, pJson)) {
|
|
return ERR_SUCCESS;
|
|
}
|
|
|
|
return -ERR_JSON_DECODE;
|
|
}
|
|
|
|
template<class T1, class T2>
|
|
int ProtolPostMessage(const TCHAR *pUrlPath,
|
|
ProtocolRequest<T1> *pReq,
|
|
ProtocolResponse<T2> *pRsp,
|
|
bool platformServer) {
|
|
int ret;
|
|
httplib::Result res;
|
|
TCHAR *pJson = nullptr;
|
|
|
|
if (lstrlen(GetGlobalCfgInfo()->platformServerUrl) == 0) {
|
|
SPDLOG_ERROR(TEXT("Platform Server URL uninitialize."));
|
|
return -ERR_SYSTEM_UNINITIALIZE;
|
|
}
|
|
|
|
if (pReq == nullptr) {
|
|
SPDLOG_ERROR(TEXT("Input pToken params error"));
|
|
SPDLOG_ERROR(TEXT("Input ProtocolRequest<T1> *pReq params error"));
|
|
return -ERR_INPUT_PARAMS;
|
|
}
|
|
|
|
if (pRsp == nullptr) {
|
|
SPDLOG_ERROR(TEXT("Input ProtocolResponse<T2> *pRsp params error"));
|
|
return -ERR_INPUT_PARAMS;
|
|
}
|
|
|
|
ret = CreateProtocolRequest(pReq, &pJson);
|
|
|
|
if (ret != ERR_SUCCESS) {
|
|
if (pJson) {
|
|
free(pJson);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
if (platformServer) {
|
|
std::string timestamp = std::to_string(time(nullptr)) + "000";
|
|
TCHAR hashValeu[MAX_PATH] = {0};
|
|
TCHAR hashBuf[1024] = {};
|
|
|
|
StringCbPrintf(hashBuf,
|
|
1024,
|
|
TEXT("%s|%s|%s|%s"),
|
|
GetGlobalCfgInfo()->clientId,
|
|
GetGlobalCfgInfo()->clientSecret,
|
|
timestamp.c_str(),
|
|
pJson);
|
|
|
|
if (lstrlen(GetGlobalCfgInfo()->clientSecret) > 0 &&
|
|
CalcHmacHash(HASH_SHA256,
|
|
reinterpret_cast<PUCHAR>(hashBuf),
|
|
lstrlen(hashBuf),
|
|
reinterpret_cast<PUCHAR>(GetGlobalCfgInfo()->clientSecret),
|
|
lstrlen(GetGlobalCfgInfo()->clientSecret),
|
|
hashValeu,
|
|
true) == ERR_SUCCESS) {
|
|
const httplib::Headers headers = {
|
|
{"gzs-client-id", GetGlobalCfgInfo()->clientId},
|
|
{"gzs-sign", hashValeu },
|
|
{"gzs-timestamp", timestamp },
|
|
};
|
|
res = g_httpCtx->Post(pUrlPath, headers, pJson, HTTP_JSON_CONTENT);
|
|
} else {
|
|
res = g_httpCtx->Post(pUrlPath, pJson, HTTP_JSON_CONTENT);
|
|
}
|
|
} else {
|
|
if (g_tunnelHttpCtx == nullptr) {
|
|
free(pJson);
|
|
SPDLOG_ERROR(TEXT("Server Control Service don't connected(g_tunnelHttpCtx is not initialize)."));
|
|
return -ERR_SYSTEM_UNINITIALIZE;
|
|
}
|
|
res = g_tunnelHttpCtx->Post(pUrlPath, pJson, HTTP_JSON_CONTENT);
|
|
}
|
|
|
|
if (res.error() != httplib::Error::Success) {
|
|
SPDLOG_ERROR(TEXT("[{0}]:Post Data {1} error: {2}"), pUrlPath, pJson, httplib::to_string(res.error()));
|
|
free(pJson);
|
|
return -ERR_HTTP_POST_DATA;
|
|
}
|
|
|
|
if (res->status != 200) {
|
|
SPDLOG_ERROR(TEXT("[{0}]:Post Data {1} server return HTTP error: {2}"), pUrlPath, pJson, res->status);
|
|
free(pJson);
|
|
return -ERR_HTTP_SERVER_RSP;
|
|
}
|
|
|
|
SPDLOG_DEBUG(TEXT("+++++ Http Request {0}\n---- Http Response {1}"), pJson, res->body.c_str());
|
|
|
|
free(pJson);
|
|
|
|
if (lstrlen(res->body.c_str()) == 0) {
|
|
SPDLOG_ERROR(TEXT("Server response empty message"));
|
|
return -ERR_READ_FILE;
|
|
}
|
|
|
|
if (DecodeProtocolResponse(pRsp, res->body.c_str()) != ERR_SUCCESS) {
|
|
SPDLOG_ERROR(TEXT("Decode JSON {0} to ProtocolResponse<{1}> error"), res->body, typeid(T2).name());
|
|
return -ERR_JSON_DECODE;
|
|
}
|
|
|
|
return ERR_SUCCESS;
|
|
}
|
|
|
|
#if 0
|
|
template<class T1> int PlatformProtolGetMessage(const TCHAR *pUrlPath, T1 *pRsp) {
|
|
httplib::Result res;
|
|
TCHAR *pJson = nullptr;
|
|
std::string timestamp = std::to_string(time(nullptr)) + "000";
|
|
TCHAR hashValeu[MAX_PATH] = {0};
|
|
TCHAR hashBuf[1024] = {};
|
|
|
|
if (lstrlen(GetGlobalCfgInfo()->platformServerUrl) == 0) {
|
|
SPDLOG_ERROR(TEXT("Platform Server URL uninitialize."));
|
|
return -ERR_SYSTEM_UNINITIALIZE;
|
|
}
|
|
|
|
if (pRsp == nullptr) {
|
|
SPDLOG_ERROR(TEXT("Input ProtocolResponse<T2> *pRsp params error"));
|
|
return -ERR_INPUT_PARAMS;
|
|
}
|
|
|
|
StringCbPrintf(hashBuf,
|
|
1024,
|
|
TEXT("%s|%s|%s|%s"),
|
|
GetGlobalCfgInfo()->clientId,
|
|
GetGlobalCfgInfo()->clientSecret,
|
|
timestamp.c_str(),
|
|
pJson);
|
|
|
|
if (lstrlen(GetGlobalCfgInfo()->clientSecret) > 0 &&
|
|
CalcHmacHash(HASH_SHA256,
|
|
reinterpret_cast<PUCHAR>(hashBuf),
|
|
lstrlen(hashBuf),
|
|
reinterpret_cast<PUCHAR>(GetGlobalCfgInfo()->clientSecret),
|
|
lstrlen(GetGlobalCfgInfo()->clientSecret),
|
|
hashValeu,
|
|
true) == ERR_SUCCESS) {
|
|
const httplib::Headers headers = {
|
|
{"gzs-client-id", GetGlobalCfgInfo()->clientId},
|
|
{"gzs-sign", hashValeu },
|
|
{"gzs-timestamp", timestamp },
|
|
};
|
|
res = g_httpCtx->Get(pUrlPath, headers);
|
|
} else {
|
|
res = g_httpCtx->Get(pUrlPath);
|
|
}
|
|
|
|
if (res.error() != httplib::Error::Success) {
|
|
SPDLOG_ERROR(TEXT("[{0}]:Post Data {1} error: {2}"), pUrlPath, pJson, httplib::to_string(res.error()));
|
|
free(pJson);
|
|
return -ERR_HTTP_POST_DATA;
|
|
}
|
|
|
|
if (res->status != 200) {
|
|
SPDLOG_ERROR(TEXT("[{0}]:Post Data {1} server return HTTP error: {2}"), pUrlPath, pJson, res->status);
|
|
free(pJson);
|
|
return -ERR_HTTP_SERVER_RSP;
|
|
}
|
|
|
|
SPDLOG_DEBUG(TEXT("+++++ Http Request {0}\n---- Http Response {1}"), pJson, res->body.c_str());
|
|
|
|
free(pJson);
|
|
|
|
if (lstrlen(res->body.c_str()) == 0) {
|
|
SPDLOG_ERROR(TEXT("Server response empty message"));
|
|
return -ERR_READ_FILE;
|
|
}
|
|
|
|
if (DecodeProtocolResponse(pRsp, res->body.c_str()) != ERR_SUCCESS) {
|
|
SPDLOG_ERROR(TEXT("Decode JSON {0} to ProtocolResponse<{1}> error"), res->body, typeid(T1).name());
|
|
return -ERR_JSON_DECODE;
|
|
}
|
|
|
|
return ERR_SUCCESS;
|
|
}
|
|
#endif
|
|
|
|
#if !USER_REAL_PLATFORM
|
|
template<class T1, class T2> int PlatformProtolPostMessage(const TCHAR *pUrlPath, T1 *pReq, T2 *pRsp) {
|
|
int ret;
|
|
httplib::Result res;
|
|
TCHAR *pJson = nullptr;
|
|
std::string timestamp = std::to_string(time(nullptr)) + "000";
|
|
TCHAR hashValeu[MAX_PATH] = {0};
|
|
TCHAR hashBuf[1024] = {};
|
|
|
|
if (lstrlen(GetGlobalCfgInfo()->platformServerUrl) == 0) {
|
|
SPDLOG_ERROR(TEXT("Platform Server URL uninitialize."));
|
|
return -ERR_SYSTEM_UNINITIALIZE;
|
|
}
|
|
|
|
if (pReq == nullptr) {
|
|
SPDLOG_ERROR(TEXT("Input pToken params error"));
|
|
SPDLOG_ERROR(TEXT("Input ProtocolRequest<T1> *pReq params error"));
|
|
return -ERR_INPUT_PARAMS;
|
|
}
|
|
|
|
if (pRsp == nullptr) {
|
|
SPDLOG_ERROR(TEXT("Input ProtocolResponse<T2> *pRsp params error"));
|
|
return -ERR_INPUT_PARAMS;
|
|
}
|
|
|
|
ret = CreateProtocolRequest(pReq, &pJson);
|
|
|
|
if (ret != ERR_SUCCESS) {
|
|
if (pJson) {
|
|
free(pJson);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
StringCbPrintf(hashBuf,
|
|
1024,
|
|
TEXT("%s|%s|%s|%s"),
|
|
GetGlobalCfgInfo()->clientId,
|
|
GetGlobalCfgInfo()->clientSecret,
|
|
timestamp.c_str(),
|
|
pJson);
|
|
|
|
if (lstrlen(GetGlobalCfgInfo()->clientSecret) > 0 &&
|
|
CalcHmacHash(HASH_SHA256,
|
|
reinterpret_cast<PUCHAR>(hashBuf),
|
|
lstrlen(hashBuf),
|
|
reinterpret_cast<PUCHAR>(GetGlobalCfgInfo()->clientSecret),
|
|
lstrlen(GetGlobalCfgInfo()->clientSecret),
|
|
hashValeu,
|
|
true) == ERR_SUCCESS) {
|
|
if (typeid(T1) == typeid(PlatformReqClientCfgParms)) {
|
|
const auto *p = reinterpret_cast<PlatformReqClientCfgParms *>(pReq);
|
|
const httplib::Headers headers = {
|
|
{"gzs-client-id", GetGlobalCfgInfo()->clientId },
|
|
{"gzs-sign", hashValeu },
|
|
{"gzs-timestamp", timestamp },
|
|
{"Authorization", ("Bearer " + p->token).c_str()},
|
|
};
|
|
|
|
res = g_httpCtx->Post(pUrlPath, headers, pJson, HTTP_JSON_CONTENT);
|
|
} else {
|
|
const httplib::Headers headers = {
|
|
{"gzs-client-id", GetGlobalCfgInfo()->clientId},
|
|
{"gzs-sign", hashValeu },
|
|
{"gzs-timestamp", timestamp },
|
|
};
|
|
|
|
res = g_httpCtx->Post(pUrlPath, headers, pJson, HTTP_JSON_CONTENT);
|
|
}
|
|
} else {
|
|
if (typeid(T1) == typeid(PlatformReqClientCfgParms)) {
|
|
const auto *p = reinterpret_cast<PlatformReqClientCfgParms *>(pReq);
|
|
const httplib::Headers headers = {
|
|
{"Authorization", ("Bearer " + p->token).c_str()},
|
|
};
|
|
res = g_httpCtx->Post(pUrlPath, headers, pJson, HTTP_JSON_CONTENT);
|
|
} else {
|
|
res = g_httpCtx->Post(pUrlPath, pJson, HTTP_JSON_CONTENT);
|
|
}
|
|
}
|
|
|
|
SPDLOG_DEBUG(TEXT("+++++ Http Request {0}\n---- Http Response {1}"), pJson, res->body.c_str());
|
|
|
|
if (res.error() != httplib::Error::Success) {
|
|
SPDLOG_ERROR(TEXT("[{0}]:Post Data {1} error: {2}"), pUrlPath, pJson, httplib::to_string(res.error()));
|
|
free(pJson);
|
|
return -ERR_HTTP_POST_DATA;
|
|
}
|
|
|
|
if (res->status != 200) {
|
|
SPDLOG_ERROR(TEXT("[{0}]:Post Data {1} server return HTTP error: {2}"), pUrlPath, pJson, res->status);
|
|
free(pJson);
|
|
return -ERR_HTTP_SERVER_RSP;
|
|
}
|
|
|
|
free(pJson);
|
|
|
|
if (lstrlen(res->body.c_str()) == 0) {
|
|
SPDLOG_ERROR(TEXT("Server response empty message"));
|
|
return -ERR_READ_FILE;
|
|
}
|
|
|
|
if (DecodeProtocolResponse(pRsp, res->body.c_str()) != ERR_SUCCESS) {
|
|
SPDLOG_ERROR(TEXT("Decode JSON {0} to ProtocolResponse<{1}> error"), res->body, typeid(T2).name());
|
|
return -ERR_JSON_DECODE;
|
|
}
|
|
|
|
return ERR_SUCCESS;
|
|
}
|
|
#endif
|
|
|
|
template int ProtolPostMessage(const TCHAR *pUrlPath,
|
|
ProtocolRequest<ReqGetUserCfgParams> *pReq,
|
|
ProtocolResponse<RspUserSevrCfgParams> *pRsp,
|
|
bool platformServer);
|
|
|
|
template int ProtolPostMessage(const TCHAR *pUrlPath,
|
|
ProtocolRequest<ReqGetUserCfgParams> *pReq,
|
|
ProtocolResponse<RspUsrCliConfigParams> *pRsp,
|
|
bool platformServer);
|
|
|
|
template int ProtolPostMessage(const TCHAR *pUrlPath,
|
|
ProtocolRequest<ReqUserSetCliCfgParams> *pReq,
|
|
ProtocolResponse<RspUserSetCliCfgParams> *pRsp,
|
|
bool platformServer);
|
|
|
|
template int ProtolPostMessage(const TCHAR *pUrlPath,
|
|
ProtocolRequest<ReqStartTunnelParams> *pReq,
|
|
ProtocolResponse<ResponseStatus> *pRsp,
|
|
bool platformServer);
|
|
|
|
template int ProtolPostMessage(const TCHAR *pUrlPath,
|
|
ProtocolRequest<ReqHeartParams> *pReq,
|
|
ProtocolResponse<RspHeartParams> *pRsp,
|
|
bool platformServer);
|
|
|
|
#if !USER_REAL_PLATFORM
|
|
template int PlatformProtolPostMessage(const TCHAR *pUrlPath,
|
|
PlatformReqServerCfgParms *pReq,
|
|
PlatformRspServerCfgParams *pRsp);
|
|
|
|
template int PlatformProtolPostMessage(const TCHAR *pUrlPath,
|
|
PlatformReqClientCfgParms *pReq,
|
|
PlatformRspClientCfgParams *pRsp);
|
|
|
|
//template int PlatformProtolGetMessage(const TCHAR *pUrlPath, PlatformRspUserClientCfgParams *pRsp);
|
|
#endif |