VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testmanager/core/testset.py@ 66998

Last change on this file since 66998 was 65980, checked in by vboxsync, 8 years ago

testmanager/core: pylint 2.0.0 fixes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 35.9 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: testset.py 65980 2017-03-07 13:00:36Z vboxsync $
3
4"""
5Test Manager - TestSet.
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: 65980 $"
30
31
32# Standard python imports.
33import os;
34import zipfile;
35import unittest;
36
37# Validation Kit imports.
38from common import utils;
39from testmanager import config;
40from testmanager.core import db;
41from testmanager.core.base import ModelDataBase, ModelDataBaseTestCase, ModelLogicBase, \
42 TMExceptionBase, TMTooManyRows, TMRowNotFound;
43from testmanager.core.testbox import TestBoxData;
44from testmanager.core.testresults import TestResultFileDataEx;
45
46
47class TestSetData(ModelDataBase):
48 """
49 TestSet Data.
50 """
51
52 ## @name TestStatus_T
53 # @{
54 ksTestStatus_Running = 'running';
55 ksTestStatus_Success = 'success';
56 ksTestStatus_Skipped = 'skipped';
57 ksTestStatus_BadTestBox = 'bad-testbox';
58 ksTestStatus_Aborted = 'aborted';
59 ksTestStatus_Failure = 'failure';
60 ksTestStatus_TimedOut = 'timed-out';
61 ksTestStatus_Rebooted = 'rebooted';
62 ## @}
63
64 ## List of relatively harmless (to testgroup/case) statuses.
65 kasHarmlessTestStatuses = [ ksTestStatus_Skipped, ksTestStatus_BadTestBox, ksTestStatus_Aborted, ];
66 ## List of bad statuses.
67 kasBadTestStatuses = [ ksTestStatus_Failure, ksTestStatus_TimedOut, ksTestStatus_Rebooted, ];
68
69 ksIdAttr = 'idTestSet';
70
71 ksParam_idTestSet = 'TestSet_idTestSet';
72 ksParam_tsConfig = 'TestSet_tsConfig';
73 ksParam_tsCreated = 'TestSet_tsCreated';
74 ksParam_tsDone = 'TestSet_tsDone';
75 ksParam_enmStatus = 'TestSet_enmStatus';
76 ksParam_idBuild = 'TestSet_idBuild';
77 ksParam_idBuildCategory = 'TestSet_idBuildCategory';
78 ksParam_idBuildTestSuite = 'TestSet_idBuildTestSuite';
79 ksParam_idGenTestBox = 'TestSet_idGenTestBox';
80 ksParam_idTestBox = 'TestSet_idTestBox';
81 ksParam_idSchedGroup = 'TestSet_idSchedGroup';
82 ksParam_idTestGroup = 'TestSet_idTestGroup';
83 ksParam_idGenTestCase = 'TestSet_idGenTestCase';
84 ksParam_idTestCase = 'TestSet_idTestCase';
85 ksParam_idGenTestCaseArgs = 'TestSet_idGenTestCaseArgs';
86 ksParam_idTestCaseArgs = 'TestSet_idTestCaseArgs';
87 ksParam_idTestResult = 'TestSet_idTestResult';
88 ksParam_sBaseFilename = 'TestSet_sBaseFilename';
89 ksParam_iGangMemberNo = 'TestSet_iGangMemberNo';
90 ksParam_idTestSetGangLeader = 'TestSet_idTestSetGangLeader';
91
92 kasAllowNullAttributes = [ 'tsDone', 'idBuildTestSuite', 'idTestSetGangLeader' ];
93 kasValidValues_enmStatus = [
94 ksTestStatus_Running,
95 ksTestStatus_Success,
96 ksTestStatus_Skipped,
97 ksTestStatus_BadTestBox,
98 ksTestStatus_Aborted,
99 ksTestStatus_Failure,
100 ksTestStatus_TimedOut,
101 ksTestStatus_Rebooted,
102 ];
103 kiMin_iGangMemberNo = 0;
104 kiMax_iGangMemberNo = 1023;
105
106
107 kcDbColumns = 20;
108
109 def __init__(self):
110 ModelDataBase.__init__(self);
111
112 #
113 # Initialize with defaults.
114 # See the database for explanations of each of these fields.
115 #
116 self.idTestSet = None;
117 self.tsConfig = None;
118 self.tsCreated = None;
119 self.tsDone = None;
120 self.enmStatus = 'running';
121 self.idBuild = None;
122 self.idBuildCategory = None;
123 self.idBuildTestSuite = None;
124 self.idGenTestBox = None;
125 self.idTestBox = None;
126 self.idSchedGroup = None;
127 self.idTestGroup = None;
128 self.idGenTestCase = None;
129 self.idTestCase = None;
130 self.idGenTestCaseArgs = None;
131 self.idTestCaseArgs = None;
132 self.idTestResult = None;
133 self.sBaseFilename = None;
134 self.iGangMemberNo = 0;
135 self.idTestSetGangLeader = None;
136
137 def initFromDbRow(self, aoRow):
138 """
139 Internal worker for initFromDbWithId and initFromDbWithGenId as well as
140 TestBoxSetLogic.
141 """
142
143 if aoRow is None:
144 raise TMRowNotFound('TestSet not found.');
145
146 self.idTestSet = aoRow[0];
147 self.tsConfig = aoRow[1];
148 self.tsCreated = aoRow[2];
149 self.tsDone = aoRow[3];
150 self.enmStatus = aoRow[4];
151 self.idBuild = aoRow[5];
152 self.idBuildCategory = aoRow[6];
153 self.idBuildTestSuite = aoRow[7];
154 self.idGenTestBox = aoRow[8];
155 self.idTestBox = aoRow[9];
156 self.idSchedGroup = aoRow[10];
157 self.idTestGroup = aoRow[11];
158 self.idGenTestCase = aoRow[12];
159 self.idTestCase = aoRow[13];
160 self.idGenTestCaseArgs = aoRow[14];
161 self.idTestCaseArgs = aoRow[15];
162 self.idTestResult = aoRow[16];
163 self.sBaseFilename = aoRow[17];
164 self.iGangMemberNo = aoRow[18];
165 self.idTestSetGangLeader = aoRow[19];
166 return self;
167
168
169 def initFromDbWithId(self, oDb, idTestSet):
170 """
171 Initialize the object from the database.
172 """
173 oDb.execute('SELECT *\n'
174 'FROM TestSets\n'
175 'WHERE idTestSet = %s\n'
176 , (idTestSet, ) );
177 aoRow = oDb.fetchOne()
178 if aoRow is None:
179 raise TMRowNotFound('idTestSet=%s not found' % (idTestSet,));
180 return self.initFromDbRow(aoRow);
181
182
183 def openFile(self, sFilename, sMode = 'rb'):
184 """
185 Opens a file.
186
187 Returns (oFile, cbFile, fIsStream) on success.
188 Returns (None, sErrorMsg, None) on failure.
189 Will not raise exceptions, unless the class instance is invalid.
190 """
191 assert sMode in [ 'rb', 'r', 'rU' ];
192
193 # Try raw file first.
194 sFile1 = os.path.join(config.g_ksFileAreaRootDir, self.sBaseFilename + '-' + sFilename);
195 try:
196 oFile = open(sFile1, sMode);
197 return (oFile, os.fstat(oFile.fileno()).st_size, False);
198 except Exception as oXcpt1:
199 # Try the zip archive next.
200 sFile2 = os.path.join(config.g_ksZipFileAreaRootDir, self.sBaseFilename + '.zip');
201 try:
202 oZipFile = zipfile.ZipFile(sFile2, 'r');
203 oFile = oZipFile.open(sFilename, sMode if sMode != 'rb' else 'r');
204 cbFile = oZipFile.getinfo(sFilename).file_size;
205 return (oFile, cbFile, True);
206 except Exception as oXcpt2:
207 # Construct a meaningful error message.
208 try:
209 if os.path.exists(sFile1):
210 return (None, 'Error opening "%s": %s' % (sFile1, oXcpt1), None);
211 if not os.path.exists(sFile2):
212 return (None, 'File "%s" not found. [%s, %s]' % (sFilename, sFile1, sFile2,), None);
213 return (None, 'Error opening "%s" inside "%s": %s' % (sFilename, sFile2, oXcpt2), None);
214 except Exception as oXcpt3:
215 return (None, 'OMG! %s; %s; %s' % (oXcpt1, oXcpt2, oXcpt3,), None);
216 return (None, 'Code not reachable!', None);
217
218 def createFile(self, sFilename, sMode = 'wb'):
219 """
220 Creates a new file.
221
222 Returns oFile on success.
223 Returns sErrorMsg on failure.
224 """
225 assert sMode in [ 'wb', 'w', 'wU' ];
226
227 # Try raw file first.
228 sFile1 = os.path.join(config.g_ksFileAreaRootDir, self.sBaseFilename + '-' + sFilename);
229 try:
230 if not os.path.exists(os.path.dirname(sFile1)):
231 os.makedirs(os.path.dirname(sFile1), 0o755);
232 oFile = open(sFile1, sMode);
233 except Exception as oXcpt1:
234 return str(oXcpt1);
235 return oFile;
236
237 @staticmethod
238 def findLogOffsetForTimestamp(sLogContent, tsTimestamp, offStart = 0, fAfter = False):
239 """
240 Log parsing utility function for finding the offset for the given timestamp.
241
242 We ASSUME the log lines are prefixed with UTC timestamps on the format
243 '09:43:55.789353'.
244
245 Return index into the sLogContent string, 0 if not found.
246 """
247 # Turn tsTimestamp into a string compatible with what we expect to find in the log.
248 oTsZulu = db.dbTimestampToZuluDatetime(tsTimestamp);
249 sWantedTs = oTsZulu.strftime('%H:%M:%S.%f');
250 assert len(sWantedTs) == 15;
251
252 # Now loop thru the string, line by line.
253 offRet = offStart;
254 off = offStart;
255 while True:
256 sThisTs = sLogContent[off : off + 15];
257 if len(sThisTs) >= 15 \
258 and sThisTs[2] == ':' \
259 and sThisTs[5] == ':' \
260 and sThisTs[8] == '.' \
261 and sThisTs[14] in '0123456789':
262 if sThisTs < sWantedTs:
263 offRet = off;
264 elif sThisTs == sWantedTs:
265 if not fAfter:
266 return off;
267 offRet = off;
268 else:
269 if fAfter:
270 offRet = off;
271 break;
272
273 # next line.
274 off = sLogContent.find('\n', off);
275 if off < 0:
276 if fAfter:
277 offRet = len(sLogContent);
278 break;
279 off += 1;
280
281 return offRet;
282
283 @staticmethod
284 def extractLogSection(sLogContent, tsStart, tsLast):
285 """
286 Returns log section from tsStart to tsLast (or all if we cannot make sense of it).
287 """
288 offStart = TestSetData.findLogOffsetForTimestamp(sLogContent, tsStart);
289 offEnd = TestSetData.findLogOffsetForTimestamp(sLogContent, tsLast, offStart, fAfter = True);
290 return sLogContent[offStart : offEnd];
291
292 @staticmethod
293 def extractLogSectionElapsed(sLogContent, tsStart, tsElapsed):
294 """
295 Returns log section from tsStart and tsElapsed forward (or all if we cannot make sense of it).
296 """
297 tsStart = db.dbTimestampToZuluDatetime(tsStart);
298 tsLast = tsStart + tsElapsed;
299 return TestSetData.extractLogSection(sLogContent, tsStart, tsLast);
300
301
302
303class TestSetLogic(ModelLogicBase):
304 """
305 TestSet logic.
306 """
307
308
309 def __init__(self, oDb):
310 ModelLogicBase.__init__(self, oDb);
311
312
313 def tryFetch(self, idTestSet):
314 """
315 Attempts to fetch a test set.
316
317 Returns a TestSetData object on success.
318 Returns None if no status was found.
319 Raises exception on other errors.
320 """
321 self._oDb.execute('SELECT *\n'
322 'FROM TestSets\n'
323 'WHERE idTestSet = %s\n',
324 (idTestSet,));
325 if self._oDb.getRowCount() == 0:
326 return None;
327 oData = TestSetData();
328 return oData.initFromDbRow(self._oDb.fetchOne());
329
330 def strTabString(self, sString, fCommit = False):
331 """
332 Gets the string table id for the given string, adding it if new.
333 """
334 ## @todo move this and make a stored procedure for it.
335 self._oDb.execute('SELECT idStr\n'
336 'FROM TestResultStrTab\n'
337 'WHERE sValue = %s'
338 , (sString,));
339 if self._oDb.getRowCount() == 0:
340 self._oDb.execute('INSERT INTO TestResultStrTab (sValue)\n'
341 'VALUES (%s)\n'
342 'RETURNING idStr\n'
343 , (sString,));
344 if fCommit:
345 self._oDb.commit();
346 return self._oDb.fetchOne()[0];
347
348 def complete(self, idTestSet, sStatus, fCommit = False):
349 """
350 Completes the testset.
351 Returns the test set ID of the gang leader, None if no gang involvement.
352 Raises exceptions on database errors and invalid input.
353 """
354
355 assert sStatus != TestSetData.ksTestStatus_Running;
356
357 #
358 # Get the basic test set data and check if there is anything to do here.
359 #
360 oData = TestSetData().initFromDbWithId(self._oDb, idTestSet);
361 if oData.enmStatus != TestSetData.ksTestStatus_Running:
362 raise TMExceptionBase('TestSet %s is already completed as %s.' % (idTestSet, oData.enmStatus));
363 if oData.idTestResult is None:
364 raise self._oDb.integrityException('idTestResult is NULL for TestSet %u' % (idTestSet,));
365
366 #
367 # Close open sub test results, count these as errors.
368 # Note! No need to propagate error counts here. Only one tree line will
369 # have open sets, and it will go all the way to the root.
370 #
371 self._oDb.execute('SELECT idTestResult\n'
372 'FROM TestResults\n'
373 'WHERE idTestSet = %s\n'
374 ' AND enmStatus = %s\n'
375 ' AND idTestResult <> %s\n'
376 'ORDER BY idTestResult DESC\n'
377 , (idTestSet, TestSetData.ksTestStatus_Running, oData.idTestResult));
378 aaoRows = self._oDb.fetchAll();
379 if aaoRows:
380 idStr = self.strTabString('Unclosed test result', fCommit = fCommit);
381 for aoRow in aaoRows:
382 self._oDb.execute('UPDATE TestResults\n'
383 'SET enmStatus = \'failure\',\n'
384 ' tsElapsed = CURRENT_TIMESTAMP - tsCreated,\n'
385 ' cErrors = cErrors + 1\n'
386 'WHERE idTestResult = %s\n'
387 , (aoRow[0],));
388 self._oDb.execute('INSERT INTO TestResultMsgs (idTestResult, idTestSet, idStrMsg, enmLevel)\n'
389 'VALUES ( %s, %s, %s, \'failure\'::TestResultMsgLevel_T)\n'
390 , (aoRow[0], idTestSet, idStr,));
391
392 #
393 # If it's a success result, check it against error counters.
394 #
395 if sStatus not in TestSetData.kasBadTestStatuses:
396 self._oDb.execute('SELECT COUNT(*)\n'
397 'FROM TestResults\n'
398 'WHERE idTestSet = %s\n'
399 ' AND cErrors > 0\n'
400 , (idTestSet,));
401 cErrors = self._oDb.fetchOne()[0];
402 if cErrors > 0:
403 sStatus = TestSetData.ksTestStatus_Failure;
404
405 #
406 # If it's an pure 'failure', check for timeouts and propagate it.
407 #
408 if sStatus == TestSetData.ksTestStatus_Failure:
409 self._oDb.execute('SELECT COUNT(*)\n'
410 'FROM TestResults\n'
411 'WHERE idTestSet = %s\n'
412 ' AND enmStatus = %s\n'
413 , ( idTestSet, TestSetData.ksTestStatus_TimedOut, ));
414 if self._oDb.fetchOne()[0] > 0:
415 sStatus = TestSetData.ksTestStatus_TimedOut;
416
417 #
418 # Complete the top level test result and then the test set.
419 #
420 self._oDb.execute('UPDATE TestResults\n'
421 'SET cErrors = (SELECT COALESCE(SUM(cErrors), 0)\n'
422 ' FROM TestResults\n'
423 ' WHERE idTestResultParent = %s)\n'
424 'WHERE idTestResult = %s\n'
425 'RETURNING cErrors\n'
426 , (oData.idTestResult, oData.idTestResult));
427 cErrors = self._oDb.fetchOne()[0];
428 if cErrors == 0 and sStatus in TestSetData.kasBadTestStatuses:
429 self._oDb.execute('UPDATE TestResults\n'
430 'SET cErrors = 1\n'
431 'WHERE idTestResult = %s\n'
432 , (oData.idTestResult,));
433 elif cErrors > 0 and sStatus not in TestSetData.kasBadTestStatuses:
434 sStatus = TestSetData.ksTestStatus_Failure; # Impossible.
435 self._oDb.execute('UPDATE TestResults\n'
436 'SET enmStatus = %s,\n'
437 ' tsElapsed = CURRENT_TIMESTAMP - tsCreated\n'
438 'WHERE idTestResult = %s\n'
439 , (sStatus, oData.idTestResult,));
440
441 self._oDb.execute('UPDATE TestSets\n'
442 'SET enmStatus = %s,\n'
443 ' tsDone = CURRENT_TIMESTAMP\n'
444 'WHERE idTestSet = %s\n'
445 , (sStatus, idTestSet,));
446
447 self._oDb.maybeCommit(fCommit);
448 return oData.idTestSetGangLeader;
449
450 def completeAsAbandoned(self, idTestSet, fCommit = False):
451 """
452 Completes the testset as abandoned if necessary.
453
454 See scenario #9:
455 file://../../docs/AutomaticTestingRevamp.html#cleaning-up-abandond-testcase
456
457 Returns True if successfully completed as abandond, False if it's already
458 completed, and raises exceptions under exceptional circumstances.
459 """
460
461 #
462 # Get the basic test set data and check if there is anything to do here.
463 #
464 oData = self.tryFetch(idTestSet);
465 if oData is None:
466 return False;
467 if oData.enmStatus != TestSetData.ksTestStatus_Running:
468 return False;
469
470 if oData.idTestResult is not None:
471 #
472 # Clean up test results, adding a message why they failed.
473 #
474 self._oDb.execute('UPDATE TestResults\n'
475 'SET enmStatus = \'failure\',\n'
476 ' tsElapsed = CURRENT_TIMESTAMP - tsCreated,\n'
477 ' cErrors = cErrors + 1\n'
478 'WHERE idTestSet = %s\n'
479 ' AND enmStatus = \'running\'::TestStatus_T\n'
480 , (idTestSet,));
481
482 idStr = self.strTabString('The test was abandond by the testbox', fCommit = fCommit);
483 self._oDb.execute('INSERT INTO TestResultMsgs (idTestResult, idTestSet, idStrMsg, enmLevel)\n'
484 'VALUES ( %s, %s, %s, \'failure\'::TestResultMsgLevel_T)\n'
485 , (oData.idTestResult, idTestSet, idStr,));
486
487 #
488 # Complete the testset.
489 #
490 self._oDb.execute('UPDATE TestSets\n'
491 'SET enmStatus = \'failure\',\n'
492 ' tsDone = CURRENT_TIMESTAMP\n'
493 'WHERE idTestSet = %s\n'
494 ' AND enmStatus = \'running\'::TestStatus_T\n'
495 , (idTestSet,));
496
497 self._oDb.maybeCommit(fCommit);
498 return True;
499
500 def completeAsGangGatheringTimeout(self, idTestSet, fCommit = False):
501 """
502 Completes the testset with a gang-gathering timeout.
503 Raises exceptions on database errors and invalid input.
504 """
505 #
506 # Get the basic test set data and check if there is anything to do here.
507 #
508 oData = TestSetData().initFromDbWithId(self._oDb, idTestSet);
509 if oData.enmStatus != TestSetData.ksTestStatus_Running:
510 raise TMExceptionBase('TestSet %s is already completed as %s.' % (idTestSet, oData.enmStatus));
511 if oData.idTestResult is None:
512 raise self._oDb.integrityException('idTestResult is NULL for TestSet %u' % (idTestSet,));
513
514 #
515 # Complete the top level test result and then the test set.
516 #
517 self._oDb.execute('UPDATE TestResults\n'
518 'SET enmStatus = \'failure\',\n'
519 ' tsElapsed = CURRENT_TIMESTAMP - tsCreated,\n'
520 ' cErrors = cErrors + 1\n'
521 'WHERE idTestSet = %s\n'
522 ' AND enmStatus = \'running\'::TestStatus_T\n'
523 , (idTestSet,));
524
525 idStr = self.strTabString('Gang gathering timed out', fCommit = fCommit);
526 self._oDb.execute('INSERT INTO TestResultMsgs (idTestResult, idTestSet, idStrMsg, enmLevel)\n'
527 'VALUES ( %s, %s, %s, \'failure\'::TestResultMsgLevel_T)\n'
528 , (oData.idTestResult, idTestSet, idStr,));
529
530 self._oDb.execute('UPDATE TestSets\n'
531 'SET enmStatus = \'failure\',\n'
532 ' tsDone = CURRENT_TIMESTAMP\n'
533 'WHERE idTestSet = %s\n'
534 , (idTestSet,));
535
536 self._oDb.maybeCommit(fCommit);
537 return True;
538
539 def createFile(self, oTestSet, sName, sMime, sKind, sDesc, cbFile, fCommit = False): # pylint: disable=R0914
540 """
541 Creates a file and associating with the current test result record in
542 the test set.
543
544 Returns file object that the file content can be written to.
545 Raises exception on database error, I/O errors, if there are too many
546 files in the test set or if they take up too much disk space.
547
548 The caller (testboxdisp.py) is expected to do basic input validation,
549 so we skip that and get on with the bits only we can do.
550 """
551
552 #
553 # Furhter input and limit checks.
554 #
555 if oTestSet.enmStatus != TestSetData.ksTestStatus_Running:
556 raise TMExceptionBase('Cannot create files on a test set with status "%s".' % (oTestSet.enmStatus,));
557
558 self._oDb.execute('SELECT TestResultStrTab.sValue\n'
559 'FROM TestResultFiles,\n'
560 ' TestResults,\n'
561 ' TestResultStrTab\n'
562 'WHERE TestResults.idTestSet = %s\n'
563 ' AND TestResultFiles.idTestResult = TestResults.idTestResult\n'
564 ' AND TestResultStrTab.idStr = TestResultFiles.idStrFile\n'
565 , ( oTestSet.idTestSet,));
566 if self._oDb.getRowCount() + 1 > config.g_kcMaxUploads:
567 raise TMExceptionBase('Uploaded too many files already (%d).' % (self._oDb.getRowCount(),));
568
569 dFiles = {}
570 cbTotalFiles = 0;
571 for aoRow in self._oDb.fetchAll():
572 dFiles[aoRow[0].lower()] = 1; # For determining a unique filename further down.
573 sFile = os.path.join(config.g_ksFileAreaRootDir, oTestSet.sBaseFilename + '-' + aoRow[0]);
574 try:
575 cbTotalFiles += os.path.getsize(sFile);
576 except:
577 cbTotalFiles += config.g_kcMbMaxUploadSingle * 1048576;
578 if (cbTotalFiles + cbFile + 1048575) / 1048576 > config.g_kcMbMaxUploadTotal:
579 raise TMExceptionBase('Will exceed total upload limit: %u bytes + %u bytes > %s MiB.' \
580 % (cbTotalFiles, cbFile, config.g_kcMbMaxUploadTotal));
581
582 #
583 # Create a new file.
584 #
585 self._oDb.execute('SELECT idTestResult\n'
586 'FROM TestResults\n'
587 'WHERE idTestSet = %s\n'
588 ' AND enmStatus = \'running\'::TestStatus_T\n'
589 'ORDER BY idTestResult DESC\n'
590 'LIMIT 1\n'
591 % ( oTestSet.idTestSet, ));
592 if self._oDb.getRowCount() < 1:
593 raise TMExceptionBase('No open test results - someone committed a capital offence or we ran into a race.');
594 idTestResult = self._oDb.fetchOne()[0];
595
596 if sName.lower() in dFiles:
597 # Note! There is in theory a race here, but that's something the
598 # test driver doing parallel upload with non-unique names
599 # should worry about. The TD should always avoid this path.
600 sOrgName = sName;
601 for i in range(2, config.g_kcMaxUploads + 6):
602 sName = '%s-%s' % (i, sName,);
603 if sName not in dFiles:
604 break;
605 sName = None;
606 if sName is None:
607 raise TMExceptionBase('Failed to find unique name for %s.' % (sOrgName,));
608
609 self._oDb.execute('INSERT INTO TestResultFiles(idTestResult, idTestSet, idStrFile, idStrDescription,\n'
610 ' idStrKind, idStrMime)\n'
611 'VALUES (%s, %s, %s, %s, %s, %s)\n'
612 , ( idTestResult,
613 oTestSet.idTestSet,
614 self.strTabString(sName),
615 self.strTabString(sDesc),
616 self.strTabString(sKind),
617 self.strTabString(sMime),
618 ));
619
620 oFile = oTestSet.createFile(sName, 'wb');
621 if utils.isString(oFile):
622 raise TMExceptionBase('Error creating "%s": %s' % (sName, oFile));
623 self._oDb.maybeCommit(fCommit);
624 return oFile;
625
626 def getGang(self, idTestSetGangLeader):
627 """
628 Returns an array of TestBoxData object representing the gang for the given testset.
629 """
630 self._oDb.execute('SELECT TestBoxesWithStrings.*\n'
631 'FROM TestBoxesWithStrings,\n'
632 ' TestSets'
633 'WHERE TestSets.idTestSetGangLeader = %s\n'
634 ' AND TestSets.idGenTestBox = TestBoxesWithStrings.idGenTestBox\n'
635 'ORDER BY iGangMemberNo ASC\n'
636 , ( idTestSetGangLeader,));
637 aaoRows = self._oDb.fetchAll();
638 aoTestBoxes = [];
639 for aoRow in aaoRows:
640 aoTestBoxes.append(TestBoxData().initFromDbRow(aoRow));
641 return aoTestBoxes;
642
643 def getFile(self, idTestSet, idTestResultFile):
644 """
645 Gets the TestResultFileEx corresponding to idTestResultFile.
646
647 Raises an exception if the file wasn't found, doesn't belong to
648 idTestSet, and on DB error.
649 """
650 self._oDb.execute('SELECT TestResultFiles.*,\n'
651 ' StrTabFile.sValue AS sFile,\n'
652 ' StrTabDesc.sValue AS sDescription,\n'
653 ' StrTabKind.sValue AS sKind,\n'
654 ' StrTabMime.sValue AS sMime\n'
655 'FROM TestResultFiles,\n'
656 ' TestResultStrTab AS StrTabFile,\n'
657 ' TestResultStrTab AS StrTabDesc,\n'
658 ' TestResultStrTab AS StrTabKind,\n'
659 ' TestResultStrTab AS StrTabMime,\n'
660 ' TestResults\n'
661 'WHERE TestResultFiles.idTestResultFile = %s\n'
662 ' AND TestResultFiles.idStrFile = StrTabFile.idStr\n'
663 ' AND TestResultFiles.idStrDescription = StrTabDesc.idStr\n'
664 ' AND TestResultFiles.idStrKind = StrTabKind.idStr\n'
665 ' AND TestResultFiles.idStrMime = StrTabMime.idStr\n'
666 ' AND TestResults.idTestResult = TestResultFiles.idTestResult\n'
667 ' AND TestResults.idTestSet = %s\n'
668 , ( idTestResultFile, idTestSet, ));
669 return TestResultFileDataEx().initFromDbRow(self._oDb.fetchOne());
670
671
672 def getById(self, idTestSet):
673 """
674 Get TestSet table record by its id
675 """
676 self._oDb.execute('SELECT *\n'
677 'FROM TestSets\n'
678 'WHERE idTestSet=%s\n',
679 (idTestSet,))
680
681 aRows = self._oDb.fetchAll()
682 if len(aRows) not in (0, 1):
683 raise TMTooManyRows('Found more than one test sets with the same credentials. Database structure is corrupted.')
684 try:
685 return TestSetData().initFromDbRow(aRows[0])
686 except IndexError:
687 return None
688
689
690 def fetchOrphaned(self):
691 """
692 Returns a list of TestSetData objects of orphaned test sets.
693
694 A test set is orphaned if tsDone is NULL and the testbox has created
695 one or more newer testsets.
696 """
697
698 self._oDb.execute('SELECT TestSets.*\n'
699 'FROM TestSets,\n'
700 ' (SELECT idTestSet, idTestBox FROM TestSets WHERE tsDone is NULL) AS t\n'
701 'WHERE TestSets.idTestSet = t.idTestSet\n'
702 ' AND EXISTS(SELECT 1 FROM TestSets st\n'
703 ' WHERE st.idTestBox = t.idTestBox AND st.idTestSet > t.idTestSet)\n'
704 ' AND NOT EXISTS(SELECT 1 FROM TestBoxStatuses tbs\n'
705 ' WHERE tbs.idTestBox = t.idTestBox AND tbs.idTestSet = t.idTestSet)\n'
706 'ORDER by TestSets.idTestBox, TestSets.idTestSet'
707 );
708 aoRet = [];
709 for aoRow in self._oDb.fetchAll():
710 aoRet.append(TestSetData().initFromDbRow(aoRow));
711 return aoRet;
712
713 def isTestBoxExecutingToRapidly(self, idTestBox):
714 """
715 Checks whether the specified test box is executing tests too rapidly.
716
717 The parameters defining too rapid execution are defined in config.py.
718
719 Returns True if it does, False if it doesn't.
720 May raise database problems.
721 """
722
723 self._oDb.execute('(\n'
724 'SELECT tsCreated\n'
725 'FROM TestSets\n'
726 'WHERE idTestBox = %s\n'
727 ' AND tsCreated >= (CURRENT_TIMESTAMP - interval \'%s seconds\')\n'
728 ') UNION (\n'
729 'SELECT tsCreated\n'
730 'FROM TestSets\n'
731 'WHERE idTestBox = %s\n'
732 ' AND tsCreated >= (CURRENT_TIMESTAMP - interval \'%s seconds\')\n'
733 ' AND enmStatus >= \'failure\'\n'
734 ')'
735 , ( idTestBox, config.g_kcSecMinSinceLastTask,
736 idTestBox, config.g_kcSecMinSinceLastFailedTask, ));
737 return self._oDb.getRowCount() > 0;
738
739
740 #
741 # The virtual test sheriff interface.
742 #
743
744 def fetchBadTestBoxIds(self, cHoursBack = 2, tsNow = None, aidFailureReasons = None):
745 """
746 Fetches a list of test box IDs which returned bad-testbox statuses in the
747 given period (tsDone).
748 """
749 if tsNow is None:
750 tsNow = self._oDb.getCurrentTimestamp();
751 if aidFailureReasons is None:
752 aidFailureReasons = [ -1, ];
753 self._oDb.execute('(SELECT idTestBox\n'
754 ' FROM TestSets\n'
755 ' WHERE TestSets.enmStatus = \'bad-testbox\'\n'
756 ' AND tsDone <= %s\n'
757 ' AND tsDone > (%s - interval \'%s hours\')\n'
758 ') UNION (\n'
759 ' SELECT TestSets.idTestBox\n'
760 ' FROM TestSets,\n'
761 ' TestResultFailures\n'
762 ' WHERE TestSets.tsDone <= %s\n'
763 ' AND TestSets.tsDone > (%s - interval \'%s hours\')\n'
764 ' AND TestSets.enmStatus >= \'failure\'::TestStatus_T\n'
765 ' AND TestSets.idTestSet = TestResultFailures.idTestSet\n'
766 ' AND TestResultFailures.tsExpire = \'infinity\'::TIMESTAMP\n'
767 ' AND TestResultFailures.idFailureReason IN ('
768 + ', '.join([str(i) for i in aidFailureReasons]) + ')\n'
769 ')\n'
770 , ( tsNow, tsNow, cHoursBack,
771 tsNow, tsNow, cHoursBack, ));
772 return [aoRow[0] for aoRow in self._oDb.fetchAll()];
773
774 def fetchSetsForTestBox(self, idTestBox, cHoursBack = 2, tsNow = None):
775 """
776 Fetches the TestSet rows for idTestBox for the given period (tsDone), w/o running ones.
777
778 Returns list of TestSetData sorted by tsDone in descending order.
779 """
780 if tsNow is None:
781 tsNow = self._oDb.getCurrentTimestamp();
782 self._oDb.execute('SELECT *\n'
783 'FROM TestSets\n'
784 'WHERE TestSets.idTestBox = %s\n'
785 ' AND tsDone IS NOT NULL\n'
786 ' AND tsDone <= %s\n'
787 ' AND tsDone > (%s - interval \'%s hours\')\n'
788 'ORDER by tsDone DESC\n'
789 , ( idTestBox, tsNow, tsNow, cHoursBack,));
790 return self._dbRowsToModelDataList(TestSetData);
791
792 def fetchFailedSetsWithoutReason(self, cHoursBack = 2, tsNow = None):
793 """
794 Fetches the TestSet failure rows without any currently (CURRENT_TIMESTAMP
795 not tsNow) assigned failure reason.
796
797 Returns list of TestSetData sorted by tsDone in descending order.
798
799 Note! Includes bad-testbox sets too as it can be useful to analyze these
800 too even if we normally count them in the 'skipped' category.
801 """
802 if tsNow is None:
803 tsNow = self._oDb.getCurrentTimestamp();
804 self._oDb.execute('SELECT TestSets.*\n'
805 'FROM TestSets\n'
806 ' LEFT OUTER JOIN TestResultFailures\n'
807 ' ON TestResultFailures.idTestSet = TestSets.idTestSet\n'
808 ' AND TestResultFailures.tsExpire = \'infinity\'::TIMESTAMP\n'
809 'WHERE TestSets.tsDone IS NOT NULL\n'
810 ' AND TestSets.enmStatus IN ( %s, %s, %s, %s )\n'
811 ' AND TestSets.tsDone <= %s\n'
812 ' AND TestSets.tsDone > (%s - interval \'%s hours\')\n'
813 ' AND TestResultFailures.idTestSet IS NULL\n'
814 'ORDER by tsDone DESC\n'
815 , ( TestSetData.ksTestStatus_Failure, TestSetData.ksTestStatus_TimedOut,
816 TestSetData.ksTestStatus_Rebooted, TestSetData.ksTestStatus_BadTestBox,
817 tsNow,
818 tsNow, cHoursBack,));
819 return self._dbRowsToModelDataList(TestSetData);
820
821
822
823#
824# Unit testing.
825#
826
827# pylint: disable=C0111
828class TestSetDataTestCase(ModelDataBaseTestCase):
829 def setUp(self):
830 self.aoSamples = [TestSetData(),];
831
832if __name__ == '__main__':
833 unittest.main();
834 # not reached.
835
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