VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testboxscript/testboxcommand.py@ 68444

Last change on this file since 68444 was 66208, checked in by vboxsync, 8 years ago

ValidationKit: on Windows, mark reboots triggered by the Validation Kit so that they can be spotted in the event log

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.0 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: testboxcommand.py 66208 2017-03-22 16:44:04Z vboxsync $
3
4"""
5TestBox Script - Command Processor.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2016 Oracle Corporation
11
12This file is part of VirtualBox Open Source Edition (OSE), as
13available from http://www.virtualbox.org. This file is free software;
14you can redistribute it and/or modify it under the terms of the GNU
15General Public License (GPL) as published by the Free Software
16Foundation, in version 2 as it comes in the "COPYING" file of the
17VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19
20The contents of this file may alternatively be used under the terms
21of the Common Development and Distribution License Version 1.0
22(CDDL) only, as it comes in the "COPYING.CDDL" file of the
23VirtualBox OSE distribution, in which case the provisions of the
24CDDL are applicable instead of those of the GPL.
25
26You may elect to license modified versions of this file under the
27terms and conditions of either the GPL or the CDDL or both.
28"""
29__version__ = "$Revision: 66208 $"
30
31# Standard python imports.
32import os;
33import sys;
34import threading;
35
36# Validation Kit imports.
37from common import constants;
38from common import utils, webutils;
39import testboxcommons;
40from testboxcommons import TestBoxException;
41from testboxscript import TBS_EXITCODE_NEED_UPGRADE;
42from testboxupgrade import upgradeFromZip;
43from testboxtasks import TestBoxExecTask, TestBoxCleanupTask, TestBoxTestDriverTask;
44
45# Figure where we are.
46try: __file__
47except: __file__ = sys.argv[0];
48g_ksTestScriptDir = os.path.dirname(os.path.abspath(__file__));
49
50
51
52class TestBoxCommand(object):
53 """
54 Implementation of Test Box command.
55 """
56
57 ## The time to wait on the current task to abort.
58 kcSecStopTimeout = 360
59 ## The time to wait on the current task to abort before rebooting.
60 kcSecStopBeforeRebootTimeout = 360
61
62 def __init__(self, oTestBoxScript):
63 """
64 Class instance init
65 """
66 self._oTestBoxScript = oTestBoxScript;
67 self._oCurTaskLock = threading.RLock();
68 self._oCurTask = None;
69
70 # List of available commands and their handlers
71 self._dfnCommands = \
72 {
73 constants.tbresp.CMD_IDLE: self._cmdIdle,
74 constants.tbresp.CMD_WAIT: self._cmdWait,
75 constants.tbresp.CMD_EXEC: self._cmdExec,
76 constants.tbresp.CMD_ABORT: self._cmdAbort,
77 constants.tbresp.CMD_REBOOT: self._cmdReboot,
78 constants.tbresp.CMD_UPGRADE: self._cmdUpgrade,
79 constants.tbresp.CMD_UPGRADE_AND_REBOOT: self._cmdUpgradeAndReboot,
80 constants.tbresp.CMD_SPECIAL: self._cmdSpecial,
81 }
82
83 def _cmdIdle(self, oResponse, oConnection):
84 """
85 Idle response, no ACK.
86 """
87 oResponse.checkParameterCount(1);
88
89 # The dispatch loop will delay for us, so nothing to do here.
90 _ = oConnection; # Leave the connection open.
91 return True;
92
93 def _cmdWait(self, oResponse, oConnection):
94 """
95 Gang scheduling wait response, no ACK.
96 """
97 oResponse.checkParameterCount(1);
98
99 # The dispatch loop will delay for us, so nothing to do here.
100 _ = oConnection; # Leave the connection open.
101 return True;
102
103 def _cmdExec(self, oResponse, oConnection):
104 """
105 Execute incoming command
106 """
107
108 # Check if required parameters given and make a little sense.
109 idResult = oResponse.getIntChecked( constants.tbresp.EXEC_PARAM_RESULT_ID, 1);
110 sScriptZips = oResponse.getStringChecked(constants.tbresp.EXEC_PARAM_SCRIPT_ZIPS);
111 sScriptCmdLine = oResponse.getStringChecked(constants.tbresp.EXEC_PARAM_SCRIPT_CMD_LINE);
112 cSecTimeout = oResponse.getIntChecked( constants.tbresp.EXEC_PARAM_TIMEOUT, 30);
113 oResponse.checkParameterCount(5);
114
115 sScriptFile = utils.argsGetFirst(sScriptCmdLine);
116 if sScriptFile is None:
117 raise TestBoxException('Bad script command line: "%s"' % (sScriptCmdLine,));
118 if len(os.path.basename(sScriptFile)) < len('t.py'):
119 raise TestBoxException('Script file name too short: "%s"' % (sScriptFile,));
120 if len(sScriptZips) < len('x.zip'):
121 raise TestBoxException('Script zip name too short: "%s"' % (sScriptFile,));
122
123 # One task at the time.
124 if self.isRunning():
125 raise TestBoxException('Already running other command');
126
127 # Don't bother running the task without the shares mounted.
128 self._oTestBoxScript.mountShares(); # Raises exception on failure.
129
130 # Kick off the task and ACK the command.
131 self._oCurTaskLock.acquire();
132 try:
133 self._oCurTask = TestBoxExecTask(self._oTestBoxScript, idResult = idResult, sScriptZips = sScriptZips,
134 sScriptCmdLine = sScriptCmdLine, cSecTimeout = cSecTimeout);
135 finally:
136 self._oCurTaskLock.release();
137 oConnection.sendAckAndClose(constants.tbresp.CMD_EXEC);
138 return True;
139
140 def _cmdAbort(self, oResponse, oConnection):
141 """
142 Abort background task
143 """
144 oResponse.checkParameterCount(1);
145 oConnection.sendAck(constants.tbresp.CMD_ABORT);
146
147 oCurTask = self._getCurTask();
148 if oCurTask is not None:
149 oCurTask.terminate();
150 oCurTask.flushLogOnConnection(oConnection);
151 oConnection.close();
152 oCurTask.wait(self.kcSecStopTimeout);
153
154 return True;
155
156 def doReboot(self):
157 """
158 Worker common to _cmdReboot and _doUpgrade that performs a system reboot.
159 """
160 # !! Not more exceptions beyond this point !!
161 testboxcommons.log('Rebooting');
162
163 # Stop anything that might be executing at this point.
164 oCurTask = self._getCurTask();
165 if oCurTask is not None:
166 oCurTask.terminate();
167 oCurTask.wait(self.kcSecStopBeforeRebootTimeout);
168
169 # Invoke shutdown command line utility.
170 sOs = utils.getHostOs();
171 asCmd2 = None;
172 if sOs == 'win':
173 asCmd = ['shutdown', '/r', '/t', '0', '/c', '"ValidationKit triggered reboot"', '/d', '4:1'];
174 elif sOs == 'os2':
175 asCmd = ['setboot', '/B'];
176 elif sOs in ('solaris',):
177 asCmd = ['/usr/sbin/reboot', '-p'];
178 asCmd2 = ['/usr/sbin/reboot']; # Hack! S10 doesn't have -p, but don't know how to reliably detect S10.
179 else:
180 asCmd = ['/sbin/shutdown', '-r', 'now'];
181 try:
182 utils.sudoProcessOutputChecked(asCmd);
183 except Exception, oXcpt:
184 if asCmd2 is not None:
185 try:
186 utils.sudoProcessOutputChecked(asCmd2);
187 except Exception, oXcpt:
188 testboxcommons.log('Error executing reboot command "%s" as well as "%s": %s' % (asCmd, asCmd2, oXcpt));
189 return False;
190 testboxcommons.log('Error executing reboot command "%s": %s' % (asCmd, oXcpt));
191 return False;
192
193 # Quit the script.
194 while True:
195 sys.exit(32);
196 return True;
197
198 def _cmdReboot(self, oResponse, oConnection):
199 """
200 Reboot Test Box
201 """
202 oResponse.checkParameterCount(1);
203 oConnection.sendAckAndClose(constants.tbresp.CMD_REBOOT);
204 return self.doReboot();
205
206 def _doUpgrade(self, oResponse, oConnection, fReboot):
207 """
208 Common worker for _cmdUpgrade and _cmdUpgradeAndReboot.
209 Will sys.exit on success!
210 """
211
212 #
213 # The server specifies a ZIP archive with the new scripts. It's ASSUMED
214 # that the zip is of selected files at g_ksValidationKitDir in SVN. It's
215 # further ASSUMED that we're executing from
216 #
217 sZipUrl = oResponse.getStringChecked(constants.tbresp.UPGRADE_PARAM_URL)
218 oResponse.checkParameterCount(2);
219
220 if utils.isRunningFromCheckout():
221 raise TestBoxException('Cannot upgrade when running from the tree!');
222 oConnection.sendAckAndClose(constants.tbresp.CMD_UPGRADE_AND_REBOOT if fReboot else constants.tbresp.CMD_UPGRADE);
223
224 testboxcommons.log('Upgrading...');
225
226 #
227 # Download the file and install it.
228 #
229 sDstFile = os.path.join(g_ksTestScriptDir, 'VBoxTestBoxScript.zip');
230 if os.path.exists(sDstFile):
231 os.unlink(sDstFile);
232 fRc = webutils.downloadFile(sZipUrl, sDstFile, self._oTestBoxScript.getPathBuilds(), testboxcommons.log);
233 if fRc is not True:
234 return False;
235
236 if upgradeFromZip(sDstFile) is not True:
237 return False;
238
239 #
240 # Restart the system or the script (we have a parent script which
241 # respawns us when we quit).
242 #
243 if fReboot:
244 self.doReboot();
245 sys.exit(TBS_EXITCODE_NEED_UPGRADE);
246
247 def _cmdUpgrade(self, oResponse, oConnection):
248 """
249 Upgrade Test Box Script
250 """
251 return self._doUpgrade(oResponse, oConnection, False);
252
253 def _cmdUpgradeAndReboot(self, oResponse, oConnection):
254 """
255 Upgrade Test Box Script
256 """
257 return self._doUpgrade(oResponse, oConnection, True);
258
259 def _cmdSpecial(self, oResponse, oConnection):
260 """
261 Reserved for future fun.
262 """
263 oConnection.sendReplyAndClose(constants.tbreq.COMMAND_NOTSUP, constants.tbresp.CMD_SPECIAL);
264 testboxcommons.log('Special command %s not supported...' % (oResponse,));
265 return False;
266
267
268 def handleCommand(self, oResponse, oConnection):
269 """
270 Handles a command from the test manager.
271
272 Some commands will close the connection, others (generally the simple
273 ones) wont, leaving the caller the option to use it for log flushing.
274
275 Returns success indicator.
276 Raises no exception.
277 """
278 try:
279 sCmdName = oResponse.getStringChecked(constants.tbresp.ALL_PARAM_RESULT);
280 except Exception, oXcpt:
281 oConnection.close();
282 return False;
283
284 # Do we know the command?
285 fRc = False;
286 if sCmdName in self._dfnCommands:
287 testboxcommons.log(sCmdName);
288 try:
289 # Execute the handler.
290 fRc = self._dfnCommands[sCmdName](oResponse, oConnection)
291 except Exception, oXcpt:
292 # NACK the command if an exception is raised during parameter validation.
293 testboxcommons.log1Xcpt('Exception executing "%s": %s' % (sCmdName, oXcpt));
294 if oConnection.isConnected():
295 try:
296 oConnection.sendReplyAndClose(constants.tbreq.COMMAND_NACK, sCmdName);
297 except Exception, oXcpt2:
298 testboxcommons.log('Failed to NACK "%s": %s' % (sCmdName, oXcpt2));
299 elif sCmdName in [constants.tbresp.STATUS_DEAD, constants.tbresp.STATUS_NACK]:
300 testboxcommons.log('Received status in stead of command: %s' % (sCmdName, ));
301 else:
302 # NOTSUP the unknown command.
303 testboxcommons.log('Received unknown command: %s' % (sCmdName, ));
304 try:
305 oConnection.sendReplyAndClose(constants.tbreq.COMMAND_NOTSUP, sCmdName);
306 except Exception, oXcpt:
307 testboxcommons.log('Failed to NOTSUP "%s": %s' % (sCmdName, oXcpt));
308 return fRc;
309
310 def resumeIncompleteCommand(self):
311 """
312 Resumes an incomplete command at startup.
313
314 The EXEC commands saves essential state information in the scratch area
315 so we can resume them in case the testbox panics or is rebooted.
316 Current "resume" means doing cleanups, but we may need to implement
317 test scenarios involving rebooting the testbox later.
318
319 Returns (idTestBox, sTestBoxName, True) if a command was resumed,
320 otherwise (-1, '', False). Raises no exceptions.
321 """
322
323 try:
324 oTask = TestBoxCleanupTask(self._oTestBoxScript);
325 except:
326 return (-1, '', False);
327
328 self._oCurTaskLock.acquire();
329 self._oCurTask = oTask;
330 self._oCurTaskLock.release();
331
332 return (oTask.idTestBox, oTask.sTestBoxName, True);
333
334 def isRunning(self):
335 """
336 Check if we're running a task or not.
337 """
338 oCurTask = self._getCurTask();
339 return oCurTask is not None and oCurTask.isRunning();
340
341 def flushLogOnConnection(self, oGivenConnection):
342 """
343 Flushes the log of any running task with a log buffer.
344 """
345 oCurTask = self._getCurTask();
346 if oCurTask is not None and isinstance(oCurTask, TestBoxTestDriverTask):
347 return oCurTask.flushLogOnConnection(oGivenConnection);
348 return None;
349
350 def _getCurTask(self):
351 """ Gets the current task in a paranoidly safe manny. """
352 self._oCurTaskLock.acquire();
353 oCurTask = self._oCurTask;
354 self._oCurTaskLock.release();
355 return oCurTask;
356
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette