/* Copyright 2021-2023 Peter Repukat - FlatspotSoftware Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ #include "CEFInject.h" #include #define _SSIZE_T_DEFINED #include // seems like a hack to me, but eh #include "../common/nlohmann_json_wstring.h" namespace CEFInject { namespace internal { httplib::Client GetHttpClient(uint16_t port) { httplib::Client cli("localhost", port); cli.set_connection_timeout(1000); return cli; } static uint32_t msg_id = 0; } bool CEFDebugAvailable(uint16_t port) { auto cli = internal::GetHttpClient(port); cli.set_connection_timeout(500); cli.set_read_timeout(500); if (auto res = cli.Get("/json")) { if (res->status == 200) { return true; } } return false; } std::vector AvailableTabs(uint16_t port) { std::vector tabs; auto cli = internal::GetHttpClient(port); if (auto res = cli.Get("/json")) { if (res->status == 200) { const auto json = nlohmann::json::parse(res->body); for (const auto& j : json) { tabs.push_back(j["title"].get()); } } } return tabs; } nlohmann::json InjectJs(const std::wstring& tabname, const std::wstring& js, uint16_t port) { return InjectJs_Unscoped(tabname, L"(async function(){\n" + js + L"\n})();", port); } nlohmann::json InjectJs_Unscoped(const std::wstring &tabname, const std::wstring &js, uint16_t port) { auto cli = internal::GetHttpClient(port); if (auto res = cli.Get("/json")) { if (res->status == 200) { const auto json = nlohmann::json::parse(res->body); for (const auto& tab : json) { if (tab["title"].get().starts_with(tabname)) { #ifdef _WIN32 INT rc; WSADATA wsaData; rc = WSAStartup(MAKEWORD(2, 2), &wsaData); if (rc) { printf("WSAStartup Failed.\n"); return nullptr; } #endif std::shared_ptr ws{ easywsclient::WebSocket::from_url(tab["webSocketDebuggerUrl"].get()) }; if (ws) { ws->send( nlohmann::json{ {"id", internal::msg_id++}, {"method", "Runtime.evaluate"}, {"params", { {"userGesture", true}, {"expression", js} }} }.dump()); nlohmann::json res = nullptr; bool exit = false; while (ws->getReadyState() != easywsclient::WebSocket::CLOSED) { ws->poll(); ws->dispatch([&ws, &res, &exit](const std::string& message) { res = nlohmann::json::parse(message)["result"]["result"]["value"]; exit = true; }); if (exit) { ws->close(); return res; } } #ifdef _WIN32 WSACleanup(); #endif return res; } #ifdef _WIN32 WSACleanup(); #endif return nullptr; } } } } return nullptr; } }