VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testmanager/core/schedgroup.py@ 65080

Last change on this file since 65080 was 65040, checked in by vboxsync, 8 years ago

testmanager: More details in the system wide changelog.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 42.5 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: schedgroup.py 65040 2016-12-31 02:29:50Z vboxsync $
3
4"""
5Test Manager - Scheduling Group.
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: 65040 $"
30
31
32# Standard python imports.
33import unittest;
34
35# Validation Kit imports.
36from testmanager.core.base import ModelDataBase, ModelDataBaseTestCase, ModelLogicBase, TMExceptionBase, \
37 TMRowInUse, TMInvalidData, TMRowAlreadyExists, TMRowNotFound;
38from testmanager.core.buildsource import BuildSourceData;
39from testmanager.core.testcase import TestCaseData;
40from testmanager.core.testcaseargs import TestCaseArgsData;
41from testmanager.core.testbox import TestBoxData, TestBoxLogic;
42from testmanager.core.testgroup import TestGroupData;
43
44
45
46class SchedGroupMemberData(ModelDataBase):
47 """
48 SchedGroupMember Data.
49 """
50
51 ksIdAttr = 'idSchedGroup';
52
53 ksParam_idSchedGroup = 'SchedGroupMember_idSchedGroup';
54 ksParam_idTestGroup = 'SchedGroupMember_idTestGroup';
55 ksParam_tsEffective = 'SchedGroupMember_tsEffective';
56 ksParam_tsExpire = 'SchedGroupMember_tsExpire';
57 ksParam_uidAuthor = 'SchedGroupMember_uidAuthor';
58 ksParam_iSchedPriority = 'SchedGroupMember_iSchedPriority';
59 ksParam_bmHourlySchedule = 'SchedGroupMember_bmHourlySchedule';
60 ksParam_idTestGroupPreReq = 'SchedGroupMember_idTestGroupPreReq';
61
62 kasAllowNullAttributes = [ 'idSchedGroup', 'idTestGroup', 'tsEffective', 'tsExpire',
63 'uidAuthor', 'bmHourlySchedule', 'idTestGroupPreReq' ];
64 kiMin_iSchedPriority = 0;
65 kiMax_iSchedPriority = 32;
66
67 kcDbColumns = 8
68
69 def __init__(self):
70 ModelDataBase.__init__(self);
71
72 #
73 # Initialize with defaults.
74 # See the database for explanations of each of these fields.
75 #
76 self.idSchedGroup = None;
77 self.idTestGroup = None;
78 self.tsEffective = None;
79 self.tsExpire = None;
80 self.uidAuthor = None;
81 self.iSchedPriority = 16;
82 self.bmHourlySchedule = None;
83 self.idTestGroupPreReq = None;
84
85 def initFromDbRow(self, aoRow):
86 """
87 Re-initializes the data with a row from a SELECT * FROM SchedGroupMembers.
88
89 Returns self. Raises exception if the row is None or otherwise invalid.
90 """
91
92 if aoRow is None:
93 raise TMRowNotFound('SchedGroupMember not found.');
94
95 self.idSchedGroup = aoRow[0];
96 self.idTestGroup = aoRow[1];
97 self.tsEffective = aoRow[2];
98 self.tsExpire = aoRow[3];
99 self.uidAuthor = aoRow[4];
100 self.iSchedPriority = aoRow[5];
101 self.bmHourlySchedule = aoRow[6]; ## @todo figure out how bitmaps are returned...
102 self.idTestGroupPreReq = aoRow[7];
103 return self;
104
105
106class SchedGroupMemberDataEx(SchedGroupMemberData):
107 """
108 Extended SchedGroupMember data class.
109 This adds the testgroups.
110 """
111
112 def __init__(self):
113 SchedGroupMemberData.__init__(self);
114 self.oTestGroup = None;
115
116 def initFromDbRow(self, aoRow):
117 """
118 Re-initializes the data with a row from a query like this:
119
120 SELECT SchedGroupMembers.*, TestGroups.*
121 FROM SchedGroupMembers
122 JOIN TestGroups
123 ON (SchedGroupMembers.idTestGroup = TestGroups.idTestGroup);
124
125 Returns self. Raises exception if the row is None or otherwise invalid.
126 """
127 SchedGroupMemberData.initFromDbRow(self, aoRow);
128 self.oTestGroup = TestGroupData().initFromDbRow(aoRow[SchedGroupMemberData.kcDbColumns:]);
129 return self;
130
131 def getDataAttributes(self):
132 asAttributes = SchedGroupMemberData.getDataAttributes(self);
133 asAttributes.remove('oTestGroup');
134 return asAttributes;
135
136 def _validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor = ModelDataBase.ksValidateFor_Other):
137 dErrors = SchedGroupMemberData._validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor);
138 if self.ksParam_idTestGroup not in dErrors:
139 self.oTestGroup = TestGroupData();
140 try:
141 self.oTestGroup.initFromDbWithId(oDb, self.idTestGroup);
142 except Exception as oXcpt:
143 self.oTestGroup = TestGroupData()
144 dErrors[self.ksParam_idTestGroup] = str(oXcpt);
145 return dErrors;
146
147
148
149class SchedGroupData(ModelDataBase):
150 """
151 SchedGroup Data.
152 """
153
154 ## @name TestBoxState_T
155 # @{
156 ksScheduler_BestEffortContinuousIntegration = 'bestEffortContinousItegration'; # sic*2
157 ksScheduler_Reserved = 'reserved';
158 ## @}
159
160
161 ksIdAttr = 'idSchedGroup';
162
163 ksParam_idSchedGroup = 'SchedGroup_idSchedGroup';
164 ksParam_tsEffective = 'SchedGroup_tsEffective';
165 ksParam_tsExpire = 'SchedGroup_tsExpire';
166 ksParam_uidAuthor = 'SchedGroup_uidAuthor';
167 ksParam_sName = 'SchedGroup_sName';
168 ksParam_sDescription = 'SchedGroup_sDescription';
169 ksParam_fEnabled = 'SchedGroup_fEnabled';
170 ksParam_enmScheduler = 'SchedGroup_enmScheduler';
171 ksParam_idBuildSrc = 'SchedGroup_idBuildSrc';
172 ksParam_idBuildSrcTestSuite = 'SchedGroup_idBuildSrcTestSuite';
173 ksParam_sComment = 'SchedGroup_sComment';
174
175 kasAllowNullAttributes = ['idSchedGroup', 'tsEffective', 'tsExpire', 'uidAuthor', 'sDescription',
176 'idBuildSrc', 'idBuildSrcTestSuite', 'sComment' ];
177 kasValidValues_enmScheduler = [ ksScheduler_BestEffortContinuousIntegration, ];
178
179 kcDbColumns = 11;
180
181 # Scheduler types
182 kasSchedulerDesc = \
183 [
184 ( ksScheduler_BestEffortContinuousIntegration, 'Best-Effort-Continuous-Integration (BECI) scheduler.', ''),
185 ]
186
187 def __init__(self):
188 ModelDataBase.__init__(self);
189
190 #
191 # Initialize with defaults.
192 # See the database for explanations of each of these fields.
193 #
194 self.idSchedGroup = None;
195 self.tsEffective = None;
196 self.tsExpire = None;
197 self.uidAuthor = None;
198 self.sName = None;
199 self.sDescription = None;
200 self.fEnabled = None;
201 self.enmScheduler = SchedGroupData.ksScheduler_BestEffortContinuousIntegration;
202 self.idBuildSrc = None;
203 self.idBuildSrcTestSuite = None;
204 self.sComment = None;
205
206 def initFromDbRow(self, aoRow):
207 """
208 Re-initializes the data with a row from a SELECT * FROM SchedGroups.
209
210 Returns self. Raises exception if the row is None or otherwise invalid.
211 """
212
213 if aoRow is None:
214 raise TMRowNotFound('SchedGroup not found.');
215
216 self.idSchedGroup = aoRow[0];
217 self.tsEffective = aoRow[1];
218 self.tsExpire = aoRow[2];
219 self.uidAuthor = aoRow[3];
220 self.sName = aoRow[4];
221 self.sDescription = aoRow[5];
222 self.fEnabled = aoRow[6];
223 self.enmScheduler = aoRow[7];
224 self.idBuildSrc = aoRow[8];
225 self.idBuildSrcTestSuite = aoRow[9];
226 self.sComment = aoRow[10];
227 return self;
228
229 def initFromDbWithId(self, oDb, idSchedGroup, tsNow = None, sPeriodBack = None):
230 """
231 Initialize the object from the database.
232 """
233 oDb.execute(self.formatSimpleNowAndPeriodQuery(oDb,
234 'SELECT *\n'
235 'FROM SchedGroups\n'
236 'WHERE idSchedGroup = %s\n'
237 , ( idSchedGroup,), tsNow, sPeriodBack));
238 aoRow = oDb.fetchOne()
239 if aoRow is None:
240 raise TMRowNotFound('idSchedGroup=%s not found (tsNow=%s, sPeriodBack=%s)' % (idSchedGroup, tsNow, sPeriodBack));
241 return self.initFromDbRow(aoRow);
242
243
244class SchedGroupDataEx(SchedGroupData):
245 """
246 Extended scheduling group data.
247
248 Note! Similar to TestGroupDataEx.
249 """
250
251 ksParam_aoMembers = 'SchedGroup_aoMembers';
252 kasAltArrayNull = [ 'aoMembers', ];
253
254 ## Helper parameter containing the comma separated list with the IDs of
255 # potential members found in the parameters.
256 ksParam_aidTestGroups = 'TestGroupDataEx_aidTestGroups';
257
258
259 def __init__(self):
260 SchedGroupData.__init__(self);
261 self.aoMembers = []; # type: SchedGroupMemberDataEx
262
263 # Two build sources for convenience sake.
264 self.oBuildSrc = None; # type: TestBoxData
265 self.oBuildSrcValidationKit = None; # type: TestBoxData
266 # List of test boxes that uses this group for convenience.
267 self.aoTestBoxes = None; # type: list[TestBoxData]
268
269 def _initExtraMembersFromDb(self, oDb, tsNow = None, sPeriodBack = None):
270 """
271 Worker shared by the initFromDb* methods.
272 Returns self. Raises exception if no row or database error.
273 """
274 #
275 # It all upfront so the object has some kind of consistency if anything
276 # below raises exceptions.
277 #
278 self.oBuildSrc = None;
279 self.oBuildSrcValidationKit = None;
280 self.aoTestBoxes = [];
281 self.aoMembers = [];
282
283 #
284 # Build source.
285 #
286 if self.idBuildSrc:
287 self.oBuildSrc = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrc, tsNow, sPeriodBack);
288
289 if self.idBuildSrcTestSuite:
290 self.oBuildSrcValidationKit = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrcTestSuite,
291 tsNow, sPeriodBack);
292
293 #
294 # Test Boxes.
295 #
296 oDb.execute('SELECT TestBoxesWithStrings.*\n'
297 'FROM TestBoxesWithStrings,\n'
298 ' TestBoxesInSchedGroups\n'
299 'WHERE TestBoxesInSchedGroups.idSchedGroup = %s\n'
300 + self.formatSimpleNowAndPeriod(oDb, tsNow, sPeriodBack, sTablePrefix = 'TestBoxesInSchedGroups.') +
301 ' AND TestBoxesWithStrings.idTestBox = TestBoxesInSchedGroups.idTestBox\n'
302 + self.formatSimpleNowAndPeriod(oDb, tsNow, sPeriodBack, sTablePrefix = 'TestBoxesWithStrings.') +
303 'ORDER BY TestBoxesWithStrings.sName, TestBoxesWithStrings.idTestBox\n'
304 , (self.idSchedGroup,));
305 for aoRow in oDb.fetchAll():
306 self.aoTestBoxes.append(TestBoxData().initFromDbRow(aoRow));
307
308 #
309 # Test groups.
310 #
311 oDb.execute('SELECT SchedGroupMembers.*, TestGroups.*\n'
312 'FROM SchedGroupMembers\n'
313 'LEFT OUTER JOIN TestGroups ON (SchedGroupMembers.idTestGroup = TestGroups.idTestGroup)\n'
314 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
315 + self.formatSimpleNowAndPeriod(oDb, tsNow, sPeriodBack, sTablePrefix = 'SchedGroupMembers.')
316 + self.formatSimpleNowAndPeriod(oDb, tsNow, sPeriodBack, sTablePrefix = 'TestGroups.') +
317 'ORDER BY SchedGroupMembers.idTestGroupPreReq, SchedGroupMembers.idTestGroup\n'
318 , (self.idSchedGroup,));
319 for aoRow in oDb.fetchAll():
320 self.aoMembers.append(SchedGroupMemberDataEx().initFromDbRow(aoRow));
321 return self;
322
323 def initFromDbRowEx(self, aoRow, oDb, tsNow = None):
324 """
325 Reinitialize from a SELECT * FROM SchedGroups row. Will query the
326 necessary additional data from oDb using tsNow.
327 Returns self. Raises exception if no row or database error.
328 """
329 SchedGroupData.initFromDbRow(self, aoRow);
330 return self._initExtraMembersFromDb(oDb, tsNow);
331
332 def initFromDbWithId(self, oDb, idSchedGroup, tsNow = None, sPeriodBack = None):
333 """
334 Initialize the object from the database.
335 """
336 SchedGroupData.initFromDbWithId(self, oDb, idSchedGroup, tsNow, sPeriodBack);
337 return self._initExtraMembersFromDb(oDb, tsNow, sPeriodBack);
338
339 def getDataAttributes(self):
340 asAttributes = SchedGroupData.getDataAttributes(self);
341 asAttributes.remove('oBuildSrc');
342 asAttributes.remove('oBuildSrcValidationKit');
343 asAttributes.remove('aoTestBoxes');
344 return asAttributes;
345
346 def getAttributeParamNullValues(self, sAttr):
347 if sAttr != 'aoMembers':
348 return SchedGroupData.getAttributeParamNullValues(self, sAttr);
349 return ['', [], None];
350
351 def convertParamToAttribute(self, sAttr, sParam, oValue, oDisp, fStrict):
352 if sAttr != 'aoMembers':
353 return SchedGroupData.convertParamToAttribute(self, sAttr, sParam, oValue, oDisp, fStrict);
354
355 aoNewValue = [];
356 aidSelected = oDisp.getListOfIntParams(sParam, iMin = 1, iMax = 0x7ffffffe, aiDefaults = [])
357 sIds = oDisp.getStringParam(self.ksParam_aidTestGroups, sDefault = '');
358 for idTestGroup in sIds.split(','):
359 try: idTestGroup = int(idTestGroup);
360 except: pass;
361 oDispWrapper = self.DispWrapper(oDisp, '%s[%s][%%s]' % (SchedGroupDataEx.ksParam_aoMembers, idTestGroup,))
362 oMember = SchedGroupMemberDataEx().initFromParams(oDispWrapper, fStrict = False);
363 if idTestGroup in aidSelected:
364 aoNewValue.append(oMember);
365 return aoNewValue;
366
367 def _validateAndConvertAttribute(self, sAttr, sParam, oValue, aoNilValues, fAllowNull, oDb):
368 if sAttr != 'aoMembers':
369 return SchedGroupData._validateAndConvertAttribute(self, sAttr, sParam, oValue, aoNilValues, fAllowNull, oDb);
370
371 asErrors = [];
372 aoNewMembers = [];
373 for oOldMember in oValue:
374 oNewMember = SchedGroupMemberDataEx().initFromOther(oOldMember);
375 aoNewMembers.append(oNewMember);
376
377 dErrors = oNewMember.validateAndConvert(oDb, ModelDataBase.ksValidateFor_Other);
378 if len(dErrors) > 0:
379 asErrors.append(str(dErrors));
380
381 if len(asErrors) == 0:
382 for i, _ in enumerate(aoNewMembers):
383 idTestGroup = aoNewMembers[i];
384 for j in range(i + 1, len(aoNewMembers)):
385 if aoNewMembers[j].idTestGroup == idTestGroup:
386 asErrors.append('Duplicate test group #%d!' % (idTestGroup, ));
387 break;
388
389 return (aoNewMembers, None if len(asErrors) == 0 else '<br>\n'.join(asErrors));
390
391 def _validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor = ModelDataBase.ksValidateFor_Other):
392 dErrors = SchedGroupData._validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor);
393
394 #
395 # Fetch the extended build source bits.
396 #
397 if self.ksParam_idBuildSrc not in dErrors:
398 if self.idBuildSrc in self.getAttributeParamNullValues('idBuildSrc') \
399 or self.idBuildSrc is None:
400 self.oBuildSrc = None;
401 else:
402 try:
403 self.oBuildSrc = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrc);
404 except Exception as oXcpt:
405 self.oBuildSrc = BuildSourceData();
406 dErrors[self.ksParam_idBuildSrc] = str(oXcpt);
407
408 if self.ksParam_idBuildSrcTestSuite not in dErrors:
409 if self.idBuildSrcTestSuite in self.getAttributeParamNullValues('idBuildSrcTestSuite') \
410 or self.idBuildSrcTestSuite is None:
411 self.oBuildSrcValidationKit = None;
412 else:
413 try:
414 self.oBuildSrcValidationKit = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrcTestSuite);
415 except Exception as oXcpt:
416 self.oBuildSrcValidationKit = BuildSourceData();
417 dErrors[self.ksParam_idBuildSrcTestSuite] = str(oXcpt);
418
419 return dErrors;
420
421
422
423class SchedGroupLogic(ModelLogicBase): # pylint: disable=R0903
424 """
425 SchedGroup logic.
426 """
427
428 def __init__(self, oDb):
429 ModelLogicBase.__init__(self, oDb);
430 self.dCache = None;
431
432 #
433 # Standard methods.
434 #
435
436 def fetchForListing(self, iStart, cMaxRows, tsNow):
437 """
438 Fetches build sources.
439
440 Returns an array (list) of BuildSourceData items, empty list if none.
441 Raises exception on error.
442 """
443
444 if tsNow is None:
445 self._oDb.execute('SELECT *\n'
446 'FROM SchedGroups\n'
447 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
448 'ORDER BY idSchedGroup DESC\n'
449 'LIMIT %s OFFSET %s\n'
450 , (cMaxRows, iStart,));
451 else:
452 self._oDb.execute('SELECT *\n'
453 'FROM SchedGroups\n'
454 'WHERE tsExpire > %s\n'
455 ' AND tsEffective <= %s\n'
456 'ORDER BY idSchedGroup DESC\n'
457 'LIMIT %s OFFSET %s\n'
458 , (tsNow, tsNow, cMaxRows, iStart,));
459
460 aoRet = [];
461 for aoRow in self._oDb.fetchAll():
462 aoRet.append(SchedGroupDataEx().initFromDbRowEx(aoRow, self._oDb, tsNow));
463 return aoRet;
464
465 def addEntry(self, oData, uidAuthor, fCommit = False):
466 """Add Scheduling Group record"""
467
468 #
469 # Validate.
470 #
471 dDataErrors = oData.validateAndConvert(self._oDb, oData.ksValidateFor_Add);
472 if len(dDataErrors) > 0:
473 raise TMInvalidData('Invalid data passed to addEntry: %s' % (dDataErrors,));
474 if self.exists(oData.sName):
475 raise TMRowAlreadyExists('Scheduling group "%s" already exists.' % (oData.sName,));
476
477 #
478 # Add it.
479 #
480 self._oDb.execute('INSERT INTO SchedGroups (\n'
481 ' uidAuthor,\n'
482 ' sName,\n'
483 ' sDescription,\n'
484 ' fEnabled,\n'
485 ' enmScheduler,\n'
486 ' idBuildSrc,\n'
487 ' idBuildSrcTestSuite,\n'
488 ' sComment)\n'
489 'VALUES (%s, %s, %s, %s, %s, %s, %s, %s)\n'
490 'RETURNING idSchedGroup\n'
491 , ( uidAuthor,
492 oData.sName,
493 oData.sDescription,
494 oData.fEnabled,
495 oData.enmScheduler,
496 oData.idBuildSrc,
497 oData.idBuildSrcTestSuite,
498 oData.sComment ));
499 idSchedGroup = self._oDb.fetchOne()[0];
500 oData.idSchedGroup = idSchedGroup;
501
502 for oMember in oData.aoMembers:
503 oMember.idSchedGroup = idSchedGroup;
504 self._addSchedGroupMember(uidAuthor, oMember);
505
506 self._oDb.maybeCommit(fCommit);
507 return True;
508
509 def editEntry(self, oData, uidAuthor, fCommit = False):
510 """Edit Scheduling Group record"""
511
512 #
513 # Validate input and retrieve the old data.
514 #
515 dErrors = oData.validateAndConvert(self._oDb, oData.ksValidateFor_Edit);
516 if len(dErrors) > 0:
517 raise TMInvalidData('editEntry got invalid data: %s' % (dErrors,));
518 self._assertUnique(oData.sName, oData.idSchedGroup);
519 oOldData = SchedGroupDataEx().initFromDbWithId(self._oDb, oData.idSchedGroup);
520
521 #
522 # Make the changes.
523 #
524 if not oData.isEqualEx(oOldData, [ 'tsEffective', 'tsExpire', 'uidAuthor', 'aoMembers', 'aoTestBoxes',
525 'oBuildSrc', 'oBuildSrcValidationKit', ]):
526 self._historizeEntry(oData.idSchedGroup);
527 self._readdEntry(uidAuthor, oData);
528
529 # Remove groups.
530 for oOld in oOldData.aoMembers:
531 fRemove = True;
532 for oNew in oData.aoMembers:
533 if oNew.idTestGroup == oOld.idTestGroup:
534 fRemove = False;
535 break;
536 if fRemove:
537 self._removeSchedGroupMember(uidAuthor, oOld);
538
539 # Add / modify groups.
540 for oMember in oData.aoMembers:
541 oOldMember = None;
542 for oOld in oOldData.aoMembers:
543 if oOld.idTestGroup == oMember.idTestGroup:
544 oOldMember = oOld;
545 break;
546
547 oMember.idSchedGroup = oData.idSchedGroup;
548 if oOldMember is None:
549 self._addSchedGroupMember(uidAuthor, oMember);
550 elif not oMember.isEqualEx(oOldMember, ['tsEffective', 'tsExpire', 'uidAuthor', 'oTestGroup']):
551 self._historizeSchedGroupMember(oMember);
552 self._addSchedGroupMember(uidAuthor, oMember);
553
554 self._oDb.maybeCommit(fCommit);
555 return True;
556
557 def removeEntry(self, uidAuthor, idSchedGroup, fCascade = False, fCommit = False):
558 """
559 Deletes a scheduling group.
560 """
561
562 #
563 # Input validation and retrival of current data.
564 #
565 if idSchedGroup == 1:
566 raise TMRowInUse('Cannot remove the default scheduling group (id 1).');
567 oData = SchedGroupDataEx().initFromDbWithId(self._oDb, idSchedGroup);
568
569 #
570 # We use cascade a little different here... We don't actually delete
571 # associated testboxes or testgroups.
572 #
573 if len(oData.aoTestBoxes) > 0:
574 if fCascade is not True:
575 # Complain about there being associated testboxes.
576 asTestBoxes = ['%s (#%d)' % (oTestBox.sName, oTestBox.idTestBox) for oTestBox in oData.aoTestBoxes];
577 raise TMRowInUse('Scheduling group #%d is associated with one or more test boxes: %s'
578 % (idSchedGroup, ', '.join(asTestBoxes),));
579 else:
580 # Reassign testboxes to scheduling group #1 (the default group).
581 oTbLogic = TestBoxLogic(self._oDb);
582 for oTestBox in oData.aoTestBoxes:
583 oTbCopy = TestBoxData().initFromOther(oTestBox);
584 oTbCopy.idSchedGroup = 1;
585 oTbLogic.editEntry(oTbCopy, uidAuthor, fCommit = False);
586
587 oData = SchedGroupDataEx().initFromDbWithId(self._oDb, idSchedGroup);
588 if len(oData.aoTestBoxes) != 0:
589 raise TMRowInUse('More testboxes was added to the scheduling group as we were trying to delete it.');
590
591 #
592 # Remove the group and all member records.
593 #
594 for oMember in oData.aoMembers:
595 self._removeSchedGroupMember(uidAuthor, oMember);
596 self._oDb.execute('UPDATE SchedGroupMembers\n'
597 'SET tsExpire = CURRENT_TIMESTAMP\n'
598 'WHERE idSchedGroup = %s\n'
599 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
600 , (idSchedGroup,));
601
602 (tsCur, tsCurMinusOne) = self._oDb.getCurrentTimestamps();
603 if oData.tsEffective != tsCur and oData.tsEffective != tsCurMinusOne:
604 self._historizeEntry(idSchedGroup, tsCurMinusOne);
605 self._readdEntry(uidAuthor, oData, tsCurMinusOne);
606 self._historizeEntry(idSchedGroup);
607 self._oDb.execute('UPDATE SchedGroups\n'
608 'SET tsExpire = CURRENT_TIMESTAMP\n'
609 'WHERE idSchedGroup = %s\n'
610 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
611 , (idSchedGroup,))
612
613 self._oDb.maybeCommit(fCommit)
614 return True;
615
616
617 def cachedLookup(self, idSchedGroup):
618 """
619 Looks up the most recent SchedGroupData object for idSchedGroup
620 via an object cache.
621
622 Returns a shared SchedGroupData object. None if not found.
623 Raises exception on DB error.
624 """
625 if self.dCache is None:
626 self.dCache = self._oDb.getCache('SchedGroup');
627
628 oEntry = self.dCache.get(idSchedGroup, None);
629 if oEntry is None:
630 self._oDb.execute('SELECT *\n'
631 'FROM SchedGroups\n'
632 'WHERE idSchedGroup = %s\n'
633 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
634 , (idSchedGroup, ));
635 if self._oDb.getRowCount() == 0:
636 # Maybe it was deleted, try get the last entry.
637 self._oDb.execute('SELECT *\n'
638 'FROM SchedGroups\n'
639 'WHERE idSchedGroup = %s\n'
640 'ORDER BY tsExpire DESC\n'
641 'LIMIT 1\n'
642 , (idSchedGroup, ));
643 elif self._oDb.getRowCount() > 1:
644 raise self._oDb.integrityException('%s infinity rows for %s' % (self._oDb.getRowCount(), idSchedGroup));
645
646 if self._oDb.getRowCount() == 1:
647 oEntry = SchedGroupData().initFromDbRow(self._oDb.fetchOne());
648 self.dCache[idSchedGroup] = oEntry;
649 return oEntry;
650
651
652 #
653 # Other methods.
654 #
655
656 def fetchOrderedByName(self, tsNow = None):
657 """
658 Return list of objects of type SchedGroups ordered by name.
659 May raise exception on database error.
660 """
661 if tsNow is None:
662 self._oDb.execute('SELECT *\n'
663 'FROM SchedGroups\n'
664 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
665 'ORDER BY sName ASC\n');
666 else:
667 self._oDb.execute('SELECT *\n'
668 'FROM SchedGroups\n'
669 'WHERE tsExpire > %s\n'
670 ' AND tsEffective <= %s\n'
671 'ORDER BY sName ASC\n'
672 , (tsNow, tsNow,));
673 aoRet = []
674 for _ in range(self._oDb.getRowCount()):
675 aoRet.append(SchedGroupData().initFromDbRow(self._oDb.fetchOne()));
676 return aoRet;
677
678
679 def getAll(self, tsEffective = None):
680 """
681 Gets the list of all scheduling groups.
682 Returns an array of SchedGroupData instances.
683 """
684 if tsEffective is None:
685 self._oDb.execute('SELECT *\n'
686 'FROM SchedGroups\n'
687 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n');
688 else:
689 self._oDb.execute('SELECT *\n'
690 'FROM SchedGroups\n'
691 'WHERE tsExpire > %s\n'
692 ' AND tsEffective <= %s\n'
693 , (tsEffective, tsEffective));
694 aoRet = [];
695 for aoRow in self._oDb.fetchAll():
696 aoRet.append(SchedGroupData().initFromDbRow(aoRow));
697 return aoRet;
698
699 def getSchedGroupsForCombo(self, tsEffective = None):
700 """
701 Gets the list of active scheduling groups for a combo box.
702 Returns an array of (value [idSchedGroup], drop-down-name [sName],
703 hover-text [sDescription]) tuples.
704 """
705 if tsEffective is None:
706 self._oDb.execute('SELECT idSchedGroup, sName, sDescription\n'
707 'FROM SchedGroups\n'
708 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
709 'ORDER BY sName');
710 else:
711 self._oDb.execute('SELECT idSchedGroup, sName, sDescription\n'
712 'FROM SchedGroups\n'
713 'WHERE tsExpire > %s\n'
714 ' AND tsEffective <= %s\n'
715 'ORDER BY sName'
716 , (tsEffective, tsEffective));
717 return self._oDb.fetchAll();
718
719
720 def getMembers(self, idSchedGroup, tsEffective = None):
721 """
722 Gets the scheduling groups members for the given scheduling group.
723
724 Returns an array of SchedGroupMemberDataEx instances (sorted by
725 priority and idTestGroup). May raise exception DB error.
726 """
727
728 if tsEffective is None:
729 self._oDb.execute('SELECT *\n'
730 'FROM SchedGroupMembers, TestGroups\n'
731 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
732 ' AND SchedGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
733 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
734 ' AND TestGroups.tsExpire = \'infinity\'::TIMESTAMP\n'
735 'ORDER BY SchedGroupMembers.iSchedPriority, SchedGroupMembers.idTestGroup\n'
736 , (idSchedGroup,));
737 else:
738 self._oDb.execute('SELECT *\n'
739 'FROM SchedGroupMembers, TestGroups\n'
740 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
741 ' AND SchedGroupMembers.tsExpire < %s\n'
742 ' AND SchedGroupMembers.tsEffective >= %s\n'
743 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
744 ' AND TestGroups.tsExpire < %s\n'
745 ' AND TestGroups.tsEffective >= %s\n'
746 'ORDER BY SchedGroupMembers.iSchedPriority DESC, SchedGroupMembers.idTestGroup\n'
747 , (idSchedGroup, tsEffective, tsEffective, tsEffective, tsEffective, ));
748 aaoRows = self._oDb.fetchAll();
749 aoRet = [];
750 for aoRow in aaoRows:
751 aoRet.append(SchedGroupMemberDataEx().initFromDbRow(aoRow));
752 return aoRet;
753
754 def getTestCasesForGroup(self, idSchedGroup, cMax = None):
755 """
756 Gets the enabled testcases w/ testgroup+priority for the given scheduling group.
757
758 Returns an array TestCaseData instance (group id, testcase priority and
759 testcase ids) with an extra iSchedPriority member.
760 May raise exception on DB error or if the result exceeds cMax.
761 """
762
763 self._oDb.execute('SELECT TestGroupMembers.idTestGroup, TestGroupMembers.iSchedPriority, TestCases.*\n'
764 'FROM SchedGroupMembers, TestGroups, TestGroupMembers, TestCases\n'
765 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
766 ' AND SchedGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
767 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
768 ' AND TestGroups.tsExpire = \'infinity\'::TIMESTAMP\n'
769 ' AND TestGroupMembers.idTestGroup = TestGroups.idTestGroup\n'
770 ' AND TestGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
771 ' AND TestCases.idTestCase = TestGroupMembers.idTestCase\n'
772 ' AND TestCases.tsExpire = \'infinity\'::TIMESTAMP\n'
773 ' AND TestCases.fEnabled = TRUE\n'
774 'ORDER BY TestGroupMembers.idTestGroup, TestGroupMembers.iSchedPriority, TestCases.idTestCase\n'
775 , (idSchedGroup,));
776
777 if cMax is not None and self._oDb.getRowCount() > cMax:
778 raise TMExceptionBase('Too many testcases for scheduling group %s: %s, max %s'
779 % (idSchedGroup, cMax, self._oDb.getRowCount(),));
780
781 aoRet = [];
782 for aoRow in self._oDb.fetchAll():
783 oTestCase = TestCaseData().initFromDbRow(aoRow[2:]);
784 oTestCase.idTestGroup = aoRow[0];
785 oTestCase.iSchedPriority = aoRow[1];
786 aoRet.append(oTestCase);
787 return aoRet;
788
789 def getTestCaseArgsForGroup(self, idSchedGroup, cMax = None):
790 """
791 Gets the testcase argument variation w/ testgroup+priority for the given scheduling group.
792
793 Returns an array TestCaseArgsData instance (sorted by group and
794 variation id) with an extra iSchedPriority member.
795 May raise exception on DB error or if the result exceeds cMax.
796 """
797
798 self._oDb.execute('SELECT TestGroupMembers.idTestGroup, TestGroupMembers.iSchedPriority, TestCaseArgs.*\n'
799 'FROM SchedGroupMembers, TestGroups, TestGroupMembers, TestCaseArgs, TestCases\n'
800 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
801 ' AND SchedGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
802 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
803 ' AND TestGroups.tsExpire = \'infinity\'::TIMESTAMP\n'
804 ' AND TestGroupMembers.idTestGroup = TestGroups.idTestGroup\n'
805 ' AND TestGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
806 ' AND TestCaseArgs.idTestCase = TestGroupMembers.idTestCase\n'
807 ' AND TestCaseArgs.tsExpire = \'infinity\'::TIMESTAMP\n'
808 ' AND ( TestGroupMembers.aidTestCaseArgs is NULL\n'
809 ' OR TestCaseArgs.idTestCaseArgs = ANY(TestGroupMembers.aidTestCaseArgs) )\n'
810 ' AND TestCases.idTestCase = TestCaseArgs.idTestCase\n'
811 ' AND TestCases.tsExpire = \'infinity\'::TIMESTAMP\n'
812 ' AND TestCases.fEnabled = TRUE\n'
813 'ORDER BY TestGroupMembers.idTestGroup, TestGroupMembers.idTestCase, TestCaseArgs.idTestCaseArgs\n'
814 , (idSchedGroup,));
815
816 if cMax is not None and self._oDb.getRowCount() > cMax:
817 raise TMExceptionBase('Too many argument variations for scheduling group %s: %s, max %s'
818 % (idSchedGroup, cMax, self._oDb.getRowCount(),));
819
820 aoRet = [];
821 for aoRow in self._oDb.fetchAll():
822 oVariation = TestCaseArgsData().initFromDbRow(aoRow[2:]);
823 oVariation.idTestGroup = aoRow[0];
824 oVariation.iSchedPriority = aoRow[1];
825 aoRet.append(oVariation);
826 return aoRet;
827
828 def exists(self, sName):
829 """Checks if a group with the given name exists."""
830 self._oDb.execute('SELECT idSchedGroup\n'
831 'FROM SchedGroups\n'
832 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
833 ' AND sName = %s\n'
834 'LIMIT 1\n'
835 , (sName,));
836 return self._oDb.getRowCount() > 0;
837
838 def getById(self, idSchedGroup):
839 """Get Scheduling Group data by idSchedGroup"""
840 self._oDb.execute('SELECT *\n'
841 'FROM SchedGroups\n'
842 'WHERE tsExpire = \'infinity\'::timestamp\n'
843 ' AND idSchedGroup = %s;', (idSchedGroup,))
844 aRows = self._oDb.fetchAll()
845 if len(aRows) not in (0, 1):
846 raise self._oDb.integrityException(
847 'Found more than one scheduling groups with the same credentials. Database structure is corrupted.')
848 try:
849 return SchedGroupData().initFromDbRow(aRows[0])
850 except IndexError:
851 return None
852
853
854 #
855 # Internal helpers.
856 #
857
858 def _assertUnique(self, sName, idSchedGroupIgnore = None):
859 """
860 Checks that the scheduling group name is unique.
861 Raises exception if the name is already in use.
862 """
863 if idSchedGroupIgnore is None:
864 self._oDb.execute('SELECT idSchedGroup\n'
865 'FROM SchedGroups\n'
866 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
867 ' AND sName = %s\n'
868 , ( sName, ) );
869 else:
870 self._oDb.execute('SELECT idSchedGroup\n'
871 'FROM SchedGroups\n'
872 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
873 ' AND sName = %s\n'
874 ' AND idSchedGroup <> %s\n'
875 , ( sName, idSchedGroupIgnore, ) );
876 if self._oDb.getRowCount() > 0:
877 raise TMRowInUse('Scheduling group name (%s) is already in use.' % (sName,));
878 return True;
879
880 def _readdEntry(self, uidAuthor, oData, tsEffective = None):
881 """
882 Re-adds the SchedGroups entry. Used by editEntry and removeEntry.
883 """
884 if tsEffective is None:
885 tsEffective = self._oDb.getCurrentTimestamp();
886 self._oDb.execute('INSERT INTO SchedGroups (\n'
887 ' uidAuthor,\n'
888 ' tsEffective,\n'
889 ' idSchedGroup,\n'
890 ' sName,\n'
891 ' sDescription,\n'
892 ' fEnabled,\n'
893 ' enmScheduler,\n'
894 ' idBuildSrc,\n'
895 ' idBuildSrcTestSuite,\n'
896 ' sComment )\n'
897 'VALUES ( %s, %s, %s, %s, %s, %s, %s, %s, %s, %s )\n'
898 , ( uidAuthor,
899 tsEffective,
900 oData.idSchedGroup,
901 oData.sName,
902 oData.sDescription,
903 oData.fEnabled,
904 oData.enmScheduler,
905 oData.idBuildSrc,
906 oData.idBuildSrcTestSuite,
907 oData.sComment, ));
908 return True;
909
910 def _historizeEntry(self, idSchedGroup, tsExpire = None):
911 """
912 Historizes the current entry for the given scheduling group.
913 """
914 if tsExpire is None:
915 tsExpire = self._oDb.getCurrentTimestamp();
916 self._oDb.execute('UPDATE SchedGroups\n'
917 'SET tsExpire = %s\n'
918 'WHERE idSchedGroup = %s\n'
919 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
920 , ( tsExpire, idSchedGroup, ));
921 return True;
922
923 def _addSchedGroupMember(self, uidAuthor, oMember, tsEffective = None):
924 """
925 addEntry worker for adding a scheduling group member.
926 """
927 if tsEffective is None:
928 tsEffective = self._oDb.getCurrentTimestamp();
929 self._oDb.execute('INSERT INTO SchedGroupMembers(\n'
930 ' idSchedGroup,\n'
931 ' idTestGroup,\n'
932 ' tsEffective,\n'
933 ' uidAuthor,\n'
934 ' iSchedPriority,\n'
935 ' bmHourlySchedule,\n'
936 ' idTestGroupPreReq)\n'
937 'VALUES (%s, %s, %s, %s, %s, %s, %s)\n'
938 , ( oMember.idSchedGroup,
939 oMember.idTestGroup,
940 tsEffective,
941 uidAuthor,
942 oMember.iSchedPriority,
943 oMember.bmHourlySchedule,
944 oMember.idTestGroupPreReq, ));
945 return True;
946
947 def _removeSchedGroupMember(self, uidAuthor, oMember):
948 """
949 Removes a scheduling group member.
950 """
951
952 # Try record who removed it by adding an dummy entry that expires immediately.
953 (tsCur, tsCurMinusOne) = self._oDb.getCurrentTimestamps();
954 if oMember.tsEffective != tsCur and oMember.tsEffective != tsCurMinusOne:
955 self._historizeSchedGroupMember(oMember, tsCurMinusOne);
956 self._addSchedGroupMember(uidAuthor, oMember, tsCurMinusOne); # lazy bird.
957 self._historizeSchedGroupMember(oMember);
958 else:
959 self._historizeSchedGroupMember(oMember);
960 return True;
961
962 def _historizeSchedGroupMember(self, oMember, tsExpire = None):
963 """
964 Historizes the current entry for the given scheduling group.
965 """
966 if tsExpire is None:
967 tsExpire = self._oDb.getCurrentTimestamp();
968 self._oDb.execute('UPDATE SchedGroupMembers\n'
969 'SET tsExpire = %s\n'
970 'WHERE idSchedGroup = %s\n'
971 ' AND idTestGroup = %s\n'
972 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
973 , ( tsExpire, oMember.idSchedGroup, oMember.idTestGroup, ));
974 return True;
975
976
977
978
979#
980# Unit testing.
981#
982
983# pylint: disable=C0111
984class SchedGroupMemberDataTestCase(ModelDataBaseTestCase):
985 def setUp(self):
986 self.aoSamples = [SchedGroupMemberData(),];
987
988class SchedGroupDataTestCase(ModelDataBaseTestCase):
989 def setUp(self):
990 self.aoSamples = [SchedGroupData(),];
991
992if __name__ == '__main__':
993 unittest.main();
994 # not reached.
995
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