mirror of
https://github.com/HIllya51/LunaTranslator.git
synced 2025-01-15 08:53:53 +08:00
285 lines
9.8 KiB
C++
285 lines
9.8 KiB
C++
#include<nlohmann/json.hpp>
|
|
#include<filesystem>
|
|
#include<fstream>
|
|
#include<optional>
|
|
#include<thread>
|
|
#include<Windows.h>
|
|
#include<unordered_set>
|
|
#include<set>
|
|
#include<tlhelp32.h>
|
|
static std::wstring StringToWideString(const std::string& text, UINT encoding=CP_UTF8)
|
|
{
|
|
std::vector<wchar_t> buffer(text.size() + 1);
|
|
int length = MultiByteToWideChar(encoding, 0, text.c_str(), text.size() + 1, buffer.data(), buffer.size());
|
|
return std::wstring(buffer.data(), length - 1);
|
|
}
|
|
std::string WideStringToString(const std::wstring& text,UINT cp=CP_UTF8)
|
|
{
|
|
std::vector<char> buffer((text.size() + 1) * 4);
|
|
|
|
WideCharToMultiByte(cp, 0, text.c_str(), -1, buffer.data(), buffer.size(), nullptr, nullptr);
|
|
return buffer.data();
|
|
}
|
|
HANDLE runexe(const std::wstring &exe,const std::optional<std::wstring> &startup_argument)
|
|
{
|
|
STARTUPINFOW si;
|
|
PROCESS_INFORMATION pi;
|
|
|
|
ZeroMemory( &si, sizeof(si) );
|
|
si.cb = sizeof(si);
|
|
ZeroMemory( &pi, sizeof(pi) );
|
|
|
|
std::vector<wchar_t>argu;
|
|
if(startup_argument.has_value()){
|
|
argu.resize(startup_argument.value().size()+1);
|
|
wcscpy(argu.data(),startup_argument.value().c_str());
|
|
}
|
|
CreateProcessW( exe.c_str(), // No module name (use command line)
|
|
startup_argument.has_value()?argu.data(): NULL,
|
|
NULL, // Process handle not inheritable
|
|
NULL, // Thread handle not inheritable
|
|
FALSE, // Set handle inheritance to FALSE
|
|
0, // No creation flags
|
|
NULL, // Use parent's environment block
|
|
NULL, // Use parent's starting directory
|
|
&si, // Pointer to STARTUPINFO structure
|
|
&pi ); // Pointer to PROCESS_INFORMATION structure
|
|
|
|
return pi.hProcess;
|
|
}
|
|
|
|
std::wstring stolower(const std::wstring& s1){
|
|
auto s=s1;
|
|
std::transform(s.begin(), s.end(), s.begin(), tolower);
|
|
return s;
|
|
}
|
|
std::vector<DWORD> EnumerateProcesses(const std::wstring& exe)
|
|
{
|
|
|
|
HANDLE hSnapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
|
|
if (hSnapshot == INVALID_HANDLE_VALUE)
|
|
{
|
|
return {};
|
|
}
|
|
|
|
PROCESSENTRY32 pe32;
|
|
pe32.dwSize = sizeof(PROCESSENTRY32);
|
|
|
|
if (!Process32First(hSnapshot, &pe32))
|
|
{
|
|
CloseHandle(hSnapshot);
|
|
return {};
|
|
}
|
|
std::vector<DWORD> pids;
|
|
do
|
|
{
|
|
if(stolower(exe)==stolower(pe32.szExeFile))
|
|
pids.push_back(pe32.th32ProcessID);
|
|
} while (Process32Next(hSnapshot, &pe32));
|
|
|
|
CloseHandle(hSnapshot);
|
|
return pids;
|
|
}
|
|
enum { STRING = 12, MESSAGE_SIZE = 500, PIPE_BUFFER_SIZE = 50000, SHIFT_JIS = 932, MAX_MODULE_SIZE = 120, PATTERN_SIZE = 30, HOOK_NAME_SIZE = 60, FIXED_SPLIT_VALUE = 0x10001 ,
|
|
HOOKCODE_LEN=500};
|
|
struct ThreadParam
|
|
{
|
|
bool operator==(ThreadParam other) const { return processId == other.processId && addr == other.addr && ctx == other.ctx && ctx2 == other.ctx2; }
|
|
DWORD processId;
|
|
uint64_t addr;
|
|
uint64_t ctx; // The context of the hook: by default the first value on stack, usually the return address
|
|
uint64_t ctx2; // The subcontext of the hook: 0 by default, generated in a method specific to the hook
|
|
};
|
|
struct messagelist{
|
|
bool read;
|
|
int type;
|
|
DWORD pid;
|
|
char name[HOOK_NAME_SIZE];
|
|
wchar_t hookcode[HOOKCODE_LEN];
|
|
ThreadParam tp;
|
|
wchar_t* stringptr;
|
|
uint64_t addr;
|
|
};
|
|
|
|
class lunapatch{
|
|
public:
|
|
HANDLE hMessage;
|
|
HANDLE hwait;
|
|
nlohmann::json config;
|
|
std::map<std::string,std::string>translation;
|
|
std::unordered_set<DWORD>connectedpids;
|
|
void (*Luna_Start)( HANDLE* hRead );
|
|
void (*Luna_Inject)(DWORD pid,LPCWSTR basepath);
|
|
void (*Luna_EmbedSettings)(DWORD pid,UINT32 waittime,UINT8 fontCharSet,bool fontCharSetEnabled,wchar_t *fontFamily,UINT32 spaceadjustpolicy,UINT32 keeprawtext,bool fastskipignore);
|
|
void (*Luna_useembed)(DWORD pid,uint64_t address,uint64_t ctx1,uint64_t ctx2,bool use);
|
|
bool (*Luna_checkisusingembed)(DWORD pid,uint64_t address,uint64_t ctx1,uint64_t ctx2);
|
|
void (*Luna_embedcallback)(DWORD pid,LPCWSTR text,LPCWSTR trans);
|
|
std::set<std::string>notranslation;
|
|
HANDLE hsema;
|
|
lunapatch(std::wstring dll,nlohmann::json&&_translation,nlohmann::json&&_config):translation(_translation),config(_config){
|
|
auto LunaHost=LoadLibraryW(dll.c_str());
|
|
|
|
Luna_Start=(decltype(Luna_Start))GetProcAddress(LunaHost,"Luna_Start");
|
|
Luna_EmbedSettings=(decltype(Luna_EmbedSettings))GetProcAddress(LunaHost,"Luna_EmbedSettings");
|
|
Luna_Inject=(decltype(Luna_Inject))GetProcAddress(LunaHost,"Luna_Inject");
|
|
Luna_useembed=(decltype(Luna_useembed))GetProcAddress(LunaHost,"Luna_useembed");
|
|
Luna_checkisusingembed=(decltype(Luna_checkisusingembed))GetProcAddress(LunaHost,"Luna_checkisusingembed");
|
|
Luna_embedcallback=(decltype(Luna_embedcallback))GetProcAddress(LunaHost,"Luna_embedcallback");
|
|
hsema=CreateSemaphore(NULL,0,100,NULL);
|
|
Luna_Start(&hMessage);
|
|
std::thread([&](){Parsehostmessage();}).detach();
|
|
}
|
|
void run(){
|
|
auto target_exe=StringToWideString(config["target_exe"]);
|
|
|
|
auto _startup_argument=config["startup_argument"];
|
|
|
|
std::optional<std::wstring> startup_argument;
|
|
if(_startup_argument.is_null())
|
|
startup_argument={};
|
|
else
|
|
startup_argument=StringToWideString(config["startup_argument"]);
|
|
hwait=runexe(target_exe,startup_argument);
|
|
}
|
|
~lunapatch(){
|
|
if(notranslation.size()){
|
|
for(auto &text:notranslation){
|
|
translation[text]="";
|
|
}
|
|
auto notrs=nlohmann::json(translation).dump(4);
|
|
std::ofstream of;
|
|
of.open(std::string(config["translation_file"]));
|
|
of<<notrs;
|
|
of.close();
|
|
}
|
|
}
|
|
void wait(){
|
|
WaitForSingleObject(hwait,INFINITE);
|
|
while(connectedpids.size())
|
|
WaitForSingleObject(hsema,INFINITE);
|
|
}
|
|
void inject(){
|
|
//chrome has multi process
|
|
Sleep(config["inject_timeout"]);
|
|
for(auto exe :std::set<std::string>{config["target_exe"],config["target_exe2"]})
|
|
{
|
|
auto pids=EnumerateProcesses(StringToWideString(exe));
|
|
for(auto pid:pids){
|
|
wprintf(L"%d\n",pid);
|
|
Luna_Inject(pid,L"");
|
|
}
|
|
}
|
|
}
|
|
std::wstring findtranslation(const std::wstring& text){
|
|
auto utf8text=WideStringToString(text);
|
|
if(translation.find(utf8text)==translation.end()){
|
|
//wprintf(L"%s\n",text.c_str());
|
|
notranslation.insert(utf8text);
|
|
return {};
|
|
}
|
|
return StringToWideString(translation.at(utf8text));
|
|
}
|
|
void Parsehostmessage(){
|
|
while (true)
|
|
{
|
|
messagelist message;
|
|
DWORD _;
|
|
ReadFile(hMessage,&message,sizeof(message),&_,NULL);
|
|
switch (message.type)
|
|
{
|
|
case 0:
|
|
{
|
|
auto font =StringToWideString(config["embedsettings"]["font"]);
|
|
auto insertspace_policy=config["embedsettings"]["insertspace_policy"];
|
|
auto keeprawtext=config["embedsettings"]["keeprawtext"];
|
|
Luna_EmbedSettings(message.pid,1000,2,false,font.data(),insertspace_policy,keeprawtext,false);
|
|
connectedpids.insert(message.pid);
|
|
}
|
|
break;
|
|
case 1:
|
|
{
|
|
connectedpids.erase(message.pid);
|
|
ReleaseSemaphore(hsema,1,NULL);
|
|
}
|
|
break;
|
|
case 7:
|
|
{
|
|
std::wstring text=message.stringptr;
|
|
auto tp=message.tp;
|
|
for(auto pid:connectedpids)
|
|
{
|
|
if((Luna_checkisusingembed(pid,tp.addr,tp.ctx,tp.ctx2)))
|
|
{
|
|
auto trans=findtranslation(text);
|
|
Luna_embedcallback(pid,text.c_str(),trans.c_str());
|
|
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
case 6:
|
|
{
|
|
std::wstring newhookcode=message.stringptr;
|
|
for(auto hook:config["embedhook"]){
|
|
auto hookcode= StringToWideString(hook[0]);
|
|
uint64_t _addr=hook[1];
|
|
uint64_t _ctx1=hook[2];
|
|
uint64_t _ctx2=hook[3];
|
|
if(hookcode==newhookcode){
|
|
for(auto pid:connectedpids){
|
|
Luna_useembed(pid,message.addr,_ctx1,_ctx2,true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if(message.stringptr)
|
|
free(message.stringptr);
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
std::wstring GetExecutablePath()
|
|
{
|
|
WCHAR buffer[MAX_PATH];
|
|
GetModuleFileNameW(NULL, buffer, MAX_PATH);
|
|
|
|
std::wstring fullPath(buffer);
|
|
size_t pos = fullPath.find_last_of(L"\\/");
|
|
if (pos != std::wstring::npos)
|
|
{
|
|
return fullPath.substr(0, pos);
|
|
}
|
|
|
|
return L"";
|
|
}
|
|
bool checkisapatch(){
|
|
auto curr=std::filesystem::path(GetExecutablePath());
|
|
auto config=curr/"LunaPatch.json";
|
|
if(std::filesystem::exists(config)==false)
|
|
{
|
|
return false;
|
|
}
|
|
std::ifstream jsonfile;
|
|
jsonfile.open(config);
|
|
auto configjson=nlohmann::json::parse(jsonfile);
|
|
jsonfile.close();
|
|
std::string translation_file=configjson["translation_file"];
|
|
|
|
jsonfile.open(translation_file);
|
|
std::map<std::string,std::string> translation=nlohmann::json::parse(jsonfile);
|
|
jsonfile.close();
|
|
|
|
auto LunaHost=(curr/(std::wstring(L"LunaHost")+std::to_wstring(8*sizeof(void*)))).wstring();
|
|
|
|
lunapatch _lunapatch(LunaHost,std::move(translation),std::move(configjson));
|
|
_lunapatch.run();
|
|
_lunapatch.inject();
|
|
_lunapatch.wait();
|
|
return true;
|
|
} |