From c62c586a5981877f667448141aa52911bde36eca Mon Sep 17 00:00:00 2001 From: Akash Mozumdar <akashmozumdar@gmail.com> Date: Fri, 15 Jan 2021 09:32:23 -0700 Subject: [PATCH] fix bugs and autoload lua --- GUI/CMakeLists.txt | 11 ++-- cmake/QtUtils.cmake | 20 ------- extensions/CMakeLists.txt | 13 +++-- extensions/devtools.cpp | 66 +++++------------------ extensions/devtools.h | 5 +- extensions/devtoolsdeepltranslate.cpp | 77 ++++++++++++--------------- extensions/lua.cpp | 2 + 7 files changed, 67 insertions(+), 127 deletions(-) diff --git a/GUI/CMakeLists.txt b/GUI/CMakeLists.txt index 944e492..be99474 100644 --- a/GUI/CMakeLists.txt +++ b/GUI/CMakeLists.txt @@ -16,8 +16,11 @@ add_executable(Textractor WIN32 target_precompile_headers(Textractor REUSE_FROM pch) target_link_libraries(Textractor Qt5::Widgets shell32 winhttp) -if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Core.dll) - if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Cored.dll) - install_qt5_libs(Textractor) - endif() +if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Core.dll AND NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5Cored.dll) + add_custom_command(TARGET Textractor + POST_BUILD + COMMAND ${CMAKE_COMMAND} -E remove_directory "${CMAKE_CURRENT_BINARY_DIR}/windeployqt" + COMMAND set PATH=%PATH%$<SEMICOLON>${qt5_install_prefix}/bin + COMMAND Qt5::windeployqt --dir ${CMAKE_FINAL_OUTPUT_DIRECTORY} "${CMAKE_FINAL_OUTPUT_DIRECTORY}/Textractor.exe" + ) endif() diff --git a/cmake/QtUtils.cmake b/cmake/QtUtils.cmake index d2523d0..2c5fd3d 100644 --- a/cmake/QtUtils.cmake +++ b/cmake/QtUtils.cmake @@ -87,23 +87,3 @@ macro(find_qt5) message(FATAL_ERROR "Cannot find QT5!") endif() endmacro(find_qt5) - -# Copies required DLLs to directory with target -# Optionally can provide QML directory as second argument -function(install_qt5_libs target) - if(TARGET Qt5::windeployqt) - set(EXTRA "") - if(EXISTS ${ARGV1}) - message("QML directory to be scanned=${ARGV1}") - list(APPEND EXTRA --qmldir ${ARGV1}) - endif() - - # execute windeployqt in a tmp directory after build - add_custom_command(TARGET ${target} - POST_BUILD - COMMAND ${CMAKE_COMMAND} -E remove_directory "${CMAKE_CURRENT_BINARY_DIR}/windeployqt" - COMMAND set PATH=%PATH%$<SEMICOLON>${qt5_install_prefix}/bin - COMMAND Qt5::windeployqt --dir $<TARGET_FILE_DIR:${target}> "$<TARGET_FILE_DIR:${target}>/$<TARGET_FILE_NAME:${target}>" ${EXTRA} - ) - endif() -endfunction(install_qt5_libs) diff --git a/extensions/CMakeLists.txt b/extensions/CMakeLists.txt index b1bef90..b6ebbbc 100644 --- a/extensions/CMakeLists.txt +++ b/extensions/CMakeLists.txt @@ -44,8 +44,11 @@ target_link_libraries(Lua lua53 Qt5::Widgets) target_link_libraries(Regex\ Filter Qt5::Widgets) target_link_libraries(Thread\ Linker Qt5::Widgets) -if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSockets.dll) - if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSocketsd.dll) - install_qt5_libs(DevTools\ DeepL\ Translate) - endif() -endif() +if (NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSockets.dll AND NOT EXISTS ${CMAKE_FINAL_OUTPUT_DIRECTORY}/Qt5WebSocketsd.dll) + add_custom_command(TARGET DevTools\ DeepL\ Translate + POST_BUILD + COMMAND ${CMAKE_COMMAND} -E remove_directory "${CMAKE_CURRENT_BINARY_DIR}/windeployqt" + COMMAND set PATH=%PATH%$<SEMICOLON>${qt5_install_prefix}/bin + COMMAND Qt5::windeployqt --dir ${CMAKE_FINAL_OUTPUT_DIRECTORY} "${CMAKE_FINAL_OUTPUT_DIRECTORY}/DevTools\ DeepL\ Translate.dll" + ) +endif() diff --git a/extensions/devtools.cpp b/extensions/devtools.cpp index 5c1ba36..a2bc204 100644 --- a/extensions/devtools.cpp +++ b/extensions/devtools.cpp @@ -7,11 +7,10 @@ namespace { std::function<void(QString)> OnStatusChanged = Swallow; PROCESS_INFORMATION processInfo = {}; - std::atomic<int> idCounter = 0, idMethod = 0; + std::atomic<int> idCounter = 0; std::mutex devToolsMutex; QWebSocket webSocket; std::unordered_map<int, concurrency::task_completion_event<JSON::Value<wchar_t>>> mapQueue; - std::unordered_map<std::wstring, std::vector<JSON::Value<wchar_t>>> mapMethod; auto _ = ([] { QObject::connect(&webSocket, &QWebSocket::stateChanged, @@ -21,28 +20,27 @@ namespace { auto result = JSON::Parse(S(message)); std::scoped_lock lock(devToolsMutex); - if (auto method = result[L"method"].String()) - for (auto& [listenTo, results] : mapMethod) if (*method == listenTo) results.push_back(result[L"params"]); - if (auto id = result[L"id"].Number()) - if (auto request = mapQueue.find((int)*id); request != mapQueue.end()) - request->second.set(result), mapQueue.erase(request); + if (auto id = result[L"id"].Number()) if (auto request = mapQueue.find((int)*id); request != mapQueue.end()) + { + request->second.set(result); + mapQueue.erase(request); + } }); }(), 0); } namespace DevTools { - void Start(const std::wstring& path, std::function<void(QString)> statusChanged, bool headless, int port) + void Start(const std::wstring& path, std::function<void(QString)> statusChanged, bool headless) { OnStatusChanged = statusChanged; DWORD exitCode = 0; auto args = FormatString( - L"%s --proxy-server=direct:// --disable-extensions --disable-gpu --user-data-dir=%s\\devtoolscache --remote-debugging-port=%d", + L"%s --proxy-server=direct:// --disable-extensions --disable-gpu --user-data-dir=%s\\devtoolscache --remote-debugging-port=9222", path, - std::filesystem::current_path().wstring(), - port + std::filesystem::current_path().wstring() ); - if (headless) args += L"--headless"; + if (headless) args += L" --headless"; STARTUPINFOW DUMMY = { sizeof(DUMMY) }; if ((GetExitCodeProcess(processInfo.hProcess, &exitCode) && exitCode == STILL_ACTIVE) || CreateProcessW(NULL, args.data(), nullptr, nullptr, FALSE, 0, nullptr, nullptr, &DUMMY, &processInfo) @@ -55,7 +53,7 @@ namespace DevTools L"/json/list", "", NULL, - (DWORD)port, + 9222, NULL, WINHTTP_FLAG_ESCAPE_DISABLE }) @@ -68,29 +66,12 @@ namespace DevTools { std::scoped_lock lock(devToolsMutex); webSocket.open(S(*(*it)[L"webSocketDebuggerUrl"].String())); - if (HttpRequest httpRequest{ - L"Mozilla/5.0 Textractor", - L"127.0.0.1", - L"POST", - L"/json/version", - "", - NULL, - (DWORD)port, - NULL, - WINHTTP_FLAG_ESCAPE_DISABLE - }) - if (auto userAgent = Copy(JSON::Parse(httpRequest.response)[L"User-Agent"].String())) - if (userAgent->find(L"Headless") != std::string::npos) - SendRequest(L"Network.setUserAgentOverride", FormatString(LR"({"userAgent":"%s"})", userAgent->replace(userAgent->find(L"Headless"), 8, L""))); return; } } OnStatusChanged("Failed Connection"); } - else - { - OnStatusChanged("Failed Startup"); - } + else OnStatusChanged("Failed Startup"); } void Close() @@ -98,7 +79,6 @@ namespace DevTools std::scoped_lock lock(devToolsMutex); for (const auto& [_, task] : mapQueue) task.set_exception(std::runtime_error("closed")); webSocket.close(); - mapMethod.clear(); mapQueue.clear(); DWORD exitCode = 0; if (GetExitCodeProcess(processInfo.hProcess, &exitCode) && exitCode == STILL_ACTIVE) @@ -120,12 +100,12 @@ namespace DevTools JSON::Value<wchar_t> SendRequest(const std::wstring& method, const std::wstring& params) { - if (webSocket.state() != QAbstractSocket::ConnectedState) return {}; concurrency::task_completion_event<JSON::Value<wchar_t>> response; - int id = idCounter +=1; + int id = idCounter += 1; auto message = FormatString(LR"({"id":%d,"method":"%s","params":%s})", id, method, params); { std::scoped_lock lock(devToolsMutex); + if (webSocket.state() != QAbstractSocket::ConnectedState) return {}; mapQueue.try_emplace(id, response); webSocket.sendTextMessage(S(message)); webSocket.flush(); @@ -133,22 +113,4 @@ namespace DevTools try { if (auto result = create_task(response).get()[L"result"]) return result; } catch (...) {} return {}; } - - void StartListening(const std::wstring& method) - { - std::scoped_lock lock(devToolsMutex); - mapMethod.try_emplace(method); - } - - std::vector<JSON::Value<wchar_t>> ListenResults(const std::wstring& method) - { - std::scoped_lock lock(devToolsMutex); - return mapMethod[method]; - } - - void StopListening(const std::wstring& method) - { - std::scoped_lock lock(devToolsMutex); - mapMethod.erase(method); - } } diff --git a/extensions/devtools.h b/extensions/devtools.h index ae7d6dd..81c996f 100644 --- a/extensions/devtools.h +++ b/extensions/devtools.h @@ -3,11 +3,8 @@ namespace DevTools { - void Start(const std::wstring& path, std::function<void(QString)> statusChanged, bool headless, int port); + void Start(const std::wstring& path, std::function<void(QString)> statusChanged, bool headless); void Close(); bool Connected(); JSON::Value<wchar_t> SendRequest(const std::wstring& method, const std::wstring& params = L"{}"); - void StartListening(const std::wstring& method); - std::vector<JSON::Value<wchar_t>> ListenResults(const std::wstring& method); - void StopListening(const std::wstring& method); } diff --git a/extensions/devtoolsdeepltranslate.cpp b/extensions/devtoolsdeepltranslate.cpp index 743b4e1..1247d71 100644 --- a/extensions/devtoolsdeepltranslate.cpp +++ b/extensions/devtoolsdeepltranslate.cpp @@ -10,7 +10,7 @@ extern Settings settings; const char* TRANSLATION_PROVIDER = "DevTools DeepL Translate"; const char* GET_API_KEY_FROM = nullptr; bool translateSelectedOnly = true, rateLimitAll = false, rateLimitSelected = false, useCache = true; -int tokenCount = 30, tokenRestoreDelay = 60000, maxSentenceSize = 10000; +int tokenCount = 30, tokenRestoreDelay = 60000, maxSentenceSize = 2500; extern const char* CHROME_LOCATION; extern const char* START_DEVTOOLS; @@ -54,31 +54,50 @@ BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved display->addRow(CHROME_LOCATION, chromePathEdit); auto statusLabel = new QLabel("Stopped"); auto startButton = new QPushButton(START_DEVTOOLS), stopButton = new QPushButton(STOP_DEVTOOLS); - auto headlessCheckBox = new QCheckBox(HEADLESS_MODE); + auto headlessCheckBox = new QCheckBox(); headlessCheckBox->setChecked(settings.value(HEADLESS_MODE, true).toBool()); QObject::connect(headlessCheckBox, &QCheckBox::clicked, [](bool headless) { settings.setValue(HEADLESS_MODE, headless); }); QObject::connect(startButton, &QPushButton::clicked, [statusLabel, chromePathEdit, headlessCheckBox] { DevTools::Start( S(chromePathEdit->text()), - [statusLabel](QString status) { QMetaObject::invokeMethod(statusLabel, std::bind(&QLabel::setText, statusLabel, status)); }, - headlessCheckBox->isChecked(), - 9222 + [statusLabel](QString status) + { + QMetaObject::invokeMethod(statusLabel, std::bind(&QLabel::setText, statusLabel, status)); + if (status == "ConnectedState") std::thread([] + { + if (HttpRequest httpRequest{ + L"Mozilla/5.0 Textractor", + L"127.0.0.1", + L"POST", + L"/json/version", + "", + NULL, + 9222, + NULL, + WINHTTP_FLAG_ESCAPE_DISABLE + }) + if (auto userAgent = Copy(JSON::Parse(httpRequest.response)[L"User-Agent"].String())) + if (userAgent->find(L"Headless") != std::string::npos) + DevTools::SendRequest(L"Network.setUserAgentOverride", + FormatString(LR"({"userAgent":"%s"})", userAgent->replace(userAgent->find(L"Headless"), 8, L""))); + }).detach(); + }, + headlessCheckBox->isChecked() ); - if (!DevTools::SendRequest(L"Network.enable")) DevTools::Close(); }); QObject::connect(stopButton, &QPushButton::clicked, &DevTools::Close); auto buttons = new QHBoxLayout(); buttons->addWidget(startButton); buttons->addWidget(stopButton); - display->addRow(buttons); - display->addRow(headlessCheckBox); - auto autoStartButton = new QCheckBox(AUTO_START); + display->addRow(HEADLESS_MODE, headlessCheckBox); + auto autoStartButton = new QCheckBox(); autoStartButton->setChecked(settings.value(AUTO_START, false).toBool()); - QObject::connect(autoStartButton, &QCheckBox::clicked, [](bool autoStart) {settings.setValue(AUTO_START, autoStart); }); - display->addRow(autoStartButton); + QObject::connect(autoStartButton, &QCheckBox::clicked, [](bool autoStart) { settings.setValue(AUTO_START, autoStart); }); + display->addRow(AUTO_START, autoStartButton); + display->addRow(buttons); statusLabel->setFrameStyle(QFrame::Panel | QFrame::Sunken); display->addRow(DEVTOOLS_STATUS, statusLabel); - if (autoStartButton->isChecked()) startButton->click(); + if (autoStartButton->isChecked()) QMetaObject::invokeMethod(startButton, &QPushButton::click, Qt::QueuedConnection); } break; case DLL_PROCESS_DETACH: @@ -93,39 +112,13 @@ BOOL WINAPI DllMain(HMODULE hModule, DWORD ul_reason_for_call, LPVOID lpReserved std::pair<bool, std::wstring> Translate(const std::wstring& text) { if (!DevTools::Connected()) return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, ERROR_START_CHROME) }; - // DevTools can't handle concurrent translations yet static std::mutex translationMutex; std::scoped_lock lock(translationMutex); - - // Navigate to site and wait until it is loaded - DevTools::StartListening(L"Network.responseReceived"); DevTools::SendRequest(L"Page.navigate", FormatString(LR"({"url":"https://www.deepl.com/translator#any/%s/%s"})", translateTo.Copy(), Escape(text))); - for (int retry = 0; ++retry < 50; Sleep(100)) - for (const auto& result : DevTools::ListenResults(L"Network.responseReceived")) - if (auto URL = result[L"response"][L"url"].String()) - if (URL->find(L"deepl.com/jsonrpc") != std::string::npos) break; - DevTools::StopListening(L"Network.responseReceived"); - - // Extract translation from site - auto RemoveTags = [](const std::wstring& HTML) - { - std::wstring result; - for (unsigned i = 0; i < HTML.size(); ++i) - if (HTML[i] == '<') i = HTML.find('>', i); - else result.push_back(HTML[i]); - return result; - }; - if (auto document = Copy(DevTools::SendRequest(L"DOM.getDocument")[L"root"][L"nodeId"].Number())) - if (auto target = Copy(DevTools::SendRequest( - L"DOM.querySelector", FormatString(LR"({"nodeId":%d,"selector":"#target-dummydiv"})", (int)document.value()) - )[L"nodeId"].Number())) - if (auto outerHTML = Copy(DevTools::SendRequest(L"DOM.getOuterHTML", FormatString(LR"({"nodeId":%d})", (int)target.value()))[L"outerHTML"].String())) - if (auto translation = RemoveTags(outerHTML.value()); !translation.empty()) return { true, translation }; - else if (target = Copy(DevTools::SendRequest( - L"DOM.querySelector", FormatString(LR"({"nodeId":%d,"selector":"div.lmt__system_notification"})", (int)document.value()) - )[L"nodeId"].Number())) - if (outerHTML = Copy(DevTools::SendRequest(L"DOM.getOuterHTML", FormatString(LR"({"nodeId":%d})", (int)target.value()))[L"outerHTML"].String())) - return { false, FormatString(L"%s: %s", TRANSLATION_ERROR, RemoveTags(outerHTML.value())) }; + for (int retry = 0; ++retry < 100; Sleep(100)) + if (auto translation = Copy(DevTools::SendRequest( + L"Runtime.evaluate", LR"({"expression":"document.querySelector('#target-dummydiv').innerHTML","returnByValue":true})")[L"result"][L"value"].String()) + ) if (!translation->empty()) return { true, translation.value() }; return { false, TRANSLATION_ERROR }; } \ No newline at end of file diff --git a/extensions/lua.cpp b/extensions/lua.cpp index 4ab62f0..9d5c272 100644 --- a/extensions/lua.cpp +++ b/extensions/lua.cpp @@ -57,6 +57,8 @@ public: resize(800, 600); setWindowTitle("Lua"); QMetaObject::invokeMethod(this, &QWidget::show, Qt::QueuedConnection); + + LoadScript(); } ~Window()