1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program Execution Server
3  * ---------------------------------------------
4  *
5  * Copyright 2014 The Android Open Source Project
6  *
7  * Licensed under the Apache License, Version 2.0 (the "License");
8  * you may not use this file except in compliance with the License.
9  * You may obtain a copy of the License at
10  *
11  *      http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing, software
14  * distributed under the License is distributed on an "AS IS" BASIS,
15  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16  * See the License for the specific language governing permissions and
17  * limitations under the License.
18  *
19  *//*!
20  * \file
21  * \brief ExecServer Client.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "xsDefs.hpp"
25 #include "xsProtocol.hpp"
26 #include "deSocket.hpp"
27 
28 #include "deString.h"
29 
30 #include <memory>
31 #include <sstream>
32 #include <fstream>
33 #include <cstdio>
34 #include <cstdlib>
35 
36 using std::string;
37 using std::vector;
38 
39 namespace xs
40 {
41 
42 typedef std::auto_ptr<Message> ScopedMsgPtr;
43 
44 class SocketError : public Error
45 {
46 public:
SocketError(deSocketResult result,const char * message,const char * file,int line)47 	SocketError (deSocketResult result, const char* message, const char* file, int line)
48 		: Error		(message, deGetSocketResultName(result), file, line)
49 		, m_result	(result)
50 	{
51 	}
52 
getResult(void) const53 	deSocketResult getResult (void) const
54 	{
55 		return m_result;
56 	}
57 
58 private:
59 	deSocketResult m_result;
60 };
61 
62 // Helpers.
sendMessage(de::Socket & socket,const Message & message)63 void sendMessage (de::Socket& socket, const Message& message)
64 {
65 	// Format message.
66 	vector<deUint8> buf;
67 	message.write(buf);
68 
69 	// Write to socket.
70 	size_t pos = 0;
71 	while (pos < buf.size())
72 	{
73 		size_t			numLeft		= buf.size() - pos;
74 		size_t			numSent		= 0;
75 		deSocketResult	result		= socket.send(&buf[pos], numLeft, &numSent);
76 
77 		if (result != DE_SOCKETRESULT_SUCCESS)
78 			throw SocketError(result, "send() failed", __FILE__, __LINE__);
79 
80 		pos += numSent;
81 	}
82 }
83 
readBytes(de::Socket & socket,vector<deUint8> & dst,size_t numBytes)84 void readBytes (de::Socket& socket, vector<deUint8>& dst, size_t numBytes)
85 {
86 	size_t numRead = 0;
87 	dst.resize(numBytes);
88 	while (numRead < numBytes)
89 	{
90 		size_t			numLeft		= numBytes - numRead;
91 		size_t			curNumRead	= 0;
92 		deSocketResult	result		= socket.receive(&dst[numRead], numLeft, &curNumRead);
93 
94 		if (result != DE_SOCKETRESULT_SUCCESS)
95 			throw SocketError(result, "receive() failed", __FILE__, __LINE__);
96 
97 		numRead += curNumRead;
98 	}
99 }
100 
readMessage(de::Socket & socket)101 Message* readMessage (de::Socket& socket)
102 {
103 	// Header.
104 	vector<deUint8> header;
105 	readBytes(socket, header, MESSAGE_HEADER_SIZE);
106 
107 	MessageType	type;
108 	size_t		messageSize;
109 	Message::parseHeader(&header[0], (int)header.size(), type, messageSize);
110 
111 	// Simple messages without any data.
112 	switch (type)
113 	{
114 		case MESSAGETYPE_KEEPALIVE:				return new KeepAliveMessage();
115 		case MESSAGETYPE_PROCESS_STARTED:		return new ProcessStartedMessage();
116 		default:
117 			break; // Read message with data.
118 	}
119 
120 	vector<deUint8> messageBuf;
121 	readBytes(socket, messageBuf, messageSize-MESSAGE_HEADER_SIZE);
122 
123 	switch (type)
124 	{
125 		case MESSAGETYPE_HELLO:					return new HelloMessage(&messageBuf[0], (int)messageBuf.size());
126 		case MESSAGETYPE_TEST:					return new TestMessage(&messageBuf[0], (int)messageBuf.size());
127 		case MESSAGETYPE_PROCESS_LOG_DATA:		return new ProcessLogDataMessage(&messageBuf[0], (int)messageBuf.size());
128 		case MESSAGETYPE_INFO:					return new InfoMessage(&messageBuf[0], (int)messageBuf.size());
129 		case MESSAGETYPE_PROCESS_LAUNCH_FAILED:	return new ProcessLaunchFailedMessage(&messageBuf[0], (int)messageBuf.size());
130 		case MESSAGETYPE_PROCESS_FINISHED:		return new ProcessFinishedMessage(&messageBuf[0], (int)messageBuf.size());
131 		default:
132 			XS_FAIL("Unknown message");
133 	}
134 }
135 
136 class CommandLine
137 {
138 public:
139 	de::SocketAddress	address;
140 	std::string			program;
141 	std::string			params;
142 	std::string			workingDir;
143 	std::string			caseList;
144 	std::string			dstFileName;
145 };
146 
147 class Client
148 {
149 public:
150 						Client		(const CommandLine& cmdLine);
151 						~Client		(void);
152 
153 	void				run			(void);
154 
155 private:
156 	const CommandLine&	m_cmdLine;
157 	de::Socket			m_socket;
158 };
159 
Client(const CommandLine & cmdLine)160 Client::Client (const CommandLine& cmdLine)
161 	: m_cmdLine(cmdLine)
162 {
163 }
164 
~Client(void)165 Client::~Client (void)
166 {
167 }
168 
run(void)169 void Client::run (void)
170 {
171 	// Connect to server.
172 	m_socket.connect(m_cmdLine.address);
173 
174 	printf("Connected to %s:%d!\n", m_cmdLine.address.getHost(), m_cmdLine.address.getPort());
175 
176 	// Open result file.
177 	std::fstream out(m_cmdLine.dstFileName.c_str(), std::fstream::out|std::fstream::binary);
178 
179 	printf("  writing to %s\n", m_cmdLine.dstFileName.c_str());
180 
181 	// Send execution request.
182 	{
183 		ExecuteBinaryMessage msg;
184 
185 		msg.name		= m_cmdLine.program;
186 		msg.params		= m_cmdLine.params;
187 		msg.workDir		= m_cmdLine.workingDir;
188 		msg.caseList	= m_cmdLine.caseList;
189 
190 		sendMessage(m_socket, msg);
191 		printf("  execution request sent.\n");
192 	}
193 
194 	// Run client loop.
195 	bool isRunning = true;
196 	while (isRunning)
197 	{
198 		ScopedMsgPtr msg(readMessage(m_socket));
199 
200 		switch (msg->type)
201 		{
202 			case MESSAGETYPE_HELLO:
203 				printf("  HelloMessage\n");
204 				break;
205 
206 			case MESSAGETYPE_KEEPALIVE:
207 			{
208 				printf("  KeepAliveMessage\n");
209 
210 				// Reply with keepalive.
211 				sendMessage(m_socket, KeepAliveMessage());
212 				break;
213 			}
214 
215 			case MESSAGETYPE_INFO:
216 				printf("  InfoMessage: '%s'\n", static_cast<InfoMessage*>(msg.get())->info.c_str());
217 				break;
218 
219 			case MESSAGETYPE_PROCESS_STARTED:
220 				printf("  ProcessStartedMessage\n");
221 				break;
222 
223 			case MESSAGETYPE_PROCESS_FINISHED:
224 				printf("  ProcessFinished: exit code = %d\n", static_cast<ProcessFinishedMessage*>(msg.get())->exitCode);
225 				isRunning = false;
226 				break;
227 
228 			case MESSAGETYPE_PROCESS_LAUNCH_FAILED:
229 				printf("  ProcessLaunchFailed: '%s'\n", static_cast<ProcessLaunchFailedMessage*>(msg.get())->reason.c_str());
230 				isRunning = false;
231 				break;
232 
233 			case MESSAGETYPE_PROCESS_LOG_DATA:
234 			{
235 				ProcessLogDataMessage* logDataMsg = static_cast<ProcessLogDataMessage*>(msg.get());
236 				printf("  ProcessLogDataMessage: %d bytes\n", (int)logDataMsg->logData.length());
237 				out << logDataMsg->logData;
238 				break;
239 			}
240 
241 			default:
242 				XS_FAIL("Unknown message");
243 				break;
244 		}
245 	}
246 
247 	// Close output file.
248 	out.close();
249 
250 	// Close connection.
251 	m_socket.shutdown();
252 	m_socket.close();
253 
254 	printf("Done!\n");
255 }
256 
parseString(const char * str)257 string parseString (const char* str)
258 {
259 	if (str[0] == '\'' || str[0] == '"')
260 	{
261 		const char*			p		= str;
262 		char				endChar = *p++;
263 		std::ostringstream	o;
264 
265 		while (*p != endChar && *p)
266 		{
267 			if (*p == '\\')
268 			{
269 				switch (p[1])
270 				{
271 					case 0:		DE_ASSERT(DE_FALSE);	break;
272 					case 'n':	o << '\n';				break;
273 					case 't':	o << '\t';				break;
274 					default:	o << p[1];				break;
275 				}
276 
277 				p += 2;
278 			}
279 			else
280 				o << *p++;
281 		}
282 
283 		return o.str();
284 	}
285 	else
286 		return string(str);
287 }
288 
printHelp(const char * binName)289 void printHelp (const char* binName)
290 {
291 	printf("%s:\n", binName);
292 	printf("  --host=[host]          Connect to host [host]\n");
293 	printf("  --port=[name]          Use port [port]\n");
294 	printf("  --program=[program]    Test program\n");
295 	printf("  --params=[params]      Test program params\n");
296 	printf("  --workdir=[dir]        Working directory\n");
297 	printf("  --caselist=[caselist]  Test case list\n");
298 	printf("  --out=filename         Test result file\n");
299 }
300 
runClient(int argc,const char * const * argv)301 int runClient (int argc, const char* const* argv)
302 {
303 	CommandLine cmdLine;
304 
305 	// Defaults.
306 	cmdLine.address.setHost("127.0.0.1");
307 	cmdLine.address.setPort(50016);
308 	cmdLine.dstFileName = "TestResults.qpa";
309 
310 	// Parse command line.
311 	for (int argNdx = 1; argNdx < argc; argNdx++)
312 	{
313 		const char* arg = argv[argNdx];
314 
315 		if (deStringBeginsWith(arg, "--port="))
316 			cmdLine.address.setPort(atoi(arg+7));
317 		else if (deStringBeginsWith(arg, "--host="))
318 			cmdLine.address.setHost(parseString(arg+7).c_str());
319 		else if (deStringBeginsWith(arg, "--program="))
320 			cmdLine.program = parseString(arg+10);
321 		else if (deStringBeginsWith(arg, "--params="))
322 			cmdLine.params = parseString(arg+9);
323 		else if (deStringBeginsWith(arg, "--workdir="))
324 			cmdLine.workingDir = parseString(arg+10);
325 		else if (deStringBeginsWith(arg, "--caselist="))
326 			cmdLine.caseList = parseString(arg+11);
327 		else if  (deStringBeginsWith(arg, "--out="))
328 			cmdLine.dstFileName = parseString(arg+6);
329 		else
330 		{
331 			printHelp(argv[0]);
332 			return -1;
333 		}
334 	}
335 
336 	// Run client.
337 	try
338 	{
339 		Client client(cmdLine);
340 		client.run();
341 	}
342 	catch (const std::exception& e)
343 	{
344 		printf("%s\n", e.what());
345 		return -1;
346 	}
347 
348 	return 0;
349 }
350 
351 } // xs
352 
main(int argc,const char * const * argv)353 int main (int argc, const char* const* argv)
354 {
355 	return xs::runClient(argc, argv);
356 }
357