 Issue 10580043:
  Run a single FilterEngine instance in a separate process  (Closed)
    
  
    Issue 10580043:
  Run a single FilterEngine instance in a separate process  (Closed) 
  | Index: Shared/AdblockPlusClient.cpp | 
| =================================================================== | 
| --- a/Shared/AdblockPlusClient.cpp | 
| +++ b/Shared/AdblockPlusClient.cpp | 
| @@ -8,35 +8,160 @@ | 
| #include "PluginHttpRequest.h" | 
| #include "PluginMutex.h" | 
| #include "PluginClass.h" | 
| +#include "PluginUtil.h" | 
| #include "AdblockPlusClient.h" | 
| +namespace | 
| +{ | 
| + // TODO: pipeName, bufferSize, AutoHandle, ReadMessage, WriteMessage, MarshalStrings and UnmarshalStrings are | 
| + // duplicated in AdblockPlusEngine. We should find a way to reuse them. | 
| + | 
| + const std::wstring pipeName = L"\\\\.\\pipe\\adblockplusengine"; | 
| + const int bufferSize = 1024; | 
| + | 
| + class AutoHandle | 
| + { | 
| + public: | 
| + AutoHandle() | 
| + { | 
| + } | 
| + | 
| + AutoHandle(HANDLE handle) : handle(handle) | 
| + { | 
| + } | 
| + | 
| + ~AutoHandle() | 
| + { | 
| + CloseHandle(handle); | 
| + } | 
| + | 
| + HANDLE get() | 
| + { | 
| + return handle; | 
| + } | 
| + | 
| + private: | 
| + HANDLE handle; | 
| + static int references; | 
| + | 
| + AutoHandle(const AutoHandle& autoHandle); | 
| + AutoHandle& operator=(const AutoHandle& autoHandle); | 
| + }; | 
| + | 
| + std::string MarshalStrings(const std::vector<std::string>& strings) | 
| + { | 
| + // TODO: This is some pretty hacky marshalling, replace it with something more robust | 
| + std::string marshalledStrings; | 
| + for (std::vector<std::string>::const_iterator it = strings.begin(); it != strings.end(); it++) | 
| + marshalledStrings += *it + ';'; | 
| + return marshalledStrings; | 
| + } | 
| + | 
| + std::vector<std::string> UnmarshalStrings(const std::string& message) | 
| + { | 
| + std::stringstream stream(message); | 
| + std::vector<std::string> strings; | 
| + std::string string; | 
| + while (std::getline(stream, string, ';')) | 
| + strings.push_back(string); | 
| + return strings; | 
| + } | 
| + | 
| + std::string ReadMessage(HANDLE pipe) | 
| + { | 
| + std::stringstream stream; | 
| + std::auto_ptr<char> buffer(new char[bufferSize]); | 
| + bool hasError; | 
| + do | 
| + { | 
| + DWORD bytesRead; | 
| + hasError = !ReadFile(pipe, buffer.get(), bufferSize * sizeof(char), &bytesRead, 0); | 
| + if (hasError && GetLastError() != ERROR_MORE_DATA) | 
| + { | 
| + std::stringstream stream; | 
| + stream << "Error reading from pipe: " << GetLastError(); | 
| + throw std::runtime_error(stream.str()); | 
| + } | 
| + stream << std::string(buffer.get(), bytesRead); | 
| + } while (hasError); | 
| + return stream.str(); | 
| + } | 
| + | 
| + void WriteMessage(HANDLE pipe, const std::string& message) | 
| + { | 
| + DWORD bytesWritten; | 
| + if (!WriteFile(pipe, message.c_str(), message.length(), &bytesWritten, 0)) | 
| + throw std::runtime_error("Failed to write to pipe"); | 
| + } | 
| + | 
| + HANDLE OpenPipe(const std::wstring& name) | 
| + { | 
| + if (WaitNamedPipe(name.c_str(), 5000)) | 
| + return CreateFile(name.c_str(), GENERIC_READ | GENERIC_WRITE, 0, 0, OPEN_EXISTING, 0, 0); | 
| + return INVALID_HANDLE_VALUE; | 
| + } | 
| + | 
| + void SpawnAdblockPlusEngine() | 
| + { | 
| + std::wstring engineExecutablePath = DllDir() + L"AdblockPlusEngine.exe"; | 
| + STARTUPINFO startupInfo = {}; | 
| + PROCESS_INFORMATION processInformation = {}; | 
| + | 
| + HANDLE token; | 
| + OpenProcessToken(GetCurrentProcess(), TOKEN_DUPLICATE | TOKEN_ADJUST_DEFAULT | TOKEN_QUERY | TOKEN_ASSIGN_PRIMARY, &token); | 
| + HANDLE newToken; | 
| + DuplicateTokenEx(token, 0, 0, SecurityImpersonation, TokenPrimary, &newToken); | 
| + | 
| + if (!CreateProcessAsUser(newToken, 0, const_cast<wchar_t*>(engineExecutablePath.c_str()), 0, 0, 0, 0, 0, 0, | 
| 
Wladimir Palant
2013/05/16 11:52:16
You are leaking the handles created here: "Handles
 
Felix Dahlke
2013/05/23 12:31:20
Done.
 | 
| + &startupInfo, &processInformation)) | 
| + { | 
| + DWORD error = GetLastError(); | 
| + throw std::runtime_error("Failed to start Adblock Plus Engine"); | 
| + } | 
| + } | 
| + | 
| + HANDLE OpenAdblockPlusEnginePipe() | 
| + { | 
| + try | 
| + { | 
| + HANDLE pipe = OpenPipe(pipeName); | 
| + if (pipe == INVALID_HANDLE_VALUE) | 
| + { | 
| + SpawnAdblockPlusEngine(); | 
| 
Wladimir Palant
2013/05/16 11:52:16
I can't say that I like the logic here, timing ass
 
Oleksandr
2013/05/16 13:23:22
The documentation says "If no instances of the spe
 
Felix Dahlke
2013/05/17 09:11:36
Yes, Oleksandr is right. We are using WaitNamedPip
 
Wladimir Palant
2013/05/17 10:35:02
Ok, I wonder whether we can wait for pipe creation
 | 
| + | 
| + int timeout = 5000; | 
| + while ((pipe = OpenPipe(pipeName)) == INVALID_HANDLE_VALUE) | 
| + { | 
| + const int step = 10; | 
| + Sleep(step); | 
| + timeout -= step; | 
| + if (timeout <= 0) | 
| + throw std::runtime_error("Unable to open Adblock Plus Engine pipe"); | 
| + } | 
| + } | 
| + | 
| + DWORD mode = PIPE_READMODE_MESSAGE; | 
| + if (!SetNamedPipeHandleState(pipe, &mode, 0, 0)) | 
| + throw std::runtime_error("SetNamedPipeHandleState failed"); | 
| + | 
| + return pipe; | 
| + } | 
| + catch(std::exception e) | 
| + { | 
| + DEBUG_GENERAL(e.what()); | 
| + return INVALID_HANDLE_VALUE; | 
| + } | 
| + } | 
| +} | 
| CAdblockPlusClient* CAdblockPlusClient::s_instance = NULL; | 
| - | 
| CAdblockPlusClient::CAdblockPlusClient() : CPluginClientBase() | 
| { | 
| - try | 
| - { | 
| - DEBUG_GENERAL("Building client"); | 
| - m_filter = std::auto_ptr<CPluginFilter>(new CPluginFilter()); | 
| - AdblockPlus::AppInfo appInfo; | 
| - appInfo.name = "adblockplusie"; | 
| - appInfo.version = CT2CA(_T(IEPLUGIN_VERSION), CP_UTF8); | 
| - appInfo.platform = "msie"; | 
| - | 
| - DEBUG_GENERAL(L"Building engine"); | 
| + m_filter = std::auto_ptr<CPluginFilter>(new CPluginFilter()); | 
| +} | 
| - JsEnginePtr jsEngine(AdblockPlus::JsEngine::New(appInfo)); | 
| - filterEngine = std::auto_ptr<AdblockPlus::FilterEngine>(new AdblockPlus::FilterEngine(jsEngine)); | 
| - | 
| - } | 
| - catch(std::exception ex) | 
| - { | 
| - DEBUG_GENERAL(ex.what()); | 
| - } | 
| -} | 
| CAdblockPlusClient::~CAdblockPlusClient() | 
| { | 
| s_instance = NULL; | 
| @@ -63,10 +188,6 @@ | 
| return instance; | 
| } | 
| -AdblockPlus::FilterEngine* CAdblockPlusClient::GetFilterEngine() | 
| -{ | 
| - return filterEngine.get(); | 
| -} | 
| bool CAdblockPlusClient::ShouldBlock(CString src, int contentType, const CString& domain, bool addDebug) | 
| { | 
| @@ -163,3 +284,80 @@ | 
| RegCloseKey(hKey); | 
| return (int)(version[0] - 48); | 
| } | 
| + | 
| +std::string CallAdblockPlusEngineProcedure(const std::string& name, const std::vector<std::string>& args) | 
| 
Wladimir Palant
2013/05/16 11:52:16
Why combine the parameters here? It's a private he
 
Wladimir Palant
2013/05/23 14:25:01
This comment doesn't seem to be addressed.
 
Felix Dahlke
2013/05/23 19:10:44
Oops, seems I missed this. Addressed now, you're r
 | 
| +{ | 
| + AutoHandle pipe(OpenAdblockPlusEnginePipe()); | 
| + std::vector<std::string> strings; | 
| + strings.push_back(name); | 
| + for (std::vector<std::string>::const_iterator it = args.begin(); it != args.end(); it++) | 
| + strings.push_back(*it); | 
| + WriteMessage(pipe.get(), MarshalStrings(strings)); | 
| + return ReadMessage(pipe.get()); | 
| +} | 
| + | 
| +bool CAdblockPlusClient::Matches(const std::string& url, const std::string& contentType, const std::string& domain) | 
| +{ | 
| + std::vector<std::string> args; | 
| + args.push_back(url); | 
| + args.push_back(contentType); | 
| + args.push_back(domain); | 
| + | 
| + try | 
| + { | 
| + std::string response = CallAdblockPlusEngineProcedure("Matches", args); | 
| + return response == "1"; | 
| + } | 
| + catch (const std::exception& e) | 
| + { | 
| + DEBUG_GENERAL(e.what()); | 
| + return false; | 
| + } | 
| +} | 
| + | 
| +std::vector<std::string> CAdblockPlusClient::GetElementHidingSelectors(std::string domain) | 
| +{ | 
| + std::vector<std::string> args; | 
| + args.push_back(domain); | 
| + | 
| + try | 
| + { | 
| + std::string response = CallAdblockPlusEngineProcedure("GetElementHidingSelectors", args); | 
| + return UnmarshalStrings(response); | 
| + } | 
| + catch (const std::exception& e) | 
| + { | 
| + DEBUG_GENERAL(e.what()); | 
| + return std::vector<std::string>(); | 
| + } | 
| +} | 
| + | 
| +std::vector<AdblockPlus::SubscriptionPtr> CAdblockPlusClient::FetchAvailableSubscriptions() | 
| +{ | 
| + //TODO: implement this | 
| + return std::vector<AdblockPlus::SubscriptionPtr>(); | 
| +} | 
| + | 
| +std::vector<AdblockPlus::FilterPtr> CAdblockPlusClient::GetListedFilters() | 
| +{ | 
| + //TODO: implement this | 
| + return std::vector<AdblockPlus::FilterPtr>(); | 
| +} | 
| + | 
| +AdblockPlus::FilterPtr CAdblockPlusClient::GetFilter(std::string text) | 
| +{ | 
| + //TODO: implement this | 
| + return AdblockPlus::FilterPtr(); | 
| +} | 
| + | 
| +std::vector<AdblockPlus::SubscriptionPtr> CAdblockPlusClient::GetListedSubscriptions() | 
| +{ | 
| + //TODO: implement this | 
| + return std::vector<AdblockPlus::SubscriptionPtr>(); | 
| +} | 
| + | 
| +AdblockPlus::SubscriptionPtr CAdblockPlusClient::GetSubscription(std::string url) | 
| +{ | 
| + //TODO: imlement this | 
| + return AdblockPlus::SubscriptionPtr(); | 
| +} |