VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testmanager/core/testboxstatus.py@ 106061

Last change on this file since 106061 was 106061, checked in by vboxsync, 4 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.1 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: testboxstatus.py 106061 2024-09-16 14:03:52Z vboxsync $
3
4"""
5Test Manager - TestBoxStatus.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2024 Oracle and/or its affiliates.
11
12This file is part of VirtualBox base platform packages, as
13available from https://www.virtualbox.org.
14
15This program is free software; you can redistribute it and/or
16modify it under the terms of the GNU General Public License
17as published by the Free Software Foundation, in version 3 of the
18License.
19
20This program is distributed in the hope that it will be useful, but
21WITHOUT ANY WARRANTY; without even the implied warranty of
22MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
23General Public License for more details.
24
25You should have received a copy of the GNU General Public License
26along with this program; if not, see <https://www.gnu.org/licenses>.
27
28The contents of this file may alternatively be used under the terms
29of the Common Development and Distribution License Version 1.0
30(CDDL), a copy of it is provided in the "COPYING.CDDL" file included
31in the VirtualBox distribution, in which case the provisions of the
32CDDL are applicable instead of those of the GPL.
33
34You may elect to license modified versions of this file under the
35terms and conditions of either the GPL or the CDDL or both.
36
37SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
38"""
39__version__ = "$Revision: 106061 $"
40
41
42# Standard python imports.
43import unittest;
44
45# Validation Kit imports.
46from testmanager.core.base import ModelDataBase, ModelDataBaseTestCase, ModelLogicBase, TMTooManyRows, TMRowNotFound;
47from testmanager.core.testbox import TestBoxData;
48
49
50class TestBoxStatusData(ModelDataBase):
51 """
52 TestBoxStatus Data.
53 """
54
55 ## @name TestBoxState_T
56 # @{
57 ksTestBoxState_Idle = 'idle';
58 ksTestBoxState_Testing = 'testing';
59 ksTestBoxState_GangGathering = 'gang-gathering';
60 ksTestBoxState_GangGatheringTimedOut = 'gang-gathering-timedout';
61 ksTestBoxState_GangTesting = 'gang-testing';
62 ksTestBoxState_GangCleanup = 'gang-cleanup';
63 ksTestBoxState_Rebooting = 'rebooting';
64 ksTestBoxState_Upgrading = 'upgrading';
65 ksTestBoxState_UpgradingAndRebooting = 'upgrading-and-rebooting';
66 ksTestBoxState_DoingSpecialCmd = 'doing-special-cmd';
67 ## @}
68
69 ksParam_idTestBox = 'TestBoxStatus_idTestBox';
70 ksParam_idGenTestBox = 'TestBoxStatus_idGenTestBox'
71 ksParam_tsUpdated = 'TestBoxStatus_tsUpdated';
72 ksParam_enmState = 'TestBoxStatus_enmState';
73 ksParam_idTestSet = 'TestBoxStatus_idTestSet';
74 ksParam_iWorkItem = 'TestBoxStatus_iWorkItem';
75
76 kasAllowNullAttributes = ['idTestSet', ];
77 kasValidValues_enmState = \
78 [
79 ksTestBoxState_Idle, ksTestBoxState_Testing, ksTestBoxState_GangGathering,
80 ksTestBoxState_GangGatheringTimedOut, ksTestBoxState_GangTesting, ksTestBoxState_GangCleanup,
81 ksTestBoxState_Rebooting, ksTestBoxState_Upgrading, ksTestBoxState_UpgradingAndRebooting,
82 ksTestBoxState_DoingSpecialCmd,
83 ];
84
85 kcDbColumns = 6;
86
87 def __init__(self):
88 ModelDataBase.__init__(self);
89
90 #
91 # Initialize with defaults.
92 # See the database for explanations of each of these fields.
93 #
94 self.idTestBox = None;
95 self.idGenTestBox = None;
96 self.tsUpdated = None;
97 self.enmState = self.ksTestBoxState_Idle;
98 self.idTestSet = None;
99 self.iWorkItem = None;
100
101 def initFromDbRow(self, aoRow):
102 """
103 Internal worker for initFromDbWithId and initFromDbWithGenId as well as
104 TestBoxStatusLogic.
105 """
106
107 if aoRow is None:
108 raise TMRowNotFound('TestBoxStatus not found.');
109
110 self.idTestBox = aoRow[0];
111 self.idGenTestBox = aoRow[1];
112 self.tsUpdated = aoRow[2];
113 self.enmState = aoRow[3];
114 self.idTestSet = aoRow[4];
115 self.iWorkItem = aoRow[5];
116 return self;
117
118 def initFromDbWithId(self, oDb, idTestBox):
119 """
120 Initialize the object from the database.
121 """
122 oDb.execute('SELECT *\n'
123 'FROM TestBoxStatuses\n'
124 'WHERE idTestBox = %s\n'
125 , (idTestBox, ) );
126 return self.initFromDbRow(oDb.fetchOne());
127
128 def initFromDbWithGenId(self, oDb, idGenTestBox):
129 """
130 Initialize the object from the database.
131 """
132 oDb.execute('SELECT *\n'
133 'FROM TestBoxStatuses\n'
134 'WHERE idGenTestBox = %s\n'
135 , (idGenTestBox, ) );
136 return self.initFromDbRow(oDb.fetchOne());
137
138
139class TestBoxStatusLogic(ModelLogicBase):
140 """
141 TestBoxStatus logic.
142 """
143
144 ## The number of seconds between each time to call touchStatus() when
145 # returning CMD_IDLE.
146 kcSecIdleTouchStatus = 120;
147
148
149 def __init__(self, oDb):
150 ModelLogicBase.__init__(self, oDb);
151
152
153 def tryFetchStatus(self, idTestBox):
154 """
155 Attempts to fetch the status of the given testbox.
156
157 Returns a TestBoxStatusData object on success.
158 Returns None if no status was found.
159 Raises exception on other errors.
160 """
161 self._oDb.execute('SELECT *\n'
162 'FROM TestBoxStatuses\n'
163 'WHERE idTestBox = %s\n',
164 (idTestBox,));
165 if self._oDb.getRowCount() == 0:
166 return None;
167 oStatus = TestBoxStatusData();
168 return oStatus.initFromDbRow(self._oDb.fetchOne());
169
170 def tryFetchStatusAndConfig(self, idTestBox, sTestBoxUuid, sTestBoxAddr):
171 """
172 Tries to fetch the testbox status and current testbox config.
173
174 Returns (TestBoxStatusData, TestBoxData) on success, (None, None) if
175 not found. May throw an exception on database error.
176 """
177 self._oDb.execute('SELECT TestBoxStatuses.*,\n'
178 ' TestBoxesWithStrings.*\n'
179 'FROM TestBoxStatuses,\n'
180 ' TestBoxesWithStrings\n'
181 'WHERE TestBoxStatuses.idTestBox = %s\n'
182 ' AND TestBoxesWithStrings.idTestBox = %s\n'
183 ' AND TestBoxesWithStrings.tsExpire = \'infinity\'::TIMESTAMP\n'
184 ' AND TestBoxesWithStrings.uuidSystem = %s\n'
185 ' AND TestBoxesWithStrings.ip = %s\n'
186 , ( idTestBox,
187 idTestBox,
188 sTestBoxUuid,
189 sTestBoxAddr,) );
190 cRows = self._oDb.getRowCount();
191 if cRows != 1:
192 if cRows != 0:
193 raise TMTooManyRows('tryFetchStatusForCommandReq got %s rows for idTestBox=%s' % (cRows, idTestBox));
194 return (None, None);
195 aoRow = self._oDb.fetchOne();
196 return (TestBoxStatusData().initFromDbRow(aoRow[:TestBoxStatusData.kcDbColumns]),
197 TestBoxData().initFromDbRow(aoRow[TestBoxStatusData.kcDbColumns:]));
198
199
200 def insertIdleStatus(self, idTestBox, idGenTestBox, fCommit = False):
201 """
202 Inserts an idle status for the specified testbox.
203 """
204 self._oDb.execute('INSERT INTO TestBoxStatuses (\n'
205 ' idTestBox,\n'
206 ' idGenTestBox,\n'
207 ' enmState,\n'
208 ' idTestSet,\n'
209 ' iWorkItem)\n'
210 'VALUES ( %s,\n'
211 ' %s,\n'
212 ' \'idle\'::TestBoxState_T,\n'
213 ' NULL,\n'
214 ' 0)\n'
215 , (idTestBox, idGenTestBox) );
216 self._oDb.maybeCommit(fCommit);
217 return True;
218
219 def touchStatus(self, idTestBox, fCommit = False):
220 """
221 Touches the testbox status row, i.e. sets tsUpdated to the current time.
222 """
223 self._oDb.execute('UPDATE TestBoxStatuses\n'
224 'SET tsUpdated = CURRENT_TIMESTAMP\n'
225 'WHERE idTestBox = %s\n'
226 , (idTestBox,));
227 self._oDb.maybeCommit(fCommit);
228 return True;
229
230 def updateState(self, idTestBox, sNewState, idTestSet = None, fCommit = False):
231 """
232 Updates the testbox state.
233 """
234 self._oDb.execute('UPDATE TestBoxStatuses\n'
235 'SET enmState = %s,\n'
236 ' idTestSet = %s,\n'
237 ' tsUpdated = CURRENT_TIMESTAMP\n'
238 'WHERE idTestBox = %s\n',
239 (sNewState, idTestSet, idTestBox));
240 self._oDb.maybeCommit(fCommit);
241 return True;
242
243 def updateGangStatus(self, idTestSetGangLeader, sNewState, fCommit = False):
244 """
245 Update the state of all members of a gang.
246 """
247 self._oDb.execute('UPDATE TestBoxStatuses\n'
248 'SET enmState = %s,\n'
249 ' tsUpdated = CURRENT_TIMESTAMP\n'
250 'WHERE idTestBox IN (SELECT idTestBox\n'
251 ' FROM TestSets\n'
252 ' WHERE idTestSetGangLeader = %s)\n'
253 , (sNewState, idTestSetGangLeader,) );
254 self._oDb.maybeCommit(fCommit);
255 return True;
256
257 def updateWorkItem(self, idTestBox, iWorkItem, fCommit = False):
258 """
259 Updates the testbox state.
260 """
261 self._oDb.execute('UPDATE TestBoxStatuses\n'
262 'SET iWorkItem = %s\n'
263 'WHERE idTestBox = %s\n'
264 , ( iWorkItem, idTestBox,));
265 self._oDb.maybeCommit(fCommit);
266 return True;
267
268 def isWholeGangDoneTesting(self, idTestSetGangLeader):
269 """
270 Checks if the whole gang is done testing.
271 """
272 self._oDb.execute('SELECT COUNT(*)\n'
273 'FROM TestBoxStatuses, TestSets\n'
274 'WHERE TestBoxStatuses.idTestSet = TestSets.idTestSet\n'
275 ' AND TestSets.idTestSetGangLeader = %s\n'
276 ' AND TestBoxStatuses.enmState IN (%s, %s)\n'
277 , ( idTestSetGangLeader,
278 TestBoxStatusData.ksTestBoxState_GangGathering,
279 TestBoxStatusData.ksTestBoxState_GangTesting));
280 return self._oDb.fetchOne()[0] == 0;
281
282 def isTheWholeGangThere(self, idTestSetGangLeader):
283 """
284 Checks if the whole gang is done testing.
285 """
286 self._oDb.execute('SELECT COUNT(*)\n'
287 'FROM TestBoxStatuses, TestSets\n'
288 'WHERE TestBoxStatuses.idTestSet = TestSets.idTestSet\n'
289 ' AND TestSets.idTestSetGangLeader = %s\n'
290 ' AND TestBoxStatuses.enmState IN (%s, %s)\n'
291 , ( idTestSetGangLeader,
292 TestBoxStatusData.ksTestBoxState_GangGathering,
293 TestBoxStatusData.ksTestBoxState_GangTesting));
294 return self._oDb.fetchOne()[0] == 0;
295
296 def timeSinceLastChangeInSecs(self, oStatusData):
297 """
298 Figures the time since the last status change.
299 """
300 tsNow = self._oDb.getCurrentTimestamp();
301 oDelta = tsNow - oStatusData.tsUpdated;
302 return oDelta.seconds + oDelta.days * 24 * 3600;
303
304
305#
306# Unit testing.
307#
308
309# pylint: disable=missing-docstring
310class TestBoxStatusDataTestCase(ModelDataBaseTestCase):
311 def setUp(self):
312 self.aoSamples = [TestBoxStatusData(),];
313
314if __name__ == '__main__':
315 unittest.main();
316 # not reached.
317
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