Left: | ||
Right: |
OLD | NEW |
---|---|
1 #include <AdblockPlus.h> | 1 #include "stdafx.h" |
2 #include <iostream> | 2 |
3 #include <Lmcons.h> | 3 #include "../shared/AutoHandle.h" |
4 #include <ShlObj.h> | 4 #include "../shared/Communication.h" |
Wladimir Palant
2013/05/27 09:51:55
I wonder whether we should add an include director
Oleksandr
2013/05/27 15:32:26
Include directories are generally used for third-p
| |
5 #include <sstream> | |
6 #include <vector> | |
7 #include <Windows.h> | |
8 #include <Sddl.h> | |
9 | 5 |
10 namespace | 6 namespace |
11 { | 7 { |
12 std::wstring GetUserName() | |
13 { | |
14 const DWORD maxLength = UNLEN + 1; | |
15 std::auto_ptr<wchar_t> buffer(new wchar_t[maxLength]); | |
16 DWORD length = maxLength; | |
17 if (!::GetUserName(buffer.get(), &length)) | |
18 { | |
19 std::stringstream stream; | |
20 stream << "Failed to get the current user's name (Error code: " << GetLast Error() << ")"; | |
21 throw std::runtime_error("Failed to get the current user's name"); | |
22 } | |
23 return std::wstring(buffer.get(), length); | |
24 } | |
25 | |
26 const std::wstring pipeName = L"\\\\.\\pipe\\adblockplusengine_" + GetUserName (); | |
27 const int bufferSize = 1024; | |
28 std::auto_ptr<AdblockPlus::FilterEngine> filterEngine; | 8 std::auto_ptr<AdblockPlus::FilterEngine> filterEngine; |
29 | 9 |
30 class AutoHandle | |
31 { | |
32 public: | |
33 AutoHandle() | |
34 { | |
35 } | |
36 | |
37 AutoHandle(HANDLE handle) : handle(handle) | |
38 { | |
39 } | |
40 | |
41 ~AutoHandle() | |
42 { | |
43 CloseHandle(handle); | |
44 } | |
45 | |
46 HANDLE get() | |
47 { | |
48 return handle; | |
49 } | |
50 | |
51 private: | |
52 HANDLE handle; | |
53 | |
54 AutoHandle(const AutoHandle& autoHandle); | |
55 AutoHandle& operator=(const AutoHandle& autoHandle); | |
56 }; | |
57 | |
58 void Log(const std::string& message) | 10 void Log(const std::string& message) |
59 { | 11 { |
60 // TODO: Log to a log file | 12 // TODO: Log to a log file |
61 MessageBoxA(0, ("AdblockPlusEngine: " + message).c_str(), "", MB_OK); | 13 MessageBoxA(0, ("AdblockPlusEngine: " + message).c_str(), "", MB_OK); |
62 } | 14 } |
63 | 15 |
64 void LogLastError(const std::string& message) | 16 void LogLastError(const std::string& message) |
65 { | 17 { |
66 std::stringstream stream; | 18 std::stringstream stream; |
67 stream << message << " (Error code: " << GetLastError() << ")"; | 19 stream << message << " (Error code: " << GetLastError() << ")"; |
68 Log(stream.str()); | 20 Log(stream.str()); |
69 } | 21 } |
70 | 22 |
71 void LogException(const std::exception& exception) | 23 void LogException(const std::exception& exception) |
72 { | 24 { |
73 Log(std::string("An exception occurred: ") + exception.what()); | 25 Log(std::string("An exception occurred: ") + exception.what()); |
74 } | 26 } |
75 | 27 |
76 std::string MarshalStrings(const std::vector<std::string>& strings) | |
77 { | |
78 // TODO: This is some pretty hacky marshalling, replace it with something mo re robust | |
79 std::string marshalledStrings; | |
80 for (std::vector<std::string>::const_iterator it = strings.begin(); it != st rings.end(); it++) | |
81 marshalledStrings += *it + ';'; | |
82 return marshalledStrings; | |
83 } | |
84 | |
85 std::vector<std::string> UnmarshalStrings(const std::string& message) | |
86 { | |
87 std::stringstream stream(message); | |
88 std::vector<std::string> strings; | |
89 std::string string; | |
90 while (std::getline(stream, string, ';')) | |
91 strings.push_back(string); | |
92 return strings; | |
93 } | |
94 | |
95 std::string ToUtf8String(std::wstring str) | 28 std::string ToUtf8String(std::wstring str) |
96 { | 29 { |
97 size_t length = str.size(); | 30 size_t length = str.size(); |
98 if (length == 0) | 31 if (length == 0) |
99 return std::string(); | 32 return std::string(); |
100 | 33 |
101 DWORD utf8StringLength = WideCharToMultiByte(CP_UTF8, 0, str.c_str(), length , 0, 0, 0, 0); | 34 DWORD utf8StringLength = WideCharToMultiByte(CP_UTF8, 0, str.c_str(), length , 0, 0, 0, 0); |
102 if (utf8StringLength == 0) | 35 if (utf8StringLength == 0) |
103 throw std::runtime_error("Failed to determine the required buffer size"); | 36 throw std::runtime_error("Failed to determine the required buffer size"); |
104 | 37 |
105 std::string utf8String(utf8StringLength, '\0'); | 38 std::string utf8String(utf8StringLength, '\0'); |
106 WideCharToMultiByte(CP_UTF8, 0, str.c_str(), length, &utf8String[0], utf8Str ingLength, 0, 0); | 39 WideCharToMultiByte(CP_UTF8, 0, str.c_str(), length, &utf8String[0], utf8Str ingLength, 0, 0); |
107 return utf8String; | 40 return utf8String; |
108 } | 41 } |
109 | 42 |
110 std::string ReadMessage(HANDLE pipe) | |
111 { | |
112 std::stringstream stream; | |
113 std::auto_ptr<char> buffer(new char[bufferSize]); | |
114 bool doneReading = false; | |
115 while (!doneReading) | |
116 { | |
117 DWORD bytesRead; | |
118 if (ReadFile(pipe, buffer.get(), bufferSize * sizeof(char), &bytesRead, 0) ) | |
119 doneReading = true; | |
120 else if (GetLastError() != ERROR_MORE_DATA) | |
121 { | |
122 std::stringstream stream; | |
123 stream << "Error reading from pipe: " << GetLastError(); | |
124 throw std::runtime_error(stream.str()); | |
125 } | |
126 stream << std::string(buffer.get(), bytesRead); | |
127 } | |
128 return stream.str(); | |
129 } | |
130 | |
131 void WriteMessage(HANDLE pipe, const std::string& message) | |
132 { | |
133 DWORD bytesWritten; | |
134 if (!WriteFile(pipe, message.c_str(), message.length(), &bytesWritten, 0)) | |
135 throw std::runtime_error("Failed to write to pipe"); | |
136 } | |
137 | |
138 std::string HandleRequest(const std::vector<std::string>& strings) | 43 std::string HandleRequest(const std::vector<std::string>& strings) |
139 { | 44 { |
140 std::string procedureName = strings[0]; | 45 std::string procedureName = strings[0]; |
141 if (procedureName == "Matches") | 46 if (procedureName == "Matches") |
142 return filterEngine->Matches(strings[1], strings[2], strings[3]) ? "1" : " 0"; | 47 return filterEngine->Matches(strings[1], strings[2], strings[3]) ? "1" : " 0"; |
143 if (procedureName == "GetElementHidingSelectors") | 48 if (procedureName == "GetElementHidingSelectors") |
144 return MarshalStrings(filterEngine->GetElementHidingSelectors(strings[1])) ; | 49 return Communication::MarshalStrings(filterEngine->GetElementHidingSelecto rs(strings[1])); |
145 return ""; | 50 return ""; |
146 } | 51 } |
147 | 52 |
148 DWORD WINAPI ClientThread(LPVOID param) | 53 DWORD WINAPI ClientThread(LPVOID param) |
149 { | 54 { |
150 HANDLE pipe = static_cast<HANDLE>(param); | 55 HANDLE pipe = static_cast<HANDLE>(param); |
151 | 56 |
152 try | 57 try |
153 { | 58 { |
154 std::string message = ReadMessage(pipe); | 59 std::string message = Communication::ReadMessage(pipe); |
155 std::vector<std::string> strings = UnmarshalStrings(message); | 60 std::vector<std::string> strings = Communication::UnmarshalStrings(message ); |
156 std::string response = HandleRequest(strings); | 61 std::string response = HandleRequest(strings); |
157 WriteMessage(pipe, response); | 62 Communication::WriteMessage(pipe, response); |
158 } | 63 } |
159 catch (const std::exception& e) | 64 catch (const std::exception& e) |
160 { | 65 { |
161 LogException(e); | 66 LogException(e); |
162 } | 67 } |
163 | 68 |
164 // TODO: Keep the pipe open until the client disconnects | 69 // TODO: Keep the pipe open until the client disconnects |
165 FlushFileBuffers(pipe); | 70 FlushFileBuffers(pipe); |
166 DisconnectNamedPipe(pipe); | 71 DisconnectNamedPipe(pipe); |
167 CloseHandle(pipe); | 72 CloseHandle(pipe); |
(...skipping 54 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
222 | 127 |
223 // Low mandatory label. See http://msdn.microsoft.com/en-us/library/bb625958.a spx | 128 // Low mandatory label. See http://msdn.microsoft.com/en-us/library/bb625958.a spx |
224 LPCWSTR accessControlEntry = L"S:(ML;;NW;;;LW)"; | 129 LPCWSTR accessControlEntry = L"S:(ML;;NW;;;LW)"; |
225 PSECURITY_DESCRIPTOR securitydescriptor; | 130 PSECURITY_DESCRIPTOR securitydescriptor; |
226 ConvertStringSecurityDescriptorToSecurityDescriptor(accessControlEntry, SDDL_R EVISION_1, &securitydescriptor, 0); | 131 ConvertStringSecurityDescriptorToSecurityDescriptor(accessControlEntry, SDDL_R EVISION_1, &securitydescriptor, 0); |
227 | 132 |
228 sa.lpSecurityDescriptor = securitydescriptor; | 133 sa.lpSecurityDescriptor = securitydescriptor; |
229 sa.bInheritHandle = TRUE; | 134 sa.bInheritHandle = TRUE; |
230 | 135 |
231 HANDLE pipe = CreateNamedPipe(pipeName.c_str(), PIPE_ACCESS_DUPLEX, PIPE_TYPE_ MESSAGE | PIPE_READMODE_MESSAGE | PIPE_WAIT, | 136 HANDLE pipe = CreateNamedPipe(pipeName.c_str(), PIPE_ACCESS_DUPLEX, PIPE_TYPE_ MESSAGE | PIPE_READMODE_MESSAGE | PIPE_WAIT, |
232 PIPE_UNLIMITED_INSTANCES, bufferSize, bufferSize , 0, &sa); | 137 PIPE_UNLIMITED_INSTANCES, Communication::bufferS ize, Communication::bufferSize, 0, &sa); |
233 LocalFree(securitydescriptor); | 138 LocalFree(securitydescriptor); |
234 return pipe; | 139 return pipe; |
235 } | 140 } |
236 | 141 |
237 int WINAPI WinMain(HINSTANCE, HINSTANCE, LPSTR, int) | 142 int WINAPI WinMain(HINSTANCE, HINSTANCE, LPSTR, int) |
238 { | 143 { |
239 // TODO: Attempt to create the pipe first, and exit immediately if this | 144 // TODO: Attempt to create the pipe first, and exit immediately if this |
240 // fails. Since multiple instances of the engine could be running, | 145 // fails. Since multiple instances of the engine could be running, |
241 // this may need named mutices to avoid race conditions. | 146 // this may need named mutices to avoid race conditions. |
242 // Note that as soon as the pipe is created first, we can reduce the | 147 // Note that as soon as the pipe is created first, we can reduce the |
243 // client timeout after CreateProcess(), but should increase the one | 148 // client timeout after CreateProcess(), but should increase the one |
244 // in WaitNamedPipe(). | 149 // in WaitNamedPipe(). |
245 | 150 |
246 filterEngine = CreateFilterEngine(); | 151 filterEngine = CreateFilterEngine(); |
247 | 152 |
248 for (;;) | 153 for (;;) |
249 { | 154 { |
250 HANDLE pipe = CreatePipe(pipeName); | 155 HANDLE pipe = CreatePipe(Communication::pipeName); |
251 if (pipe == INVALID_HANDLE_VALUE) | 156 if (pipe == INVALID_HANDLE_VALUE) |
252 { | 157 { |
253 LogLastError("CreateNamedPipe failed"); | 158 LogLastError("CreateNamedPipe failed"); |
254 return 1; | 159 return 1; |
255 } | 160 } |
256 | 161 |
257 if (!ConnectNamedPipe(pipe, 0)) | 162 if (!ConnectNamedPipe(pipe, 0)) |
258 { | 163 { |
259 LogLastError("Client failed to connect"); | 164 LogLastError("Client failed to connect"); |
260 CloseHandle(pipe); | 165 CloseHandle(pipe); |
261 continue; | 166 continue; |
262 } | 167 } |
263 | 168 |
264 // TODO: Count established connections, kill the engine when none are left | 169 // TODO: Count established connections, kill the engine when none are left |
265 | 170 |
266 AutoHandle thread(CreateThread(0, 0, ClientThread, static_cast<LPVOID>(pipe) , 0, 0)); | 171 AutoHandle thread(CreateThread(0, 0, ClientThread, static_cast<LPVOID>(pipe) , 0, 0)); |
267 if (!thread.get()) | 172 if (!thread.get()) |
268 { | 173 { |
269 LogLastError("CreateThread failed"); | 174 LogLastError("CreateThread failed"); |
270 return 1; | 175 return 1; |
271 } | 176 } |
272 } | 177 } |
273 | 178 |
274 return 0; | 179 return 0; |
275 } | 180 } |
OLD | NEW |