VirtualBox

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

Last change on this file since 83387 was 83387, checked in by vboxsync, 5 years ago

TestManager/SchedQueue: Fixed various validation issues with the testbox-in-sched-group stuff. Added change log to the scheduling queue details page. [pylint]

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 60.2 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: schedgroup.py 83387 2020-03-24 14:50:45Z vboxsync $
3
4"""
5Test Manager - Scheduling Group.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2020 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: 83387 $"
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, \
38 ChangeLogEntry, AttributeChangeEntry, AttributeChangeEntryPre;
39from testmanager.core.buildsource import BuildSourceData;
40from testmanager.core import db;
41from testmanager.core.testcase import TestCaseData;
42from testmanager.core.testcaseargs import TestCaseArgsData;
43from testmanager.core.testbox import TestBoxLogic, TestBoxDataForSchedGroup;
44from testmanager.core.testgroup import TestGroupData;
45from testmanager.core.useraccount import UserAccountLogic;
46
47
48
49class SchedGroupMemberData(ModelDataBase):
50 """
51 SchedGroupMember Data.
52 """
53
54 ksIdAttr = 'idSchedGroup';
55
56 ksParam_idSchedGroup = 'SchedGroupMember_idSchedGroup';
57 ksParam_idTestGroup = 'SchedGroupMember_idTestGroup';
58 ksParam_tsEffective = 'SchedGroupMember_tsEffective';
59 ksParam_tsExpire = 'SchedGroupMember_tsExpire';
60 ksParam_uidAuthor = 'SchedGroupMember_uidAuthor';
61 ksParam_iSchedPriority = 'SchedGroupMember_iSchedPriority';
62 ksParam_bmHourlySchedule = 'SchedGroupMember_bmHourlySchedule';
63 ksParam_idTestGroupPreReq = 'SchedGroupMember_idTestGroupPreReq';
64
65 kasAllowNullAttributes = [ 'idSchedGroup', 'idTestGroup', 'tsEffective', 'tsExpire',
66 'uidAuthor', 'bmHourlySchedule', 'idTestGroupPreReq' ];
67 kiMin_iSchedPriority = 0;
68 kiMax_iSchedPriority = 32;
69
70 kcDbColumns = 8
71
72 def __init__(self):
73 ModelDataBase.__init__(self);
74
75 #
76 # Initialize with defaults.
77 # See the database for explanations of each of these fields.
78 #
79 self.idSchedGroup = None;
80 self.idTestGroup = None;
81 self.tsEffective = None;
82 self.tsExpire = None;
83 self.uidAuthor = None;
84 self.iSchedPriority = 16;
85 self.bmHourlySchedule = None;
86 self.idTestGroupPreReq = None;
87
88 def initFromDbRow(self, aoRow):
89 """
90 Re-initializes the data with a row from a SELECT * FROM SchedGroupMembers.
91
92 Returns self. Raises exception if the row is None or otherwise invalid.
93 """
94
95 if aoRow is None:
96 raise TMRowNotFound('SchedGroupMember not found.');
97
98 self.idSchedGroup = aoRow[0];
99 self.idTestGroup = aoRow[1];
100 self.tsEffective = aoRow[2];
101 self.tsExpire = aoRow[3];
102 self.uidAuthor = aoRow[4];
103 self.iSchedPriority = aoRow[5];
104 self.bmHourlySchedule = aoRow[6]; ## @todo figure out how bitmaps are returned...
105 self.idTestGroupPreReq = aoRow[7];
106 return self;
107
108
109class SchedGroupMemberDataEx(SchedGroupMemberData):
110 """
111 Extended SchedGroupMember data class.
112 This adds the testgroups.
113 """
114
115 def __init__(self):
116 SchedGroupMemberData.__init__(self);
117 self.oTestGroup = None;
118
119 def initFromDbRow(self, aoRow):
120 """
121 Re-initializes the data with a row from a query like this:
122
123 SELECT SchedGroupMembers.*, TestGroups.*
124 FROM SchedGroupMembers
125 JOIN TestGroups
126 ON (SchedGroupMembers.idTestGroup = TestGroups.idTestGroup);
127
128 Returns self. Raises exception if the row is None or otherwise invalid.
129 """
130 SchedGroupMemberData.initFromDbRow(self, aoRow);
131 self.oTestGroup = TestGroupData().initFromDbRow(aoRow[SchedGroupMemberData.kcDbColumns:]);
132 return self;
133
134 def getDataAttributes(self):
135 asAttributes = SchedGroupMemberData.getDataAttributes(self);
136 asAttributes.remove('oTestGroup');
137 return asAttributes;
138
139 def _validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor = ModelDataBase.ksValidateFor_Other):
140 dErrors = SchedGroupMemberData._validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor);
141 if self.ksParam_idTestGroup not in dErrors:
142 self.oTestGroup = TestGroupData();
143 try:
144 self.oTestGroup.initFromDbWithId(oDb, self.idTestGroup);
145 except Exception as oXcpt:
146 self.oTestGroup = TestGroupData()
147 dErrors[self.ksParam_idTestGroup] = str(oXcpt);
148 return dErrors;
149
150
151
152
153class SchedGroupData(ModelDataBase):
154 """
155 SchedGroup Data.
156 """
157
158 ## @name TestBoxState_T
159 # @{
160 ksScheduler_BestEffortContinuousIntegration = 'bestEffortContinousItegration'; # sic*2
161 ksScheduler_Reserved = 'reserved';
162 ## @}
163
164
165 ksIdAttr = 'idSchedGroup';
166
167 ksParam_idSchedGroup = 'SchedGroup_idSchedGroup';
168 ksParam_tsEffective = 'SchedGroup_tsEffective';
169 ksParam_tsExpire = 'SchedGroup_tsExpire';
170 ksParam_uidAuthor = 'SchedGroup_uidAuthor';
171 ksParam_sName = 'SchedGroup_sName';
172 ksParam_sDescription = 'SchedGroup_sDescription';
173 ksParam_fEnabled = 'SchedGroup_fEnabled';
174 ksParam_enmScheduler = 'SchedGroup_enmScheduler';
175 ksParam_idBuildSrc = 'SchedGroup_idBuildSrc';
176 ksParam_idBuildSrcTestSuite = 'SchedGroup_idBuildSrcTestSuite';
177 ksParam_sComment = 'SchedGroup_sComment';
178
179 kasAllowNullAttributes = ['idSchedGroup', 'tsEffective', 'tsExpire', 'uidAuthor', 'sDescription',
180 'idBuildSrc', 'idBuildSrcTestSuite', 'sComment' ];
181 kasValidValues_enmScheduler = [ ksScheduler_BestEffortContinuousIntegration, ];
182
183 kcDbColumns = 11;
184
185 # Scheduler types
186 kasSchedulerDesc = \
187 [
188 ( ksScheduler_BestEffortContinuousIntegration, 'Best-Effort-Continuous-Integration (BECI) scheduler.', ''),
189 ]
190
191 def __init__(self):
192 ModelDataBase.__init__(self);
193
194 #
195 # Initialize with defaults.
196 # See the database for explanations of each of these fields.
197 #
198 self.idSchedGroup = None;
199 self.tsEffective = None;
200 self.tsExpire = None;
201 self.uidAuthor = None;
202 self.sName = None;
203 self.sDescription = None;
204 self.fEnabled = None;
205 self.enmScheduler = SchedGroupData.ksScheduler_BestEffortContinuousIntegration;
206 self.idBuildSrc = None;
207 self.idBuildSrcTestSuite = None;
208 self.sComment = None;
209
210 def initFromDbRow(self, aoRow):
211 """
212 Re-initializes the data with a row from a SELECT * FROM SchedGroups.
213
214 Returns self. Raises exception if the row is None or otherwise invalid.
215 """
216
217 if aoRow is None:
218 raise TMRowNotFound('SchedGroup not found.');
219
220 self.idSchedGroup = aoRow[0];
221 self.tsEffective = aoRow[1];
222 self.tsExpire = aoRow[2];
223 self.uidAuthor = aoRow[3];
224 self.sName = aoRow[4];
225 self.sDescription = aoRow[5];
226 self.fEnabled = aoRow[6];
227 self.enmScheduler = aoRow[7];
228 self.idBuildSrc = aoRow[8];
229 self.idBuildSrcTestSuite = aoRow[9];
230 self.sComment = aoRow[10];
231 return self;
232
233 def initFromDbWithId(self, oDb, idSchedGroup, tsNow = None, sPeriodBack = None):
234 """
235 Initialize the object from the database.
236 """
237 oDb.execute(self.formatSimpleNowAndPeriodQuery(oDb,
238 'SELECT *\n'
239 'FROM SchedGroups\n'
240 'WHERE idSchedGroup = %s\n'
241 , ( idSchedGroup,), tsNow, sPeriodBack));
242 aoRow = oDb.fetchOne()
243 if aoRow is None:
244 raise TMRowNotFound('idSchedGroup=%s not found (tsNow=%s, sPeriodBack=%s)' % (idSchedGroup, tsNow, sPeriodBack));
245 return self.initFromDbRow(aoRow);
246
247
248class SchedGroupDataEx(SchedGroupData):
249 """
250 Extended scheduling group data.
251
252 Note! Similar to TestGroupDataEx.
253 """
254
255 ksParam_aoMembers = 'SchedGroup_aoMembers';
256 ksParam_aoTestBoxes = 'SchedGroup_aoTestboxes';
257 kasAltArrayNull = [ 'aoMembers', 'aoTestboxes' ];
258
259 ## Helper parameter containing the comma separated list with the IDs of
260 # potential members found in the parameters.
261 ksParam_aidTestGroups = 'TestGroupDataEx_aidTestGroups';
262 ## Ditto for testbox meembers.
263 ksParam_aidTestBoxes = 'TestGroupDataEx_aidTestBoxes';
264
265
266 def __init__(self):
267 SchedGroupData.__init__(self);
268 self.aoMembers = [] # type: list[SchedGroupMemberDataEx]
269 self.aoTestBoxes = [] # type: list[TestBoxDataForSchedGroup]
270
271 # The two build sources for the sake of convenience.
272 self.oBuildSrc = None # type: BuildSourceData
273 self.oBuildSrcValidationKit = None # type: BuildSourceData
274
275 def _initExtraMembersFromDb(self, oDb, tsNow = None, sPeriodBack = None):
276 """
277 Worker shared by the initFromDb* methods.
278 Returns self. Raises exception if no row or database error.
279 """
280 #
281 # Clear all members upfront so the object has some kind of consistency
282 # if anything below raises exceptions.
283 #
284 self.oBuildSrc = None;
285 self.oBuildSrcValidationKit = None;
286 self.aoTestBoxes = [];
287 self.aoMembers = [];
288
289 #
290 # Build source.
291 #
292 if self.idBuildSrc:
293 self.oBuildSrc = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrc, tsNow, sPeriodBack);
294
295 if self.idBuildSrcTestSuite:
296 self.oBuildSrcValidationKit = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrcTestSuite,
297 tsNow, sPeriodBack);
298
299 #
300 # Test Boxes.
301 #
302 self.aoTestBoxes = TestBoxLogic(oDb).fetchForSchedGroup(self.idSchedGroup, tsNow);
303
304 #
305 # Test groups.
306 # The fetchForChangeLog method makes ASSUMPTIONS about sorting!
307 #
308 oDb.execute('SELECT SchedGroupMembers.*, TestGroups.*\n'
309 'FROM SchedGroupMembers\n'
310 'LEFT OUTER JOIN TestGroups ON (SchedGroupMembers.idTestGroup = TestGroups.idTestGroup)\n'
311 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
312 + self.formatSimpleNowAndPeriod(oDb, tsNow, sPeriodBack, sTablePrefix = 'SchedGroupMembers.')
313 + self.formatSimpleNowAndPeriod(oDb, tsNow, sPeriodBack, sTablePrefix = 'TestGroups.') +
314 'ORDER BY SchedGroupMembers.idTestGroupPreReq ASC NULLS FIRST,\n'
315 ' TestGroups.sName,\n'
316 ' SchedGroupMembers.idTestGroup\n'
317 , (self.idSchedGroup,));
318 for aoRow in oDb.fetchAll():
319 self.aoMembers.append(SchedGroupMemberDataEx().initFromDbRow(aoRow));
320 return self;
321
322 def initFromDbRowEx(self, aoRow, oDb, tsNow = None):
323 """
324 Reinitialize from a SELECT * FROM SchedGroups row. Will query the
325 necessary additional data from oDb using tsNow.
326 Returns self. Raises exception if no row or database error.
327 """
328 SchedGroupData.initFromDbRow(self, aoRow);
329 return self._initExtraMembersFromDb(oDb, tsNow);
330
331 def initFromDbWithId(self, oDb, idSchedGroup, tsNow = None, sPeriodBack = None):
332 """
333 Initialize the object from the database.
334 """
335 SchedGroupData.initFromDbWithId(self, oDb, idSchedGroup, tsNow, sPeriodBack);
336 return self._initExtraMembersFromDb(oDb, tsNow, sPeriodBack);
337
338 def getDataAttributes(self):
339 asAttributes = SchedGroupData.getDataAttributes(self);
340 asAttributes.remove('oBuildSrc');
341 asAttributes.remove('oBuildSrcValidationKit');
342 return asAttributes;
343
344 def getAttributeParamNullValues(self, sAttr):
345 if sAttr not in [ 'aoMembers', 'aoTestBoxes' ]:
346 return SchedGroupData.getAttributeParamNullValues(self, sAttr);
347 return ['', [], None];
348
349 def convertParamToAttribute(self, sAttr, sParam, oValue, oDisp, fStrict):
350 aoNewValue = [];
351 if sAttr == 'aoMembers':
352 aidSelected = oDisp.getListOfIntParams(sParam, iMin = 1, iMax = 0x7ffffffe, aiDefaults = [])
353 sIds = oDisp.getStringParam(self.ksParam_aidTestGroups, sDefault = '');
354 for idTestGroup in sIds.split(','):
355 try: idTestGroup = int(idTestGroup);
356 except: pass;
357 oDispWrapper = self.DispWrapper(oDisp, '%s[%s][%%s]' % (SchedGroupDataEx.ksParam_aoMembers, idTestGroup,))
358 oMember = SchedGroupMemberDataEx().initFromParams(oDispWrapper, fStrict = False);
359 if idTestGroup in aidSelected:
360 oMember.idTestGroup = idTestGroup;
361 aoNewValue.append(oMember);
362 elif sAttr == 'aoTestBoxes':
363 aidSelected = oDisp.getListOfIntParams(sParam, iMin = 1, iMax = 0x7ffffffe, aiDefaults = [])
364 sIds = oDisp.getStringParam(self.ksParam_aidTestBoxes, sDefault = '');
365 for idTestBox in sIds.split(','):
366 try: idTestBox = int(idTestBox);
367 except: pass;
368 oDispWrapper = self.DispWrapper(oDisp, '%s[%s][%%s]' % (SchedGroupDataEx.ksParam_aoTestBoxes, idTestBox,))
369 oBoxInGrp = TestBoxDataForSchedGroup().initFromParams(oDispWrapper, fStrict = False);
370 if idTestBox in aidSelected:
371 oBoxInGrp.idTestBox = idTestBox;
372 aoNewValue.append(oBoxInGrp);
373 else:
374 return SchedGroupData.convertParamToAttribute(self, sAttr, sParam, oValue, oDisp, fStrict);
375 return aoNewValue;
376
377 def _validateAndConvertAttribute(self, sAttr, sParam, oValue, aoNilValues, fAllowNull, oDb):
378 if sAttr not in [ 'aoMembers', 'aoTestBoxes' ]:
379 return SchedGroupData._validateAndConvertAttribute(self, sAttr, sParam, oValue, aoNilValues, fAllowNull, oDb);
380
381 if oValue in aoNilValues:
382 return ([], None);
383
384 asErrors = [];
385 aoNewMembers = [];
386 if sAttr == 'aoMembers':
387 asAllowNulls = ['bmHourlySchedule', 'idTestGroupPreReq', 'tsEffective', 'tsExpire', 'uidAuthor', ];
388 if self.idSchedGroup in [None, '-1', -1]:
389 asAllowNulls.append('idSchedGroup'); # Probably new group, so allow null scheduling group.
390
391 for oOldMember in oValue:
392 oNewMember = SchedGroupMemberDataEx().initFromOther(oOldMember);
393 aoNewMembers.append(oNewMember);
394
395 dErrors = oNewMember.validateAndConvertEx(asAllowNulls, oDb, ModelDataBase.ksValidateFor_Other);
396 if dErrors:
397 asErrors.append(str(dErrors));
398
399 if not asErrors:
400 for i, _ in enumerate(aoNewMembers):
401 idTestGroup = aoNewMembers[i];
402 for j in range(i + 1, len(aoNewMembers)):
403 if aoNewMembers[j].idTestGroup == idTestGroup:
404 asErrors.append('Duplicate test group #%d!' % (idTestGroup, ));
405 break;
406 else:
407 asAllowNulls = list(TestBoxDataForSchedGroup.kasAllowNullAttributes);
408 if self.idSchedGroup in [None, '-1', -1]:
409 asAllowNulls.append('idSchedGroup'); # Probably new group, so allow null scheduling group.
410
411 for oOldMember in oValue:
412 oNewMember = TestBoxDataForSchedGroup().initFromOther(oOldMember);
413 aoNewMembers.append(oNewMember);
414
415 dErrors = oNewMember.validateAndConvertEx(asAllowNulls, oDb, ModelDataBase.ksValidateFor_Other);
416 if dErrors:
417 asErrors.append(str(dErrors));
418
419 if not asErrors:
420 for i, _ in enumerate(aoNewMembers):
421 idTestBox = aoNewMembers[i];
422 for j in range(i + 1, len(aoNewMembers)):
423 if aoNewMembers[j].idTestBox == idTestBox:
424 asErrors.append('Duplicate test box #%d!' % (idTestBox, ));
425 break;
426
427 return (aoNewMembers, None if not asErrors else '<br>\n'.join(asErrors));
428
429 def _validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor = ModelDataBase.ksValidateFor_Other):
430 dErrors = SchedGroupData._validateAndConvertWorker(self, asAllowNullAttributes, oDb, enmValidateFor);
431
432 #
433 # Fetch the extended build source bits.
434 #
435 if self.ksParam_idBuildSrc not in dErrors:
436 if self.idBuildSrc in self.getAttributeParamNullValues('idBuildSrc') \
437 or self.idBuildSrc is None:
438 self.oBuildSrc = None;
439 else:
440 try:
441 self.oBuildSrc = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrc);
442 except Exception as oXcpt:
443 self.oBuildSrc = BuildSourceData();
444 dErrors[self.ksParam_idBuildSrc] = str(oXcpt);
445
446 if self.ksParam_idBuildSrcTestSuite not in dErrors:
447 if self.idBuildSrcTestSuite in self.getAttributeParamNullValues('idBuildSrcTestSuite') \
448 or self.idBuildSrcTestSuite is None:
449 self.oBuildSrcValidationKit = None;
450 else:
451 try:
452 self.oBuildSrcValidationKit = BuildSourceData().initFromDbWithId(oDb, self.idBuildSrcTestSuite);
453 except Exception as oXcpt:
454 self.oBuildSrcValidationKit = BuildSourceData();
455 dErrors[self.ksParam_idBuildSrcTestSuite] = str(oXcpt);
456
457 return dErrors;
458
459
460
461class SchedGroupLogic(ModelLogicBase): # pylint: disable=too-few-public-methods
462 """
463 SchedGroup logic.
464 """
465
466 def __init__(self, oDb):
467 ModelLogicBase.__init__(self, oDb);
468 self.dCache = None;
469
470 #
471 # Standard methods.
472 #
473
474 def fetchForListing(self, iStart, cMaxRows, tsNow, aiSortColumns = None):
475 """
476 Fetches build sources.
477
478 Returns an array (list) of BuildSourceData items, empty list if none.
479 Raises exception on error.
480 """
481 _ = aiSortColumns;
482
483 if tsNow is None:
484 self._oDb.execute('SELECT *\n'
485 'FROM SchedGroups\n'
486 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
487 'ORDER BY fEnabled DESC, sName DESC\n'
488 'LIMIT %s OFFSET %s\n'
489 , (cMaxRows, iStart,));
490 else:
491 self._oDb.execute('SELECT *\n'
492 'FROM SchedGroups\n'
493 'WHERE tsExpire > %s\n'
494 ' AND tsEffective <= %s\n'
495 'ORDER BY fEnabled DESC, sName DESC\n'
496 'LIMIT %s OFFSET %s\n'
497 , (tsNow, tsNow, cMaxRows, iStart,));
498
499 aoRet = [];
500 for aoRow in self._oDb.fetchAll():
501 aoRet.append(SchedGroupDataEx().initFromDbRowEx(aoRow, self._oDb, tsNow));
502 return aoRet;
503
504 def fetchForChangeLog(self, idSchedGroup, iStart, cMaxRows, tsNow): # pylint: disable=too-many-locals,too-many-statements
505 """
506 Fetches change log entries for a scheduling group.
507
508 Returns an array of ChangeLogEntry instance and an indicator whether
509 there are more entries.
510 Raises exception on error.
511 """
512
513 ## @todo calc changes to scheduler group!
514
515 if tsNow is None:
516 tsNow = self._oDb.getCurrentTimestamp();
517
518 #
519 # First gather the change log timeline using the effective dates.
520 # (ASSUMES that we'll always have a separate delete entry, rather
521 # than just setting tsExpire.)
522 #
523 self._oDb.execute('''
524(
525SELECT tsEffective,
526 uidAuthor
527FROM SchedGroups
528WHERE idSchedGroup = %s
529 AND tsEffective <= %s
530ORDER BY tsEffective DESC
531) UNION (
532SELECT tsEffective,
533 uidAuthor
534FROM SchedGroupMembers
535WHERE idSchedGroup = %s
536 AND tsEffective <= %s
537ORDER BY tsEffective DESC
538) UNION (
539SELECT tsEffective,
540 uidAuthor
541FROM TestBoxesInSchedGroups
542WHERE idSchedGroup = %s
543 AND tsEffective <= %s
544ORDER BY tsEffective DESC
545)
546ORDER BY tsEffective DESC
547LIMIT %s OFFSET %s
548''', (idSchedGroup, tsNow, idSchedGroup, tsNow, idSchedGroup, tsNow, cMaxRows + 1, iStart, ));
549
550 aoEntries = [] # type: list[ChangeLogEntry]
551 tsPrevious = tsNow;
552 for aoDbRow in self._oDb.fetchAll():
553 (tsEffective, uidAuthor) = aoDbRow;
554 aoEntries.append(ChangeLogEntry(uidAuthor, None, tsEffective, tsPrevious, None, None, []));
555 tsPrevious = db.dbTimestampPlusOneTick(tsEffective);
556
557 if True: # pylint: disable=using-constant-test
558 #
559 # Fetch data for each for each change log entry point.
560 #
561 # We add one tick to the timestamp here to skip past delete records
562 # that only there to record the user doing the deletion.
563 #
564 for iEntry, oEntry in enumerate(aoEntries):
565 oEntry.oNewRaw = SchedGroupDataEx().initFromDbWithId(self._oDb, idSchedGroup,
566 db.dbTimestampPlusOneTick(oEntry.tsEffective));
567 if iEntry > 0:
568 aoEntries[iEntry - 1].oOldRaw = oEntry.oNewRaw;
569
570 # Chop off the +1 entry, if any.
571 fMore = len(aoEntries) > cMaxRows;
572 if fMore:
573 aoEntries = aoEntries[:-1];
574
575 # Figure out the changes.
576 for oEntry in aoEntries:
577 oOld = oEntry.oOldRaw;
578 if not oOld:
579 break;
580 oNew = oEntry.oNewRaw;
581 aoChanges = oEntry.aoChanges;
582 for sAttr in oNew.getDataAttributes():
583 if sAttr in [ 'tsEffective', 'tsExpire', 'uidAuthor', ]:
584 continue;
585 oOldAttr = getattr(oOld, sAttr);
586 oNewAttr = getattr(oNew, sAttr);
587 if oOldAttr == oNewAttr:
588 continue;
589 if sAttr in [ 'aoMembers', 'aoTestBoxes', ]:
590 iNew = 0;
591 iOld = 0;
592 asNewAttr = [];
593 asOldAttr = [];
594 if sAttr == 'aoMembers':
595 # ASSUMES aoMembers is sorted by idTestGroupPreReq (nulls first), oTestGroup.sName, idTestGroup!
596 while iNew < len(oNewAttr) and iOld < len(oOldAttr):
597 if oNewAttr[iNew].idTestGroup == oOldAttr[iOld].idTestGroup:
598 if oNewAttr[iNew].idTestGroupPreReq != oOldAttr[iOld].idTestGroupPreReq:
599 if oNewAttr[iNew].idTestGroupPreReq is None:
600 asOldAttr.append('Dropped test group #%s (%s) dependency on #%s'
601 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName,
602 oOldAttr[iOld].idTestGroupPreReq));
603 elif oOldAttr[iOld].idTestGroupPreReq is None:
604 asNewAttr.append('Added test group #%s (%s) dependency on #%s'
605 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName,
606 oNewAttr[iOld].idTestGroupPreReq));
607 else:
608 asNewAttr.append('Test group #%s (%s) dependency on #%s'
609 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName,
610 oNewAttr[iNew].idTestGroupPreReq));
611 asOldAttr.append('Test group #%s (%s) dependency on #%s'
612 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName,
613 oOldAttr[iOld].idTestGroupPreReq));
614 if oNewAttr[iNew].iSchedPriority != oOldAttr[iOld].iSchedPriority:
615 asNewAttr.append('Test group #%s (%s) priority %s'
616 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName,
617 oNewAttr[iNew].iSchedPriority));
618 asOldAttr.append('Test group #%s (%s) priority %s'
619 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName,
620 oOldAttr[iOld].iSchedPriority));
621 iNew += 1;
622 iOld += 1;
623 elif oNewAttr[iNew].oTestGroup.sName < oOldAttr[iOld].oTestGroup.sName \
624 or ( oNewAttr[iNew].oTestGroup.sName == oOldAttr[iOld].oTestGroup.sName
625 and oNewAttr[iNew].idTestGroup < oOldAttr[iOld].idTestGroup):
626 asNewAttr.append('New test group #%s - %s'
627 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName));
628 iNew += 1;
629 else:
630 asOldAttr.append('Removed test group #%s - %s'
631 % (oOldAttr[iOld].idTestGroup, oOldAttr[iOld].oTestGroup.sName));
632 iOld += 1;
633 while iNew < len(oNewAttr):
634 asNewAttr.append('New test group #%s - %s'
635 % (oNewAttr[iNew].idTestGroup, oNewAttr[iNew].oTestGroup.sName));
636 iNew += 1;
637 while iOld < len(oOldAttr):
638 asOldAttr.append('Removed test group #%s - %s'
639 % (oOldAttr[iOld].idTestGroup, oOldAttr[iOld].oTestGroup.sName));
640 iOld += 1;
641 else:
642 dNewIds = { oBoxInGrp.idTestBox: oBoxInGrp for oBoxInGrp in oNewAttr };
643 dOldIds = { oBoxInGrp.idTestBox: oBoxInGrp for oBoxInGrp in oOldAttr };
644 hCommonIds = set(dNewIds.keys()) & set(dOldIds.keys());
645 for idTestBox in hCommonIds:
646 oNewBoxInGrp = dNewIds[idTestBox];
647 oOldBoxInGrp = dOldIds[idTestBox];
648 if oNewBoxInGrp.iSchedPriority != oOldBoxInGrp.iSchedPriority:
649 asNewAttr.append('Test box \'%s\' (#%s) priority %s'
650 % (getattr(oNewBoxInGrp.oTestBox, 'sName', '[Partial DB]'),
651 oNewBoxInGrp.idTestBox, oNewBoxInGrp.iSchedPriority));
652 asOldAttr.append('Test box \'%s\' (#%s) priority %s'
653 % (getattr(oOldBoxInGrp.oTestBox, 'sName', '[Partial DB]'),
654 oOldBoxInGrp.idTestBox, oOldBoxInGrp.iSchedPriority));
655 asNewAttr = sorted(asNewAttr);
656 asOldAttr = sorted(asOldAttr);
657 for idTestBox in set(dNewIds.keys()) - hCommonIds:
658 oNewBoxInGrp = dNewIds[idTestBox];
659 asNewAttr.append('New test box \'%s\' (#%s) priority %s'
660 % (getattr(oNewBoxInGrp.oTestBox, 'sName', '[Partial DB]'),
661 oNewBoxInGrp.idTestBox, oNewBoxInGrp.iSchedPriority));
662 for idTestBox in set(dOldIds.keys()) - hCommonIds:
663 oOldBoxInGrp = dOldIds[idTestBox];
664 asOldAttr.append('Removed test box \'%s\' (#%s) priority %s'
665 % (getattr(oOldBoxInGrp.oTestBox, 'sName', '[Partial DB]'),
666 oOldBoxInGrp.idTestBox, oOldBoxInGrp.iSchedPriority));
667
668 if asNewAttr or asOldAttr:
669 aoChanges.append(AttributeChangeEntryPre(sAttr, oNewAttr, oOldAttr,
670 '\n'.join(asNewAttr), '\n'.join(asOldAttr)));
671 else:
672 aoChanges.append(AttributeChangeEntry(sAttr, oNewAttr, oOldAttr, str(oNewAttr), str(oOldAttr)));
673
674 else:
675 ##
676 ## @todo Incomplete: A more complicate apporach, probably faster though.
677 ##
678 def findEntry(tsEffective, iPrev = 0):
679 """ Find entry with matching effective + expiration time """
680 self._oDb.dprint('findEntry: iPrev=%s len(aoEntries)=%s tsEffective=%s' % (iPrev, len(aoEntries), tsEffective));
681 while iPrev < len(aoEntries):
682 self._oDb.dprint('%s iPrev=%u' % (aoEntries[iPrev].tsEffective, iPrev, ));
683 if aoEntries[iPrev].tsEffective > tsEffective:
684 iPrev += 1;
685 elif aoEntries[iPrev].tsEffective == tsEffective:
686 self._oDb.dprint('hit %u' % (iPrev,));
687 return iPrev;
688 else:
689 break;
690 self._oDb.dprint('%s not found!' % (tsEffective,));
691 return -1;
692
693 fMore = True;
694
695 #
696 # Track scheduling group changes. Not terribly efficient for large cMaxRows
697 # values, but not in the mood for figure out if there is any way to optimize that.
698 #
699 self._oDb.execute('''
700SELECT *
701FROM SchedGroups
702WHERE idSchedGroup = %s
703 AND tsEffective <= %s
704ORDER BY tsEffective DESC
705LIMIT %s''', (idSchedGroup, aoEntries[0].tsEffective, cMaxRows + 1,));
706
707 iEntry = 0;
708 aaoRows = self._oDb.fetchAll();
709 for iRow, oRow in enumerate(aaoRows):
710 oNew = SchedGroupData().initFromDbRow(oRow);
711 iEntry = findEntry(oNew.tsEffective, iEntry);
712 self._oDb.dprint('iRow=%s iEntry=%s' % (iRow, iEntry));
713 if iEntry < 0:
714 break;
715 oEntry = aoEntries[iEntry];
716 aoChanges = oEntry.aoChanges;
717 oEntry.oNewRaw = oNew;
718 if iRow + 1 < len(aaoRows):
719 oOld = SchedGroupData().initFromDbRow(aaoRows[iRow + 1]);
720 self._oDb.dprint('oOld=%s' % (oOld,));
721 for sAttr in oNew.getDataAttributes():
722 if sAttr not in [ 'tsEffective', 'tsExpire', 'uidAuthor', ]:
723 oOldAttr = getattr(oOld, sAttr);
724 oNewAttr = getattr(oNew, sAttr);
725 if oOldAttr != oNewAttr:
726 aoChanges.append(AttributeChangeEntry(sAttr, oNewAttr, oOldAttr, str(oNewAttr), str(oOldAttr)));
727 else:
728 self._oDb.dprint('New');
729
730 #
731 # ...
732 #
733
734 # FInally
735 UserAccountLogic(self._oDb).resolveChangeLogAuthors(aoEntries);
736 return (aoEntries, fMore);
737
738
739 def addEntry(self, oData, uidAuthor, fCommit = False):
740 """Add Scheduling Group record"""
741
742 #
743 # Validate.
744 #
745 dDataErrors = oData.validateAndConvert(self._oDb, oData.ksValidateFor_Add);
746 if dDataErrors:
747 raise TMInvalidData('Invalid data passed to addEntry: %s' % (dDataErrors,));
748 if self.exists(oData.sName):
749 raise TMRowAlreadyExists('Scheduling group "%s" already exists.' % (oData.sName,));
750
751 #
752 # Add it.
753 #
754 self._oDb.execute('INSERT INTO SchedGroups (\n'
755 ' uidAuthor,\n'
756 ' sName,\n'
757 ' sDescription,\n'
758 ' fEnabled,\n'
759 ' enmScheduler,\n'
760 ' idBuildSrc,\n'
761 ' idBuildSrcTestSuite,\n'
762 ' sComment)\n'
763 'VALUES (%s, %s, %s, %s, %s, %s, %s, %s)\n'
764 'RETURNING idSchedGroup\n'
765 , ( uidAuthor,
766 oData.sName,
767 oData.sDescription,
768 oData.fEnabled,
769 oData.enmScheduler,
770 oData.idBuildSrc,
771 oData.idBuildSrcTestSuite,
772 oData.sComment ));
773 idSchedGroup = self._oDb.fetchOne()[0];
774 oData.idSchedGroup = idSchedGroup;
775
776 for oBoxInGrp in oData.aoTestBoxes:
777 oBoxInGrp.idSchedGroup = idSchedGroup;
778 self._addSchedGroupTestBox(uidAuthor, oBoxInGrp);
779
780 for oMember in oData.aoMembers:
781 oMember.idSchedGroup = idSchedGroup;
782 self._addSchedGroupMember(uidAuthor, oMember);
783
784 self._oDb.maybeCommit(fCommit);
785 return True;
786
787 def editEntry(self, oData, uidAuthor, fCommit = False):
788 """Edit Scheduling Group record"""
789
790 #
791 # Validate input and retrieve the old data.
792 #
793 dErrors = oData.validateAndConvert(self._oDb, oData.ksValidateFor_Edit);
794 if dErrors:
795 raise TMInvalidData('editEntry got invalid data: %s' % (dErrors,));
796 self._assertUnique(oData.sName, oData.idSchedGroup);
797 oOldData = SchedGroupDataEx().initFromDbWithId(self._oDb, oData.idSchedGroup);
798
799 #
800 # Make the changes.
801 #
802 if not oData.isEqualEx(oOldData, [ 'tsEffective', 'tsExpire', 'uidAuthor', 'aoMembers', 'aoTestBoxes',
803 'oBuildSrc', 'oBuildSrcValidationKit', ]):
804 self._historizeEntry(oData.idSchedGroup);
805 self._readdEntry(uidAuthor, oData);
806
807 # Remove groups.
808 for oOld in oOldData.aoMembers:
809 fRemove = True;
810 for oNew in oData.aoMembers:
811 if oNew.idTestGroup == oOld.idTestGroup:
812 fRemove = False;
813 break;
814 if fRemove:
815 self._removeSchedGroupMember(uidAuthor, oOld);
816
817 # Add / modify groups.
818 for oMember in oData.aoMembers:
819 oOldMember = None;
820 for oOld in oOldData.aoMembers:
821 if oOld.idTestGroup == oMember.idTestGroup:
822 oOldMember = oOld;
823 break;
824
825 oMember.idSchedGroup = oData.idSchedGroup;
826 if oOldMember is None:
827 self._addSchedGroupMember(uidAuthor, oMember);
828 elif not oMember.isEqualEx(oOldMember, ['tsEffective', 'tsExpire', 'uidAuthor', 'oTestGroup']):
829 self._historizeSchedGroupMember(oMember);
830 self._addSchedGroupMember(uidAuthor, oMember);
831
832 # Remove testboxes.
833 for oOld in oOldData.aoTestBoxes:
834 fRemove = True;
835 for oNew in oData.aoTestBoxes:
836 if oNew.idTestBox == oOld.idTestBox:
837 fRemove = False;
838 break;
839 if fRemove:
840 self._removeSchedGroupTestBox(uidAuthor, oOld);
841
842 # Add / modify testboxes.
843 for oBoxInGrp in oData.aoTestBoxes:
844 oOldBoxInGrp = None;
845 for oOld in oOldData.aoTestBoxes:
846 if oOld.idTestBox == oBoxInGrp.idTestBox:
847 oOldBoxInGrp = oOld;
848 break;
849
850 oBoxInGrp.idSchedGroup = oData.idSchedGroup;
851 if oOldBoxInGrp is None:
852 self._addSchedGroupTestBox(uidAuthor, oBoxInGrp);
853 elif not oBoxInGrp.isEqualEx(oOldBoxInGrp, ['tsEffective', 'tsExpire', 'uidAuthor', 'oTestBox']):
854 self._historizeSchedGroupTestBox(oBoxInGrp);
855 self._addSchedGroupTestBox(uidAuthor, oBoxInGrp);
856
857 self._oDb.maybeCommit(fCommit);
858 return True;
859
860 def removeEntry(self, uidAuthor, idSchedGroup, fCascade = False, fCommit = False):
861 """
862 Deletes a scheduling group.
863 """
864 _ = fCascade;
865
866 #
867 # Input validation and retrival of current data.
868 #
869 if idSchedGroup == 1:
870 raise TMRowInUse('Cannot remove the default scheduling group (id 1).');
871 oData = SchedGroupDataEx().initFromDbWithId(self._oDb, idSchedGroup);
872
873 #
874 # Remove the test box member records.
875 #
876 for oBoxInGrp in oData.aoTestBoxes:
877 self._removeSchedGroupTestBox(uidAuthor, oBoxInGrp);
878 self._oDb.execute('UPDATE TestBoxesInSchedGroups\n'
879 'SET tsExpire = CURRENT_TIMESTAMP\n'
880 'WHERE idSchedGroup = %s\n'
881 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
882 , (idSchedGroup,));
883
884 #
885 # Remove the test group member records.
886 #
887 for oMember in oData.aoMembers:
888 self._removeSchedGroupMember(uidAuthor, oMember);
889 self._oDb.execute('UPDATE SchedGroupMembers\n'
890 'SET tsExpire = CURRENT_TIMESTAMP\n'
891 'WHERE idSchedGroup = %s\n'
892 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
893 , (idSchedGroup,));
894
895 #
896 # Now the SchedGroups entry.
897 #
898 (tsCur, tsCurMinusOne) = self._oDb.getCurrentTimestamps();
899 if oData.tsEffective != tsCur and oData.tsEffective != tsCurMinusOne:
900 self._historizeEntry(idSchedGroup, tsCurMinusOne);
901 self._readdEntry(uidAuthor, oData, tsCurMinusOne);
902 self._historizeEntry(idSchedGroup);
903 self._oDb.execute('UPDATE SchedGroups\n'
904 'SET tsExpire = CURRENT_TIMESTAMP\n'
905 'WHERE idSchedGroup = %s\n'
906 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
907 , (idSchedGroup,))
908
909 self._oDb.maybeCommit(fCommit)
910 return True;
911
912
913 def cachedLookup(self, idSchedGroup):
914 """
915 Looks up the most recent SchedGroupData object for idSchedGroup
916 via an object cache.
917
918 Returns a shared SchedGroupData object. None if not found.
919 Raises exception on DB error.
920 """
921 if self.dCache is None:
922 self.dCache = self._oDb.getCache('SchedGroup');
923
924 oEntry = self.dCache.get(idSchedGroup, None);
925 if oEntry is None:
926 self._oDb.execute('SELECT *\n'
927 'FROM SchedGroups\n'
928 'WHERE idSchedGroup = %s\n'
929 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
930 , (idSchedGroup, ));
931 if self._oDb.getRowCount() == 0:
932 # Maybe it was deleted, try get the last entry.
933 self._oDb.execute('SELECT *\n'
934 'FROM SchedGroups\n'
935 'WHERE idSchedGroup = %s\n'
936 'ORDER BY tsExpire DESC\n'
937 'LIMIT 1\n'
938 , (idSchedGroup, ));
939 elif self._oDb.getRowCount() > 1:
940 raise self._oDb.integrityException('%s infinity rows for %s' % (self._oDb.getRowCount(), idSchedGroup));
941
942 if self._oDb.getRowCount() == 1:
943 oEntry = SchedGroupData().initFromDbRow(self._oDb.fetchOne());
944 self.dCache[idSchedGroup] = oEntry;
945 return oEntry;
946
947
948 #
949 # Other methods.
950 #
951
952 def fetchOrderedByName(self, tsNow = None):
953 """
954 Return list of objects of type SchedGroups ordered by name.
955 May raise exception on database error.
956 """
957 if tsNow is None:
958 self._oDb.execute('SELECT *\n'
959 'FROM SchedGroups\n'
960 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
961 'ORDER BY sName ASC\n');
962 else:
963 self._oDb.execute('SELECT *\n'
964 'FROM SchedGroups\n'
965 'WHERE tsExpire > %s\n'
966 ' AND tsEffective <= %s\n'
967 'ORDER BY sName ASC\n'
968 , (tsNow, tsNow,));
969 aoRet = []
970 for _ in range(self._oDb.getRowCount()):
971 aoRet.append(SchedGroupData().initFromDbRow(self._oDb.fetchOne()));
972 return aoRet;
973
974
975 def getAll(self, tsEffective = None):
976 """
977 Gets the list of all scheduling groups.
978 Returns an array of SchedGroupData instances.
979 """
980 if tsEffective is None:
981 self._oDb.execute('SELECT *\n'
982 'FROM SchedGroups\n'
983 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n');
984 else:
985 self._oDb.execute('SELECT *\n'
986 'FROM SchedGroups\n'
987 'WHERE tsExpire > %s\n'
988 ' AND tsEffective <= %s\n'
989 , (tsEffective, tsEffective));
990 aoRet = [];
991 for aoRow in self._oDb.fetchAll():
992 aoRet.append(SchedGroupData().initFromDbRow(aoRow));
993 return aoRet;
994
995 def getSchedGroupsForCombo(self, tsEffective = None):
996 """
997 Gets the list of active scheduling groups for a combo box.
998 Returns an array of (value [idSchedGroup], drop-down-name [sName],
999 hover-text [sDescription]) tuples.
1000 """
1001 if tsEffective is None:
1002 self._oDb.execute('SELECT idSchedGroup, sName, sDescription\n'
1003 'FROM SchedGroups\n'
1004 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
1005 'ORDER BY sName');
1006 else:
1007 self._oDb.execute('SELECT idSchedGroup, sName, sDescription\n'
1008 'FROM SchedGroups\n'
1009 'WHERE tsExpire > %s\n'
1010 ' AND tsEffective <= %s\n'
1011 'ORDER BY sName'
1012 , (tsEffective, tsEffective));
1013 return self._oDb.fetchAll();
1014
1015
1016 def getMembers(self, idSchedGroup, tsEffective = None):
1017 """
1018 Gets the scheduling groups members for the given scheduling group.
1019
1020 Returns an array of SchedGroupMemberDataEx instances (sorted by
1021 priority (descending) and idTestGroup). May raise exception DB error.
1022 """
1023
1024 if tsEffective is None:
1025 self._oDb.execute('SELECT *\n'
1026 'FROM SchedGroupMembers, TestGroups\n'
1027 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
1028 ' AND SchedGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
1029 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
1030 ' AND TestGroups.tsExpire = \'infinity\'::TIMESTAMP\n'
1031 'ORDER BY SchedGroupMembers.iSchedPriority DESC, SchedGroupMembers.idTestGroup\n'
1032 , (idSchedGroup,));
1033 else:
1034 self._oDb.execute('SELECT *\n'
1035 'FROM SchedGroupMembers, TestGroups\n'
1036 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
1037 ' AND SchedGroupMembers.tsExpire < %s\n'
1038 ' AND SchedGroupMembers.tsEffective >= %s\n'
1039 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
1040 ' AND TestGroups.tsExpire < %s\n'
1041 ' AND TestGroups.tsEffective >= %s\n'
1042 'ORDER BY SchedGroupMembers.iSchedPriority DESC, SchedGroupMembers.idTestGroup\n'
1043 , (idSchedGroup, tsEffective, tsEffective, tsEffective, tsEffective, ));
1044 aaoRows = self._oDb.fetchAll();
1045 aoRet = [];
1046 for aoRow in aaoRows:
1047 aoRet.append(SchedGroupMemberDataEx().initFromDbRow(aoRow));
1048 return aoRet;
1049
1050 def getTestCasesForGroup(self, idSchedGroup, cMax = None):
1051 """
1052 Gets the enabled testcases w/ testgroup+priority for the given scheduling group.
1053
1054 Returns an array of TestCaseData instances (ordered by group id, descending
1055 testcase priority, and testcase IDs) with an extra iSchedPriority member.
1056 May raise exception on DB error or if the result exceeds cMax.
1057 """
1058
1059 self._oDb.execute('SELECT TestGroupMembers.idTestGroup, TestGroupMembers.iSchedPriority, TestCases.*\n'
1060 'FROM SchedGroupMembers, TestGroups, TestGroupMembers, TestCases\n'
1061 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
1062 ' AND SchedGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
1063 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
1064 ' AND TestGroups.tsExpire = \'infinity\'::TIMESTAMP\n'
1065 ' AND TestGroupMembers.idTestGroup = TestGroups.idTestGroup\n'
1066 ' AND TestGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
1067 ' AND TestCases.idTestCase = TestGroupMembers.idTestCase\n'
1068 ' AND TestCases.tsExpire = \'infinity\'::TIMESTAMP\n'
1069 ' AND TestCases.fEnabled = TRUE\n'
1070 'ORDER BY TestGroupMembers.idTestGroup, TestGroupMembers.iSchedPriority DESC, TestCases.idTestCase\n'
1071 , (idSchedGroup,));
1072
1073 if cMax is not None and self._oDb.getRowCount() > cMax:
1074 raise TMExceptionBase('Too many testcases for scheduling group %s: %s, max %s'
1075 % (idSchedGroup, cMax, self._oDb.getRowCount(),));
1076
1077 aoRet = [];
1078 for aoRow in self._oDb.fetchAll():
1079 oTestCase = TestCaseData().initFromDbRow(aoRow[2:]);
1080 oTestCase.idTestGroup = aoRow[0];
1081 oTestCase.iSchedPriority = aoRow[1];
1082 aoRet.append(oTestCase);
1083 return aoRet;
1084
1085 def getTestCaseArgsForGroup(self, idSchedGroup, cMax = None):
1086 """
1087 Gets the testcase argument variation w/ testgroup+priority for the given scheduling group.
1088
1089 Returns an array TestCaseArgsData instance (sorted by group and
1090 variation id) with an extra iSchedPriority member.
1091 May raise exception on DB error or if the result exceeds cMax.
1092 """
1093
1094 self._oDb.execute('SELECT TestGroupMembers.idTestGroup, TestGroupMembers.iSchedPriority, TestCaseArgs.*\n'
1095 'FROM SchedGroupMembers, TestGroups, TestGroupMembers, TestCaseArgs, TestCases\n'
1096 'WHERE SchedGroupMembers.idSchedGroup = %s\n'
1097 ' AND SchedGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
1098 ' AND TestGroups.idTestGroup = SchedGroupMembers.idTestGroup\n'
1099 ' AND TestGroups.tsExpire = \'infinity\'::TIMESTAMP\n'
1100 ' AND TestGroupMembers.idTestGroup = TestGroups.idTestGroup\n'
1101 ' AND TestGroupMembers.tsExpire = \'infinity\'::TIMESTAMP\n'
1102 ' AND TestCaseArgs.idTestCase = TestGroupMembers.idTestCase\n'
1103 ' AND TestCaseArgs.tsExpire = \'infinity\'::TIMESTAMP\n'
1104 ' AND ( TestGroupMembers.aidTestCaseArgs is NULL\n'
1105 ' OR TestCaseArgs.idTestCaseArgs = ANY(TestGroupMembers.aidTestCaseArgs) )\n'
1106 ' AND TestCases.idTestCase = TestCaseArgs.idTestCase\n'
1107 ' AND TestCases.tsExpire = \'infinity\'::TIMESTAMP\n'
1108 ' AND TestCases.fEnabled = TRUE\n'
1109 'ORDER BY TestGroupMembers.idTestGroup, TestGroupMembers.idTestCase, TestCaseArgs.idTestCaseArgs\n'
1110 , (idSchedGroup,));
1111
1112 if cMax is not None and self._oDb.getRowCount() > cMax:
1113 raise TMExceptionBase('Too many argument variations for scheduling group %s: %s, max %s'
1114 % (idSchedGroup, cMax, self._oDb.getRowCount(),));
1115
1116 aoRet = [];
1117 for aoRow in self._oDb.fetchAll():
1118 oVariation = TestCaseArgsData().initFromDbRow(aoRow[2:]);
1119 oVariation.idTestGroup = aoRow[0];
1120 oVariation.iSchedPriority = aoRow[1];
1121 aoRet.append(oVariation);
1122 return aoRet;
1123
1124 def exists(self, sName):
1125 """Checks if a group with the given name exists."""
1126 self._oDb.execute('SELECT idSchedGroup\n'
1127 'FROM SchedGroups\n'
1128 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
1129 ' AND sName = %s\n'
1130 'LIMIT 1\n'
1131 , (sName,));
1132 return self._oDb.getRowCount() > 0;
1133
1134 def getById(self, idSchedGroup):
1135 """Get Scheduling Group data by idSchedGroup"""
1136 self._oDb.execute('SELECT *\n'
1137 'FROM SchedGroups\n'
1138 'WHERE tsExpire = \'infinity\'::timestamp\n'
1139 ' AND idSchedGroup = %s;', (idSchedGroup,))
1140 aRows = self._oDb.fetchAll()
1141 if len(aRows) not in (0, 1):
1142 raise self._oDb.integrityException(
1143 'Found more than one scheduling groups with the same credentials. Database structure is corrupted.')
1144 try:
1145 return SchedGroupData().initFromDbRow(aRows[0])
1146 except IndexError:
1147 return None
1148
1149
1150 #
1151 # Internal helpers.
1152 #
1153
1154 def _assertUnique(self, sName, idSchedGroupIgnore = None):
1155 """
1156 Checks that the scheduling group name is unique.
1157 Raises exception if the name is already in use.
1158 """
1159 if idSchedGroupIgnore is None:
1160 self._oDb.execute('SELECT idSchedGroup\n'
1161 'FROM SchedGroups\n'
1162 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
1163 ' AND sName = %s\n'
1164 , ( sName, ) );
1165 else:
1166 self._oDb.execute('SELECT idSchedGroup\n'
1167 'FROM SchedGroups\n'
1168 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
1169 ' AND sName = %s\n'
1170 ' AND idSchedGroup <> %s\n'
1171 , ( sName, idSchedGroupIgnore, ) );
1172 if self._oDb.getRowCount() > 0:
1173 raise TMRowInUse('Scheduling group name (%s) is already in use.' % (sName,));
1174 return True;
1175
1176 def _readdEntry(self, uidAuthor, oData, tsEffective = None):
1177 """
1178 Re-adds the SchedGroups entry. Used by editEntry and removeEntry.
1179 """
1180 if tsEffective is None:
1181 tsEffective = self._oDb.getCurrentTimestamp();
1182 self._oDb.execute('INSERT INTO SchedGroups (\n'
1183 ' uidAuthor,\n'
1184 ' tsEffective,\n'
1185 ' idSchedGroup,\n'
1186 ' sName,\n'
1187 ' sDescription,\n'
1188 ' fEnabled,\n'
1189 ' enmScheduler,\n'
1190 ' idBuildSrc,\n'
1191 ' idBuildSrcTestSuite,\n'
1192 ' sComment )\n'
1193 'VALUES ( %s, %s, %s, %s, %s, %s, %s, %s, %s, %s )\n'
1194 , ( uidAuthor,
1195 tsEffective,
1196 oData.idSchedGroup,
1197 oData.sName,
1198 oData.sDescription,
1199 oData.fEnabled,
1200 oData.enmScheduler,
1201 oData.idBuildSrc,
1202 oData.idBuildSrcTestSuite,
1203 oData.sComment, ));
1204 return True;
1205
1206 def _historizeEntry(self, idSchedGroup, tsExpire = None):
1207 """
1208 Historizes the current entry for the given scheduling group.
1209 """
1210 if tsExpire is None:
1211 tsExpire = self._oDb.getCurrentTimestamp();
1212 self._oDb.execute('UPDATE SchedGroups\n'
1213 'SET tsExpire = %s\n'
1214 'WHERE idSchedGroup = %s\n'
1215 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
1216 , ( tsExpire, idSchedGroup, ));
1217 return True;
1218
1219 def _addSchedGroupMember(self, uidAuthor, oMember, tsEffective = None):
1220 """
1221 addEntry worker for adding a scheduling group member.
1222 """
1223 if tsEffective is None:
1224 tsEffective = self._oDb.getCurrentTimestamp();
1225 self._oDb.execute('INSERT INTO SchedGroupMembers(\n'
1226 ' idSchedGroup,\n'
1227 ' idTestGroup,\n'
1228 ' tsEffective,\n'
1229 ' uidAuthor,\n'
1230 ' iSchedPriority,\n'
1231 ' bmHourlySchedule,\n'
1232 ' idTestGroupPreReq)\n'
1233 'VALUES (%s, %s, %s, %s, %s, %s, %s)\n'
1234 , ( oMember.idSchedGroup,
1235 oMember.idTestGroup,
1236 tsEffective,
1237 uidAuthor,
1238 oMember.iSchedPriority,
1239 oMember.bmHourlySchedule,
1240 oMember.idTestGroupPreReq, ));
1241 return True;
1242
1243 def _removeSchedGroupMember(self, uidAuthor, oMember):
1244 """
1245 Removes a scheduling group member.
1246 """
1247
1248 # Try record who removed it by adding an dummy entry that expires immediately.
1249 (tsCur, tsCurMinusOne) = self._oDb.getCurrentTimestamps();
1250 if oMember.tsEffective != tsCur and oMember.tsEffective != tsCurMinusOne:
1251 self._historizeSchedGroupMember(oMember, tsCurMinusOne);
1252 self._addSchedGroupMember(uidAuthor, oMember, tsCurMinusOne); # lazy bird.
1253 self._historizeSchedGroupMember(oMember);
1254 else:
1255 self._historizeSchedGroupMember(oMember);
1256 return True;
1257
1258 def _historizeSchedGroupMember(self, oMember, tsExpire = None):
1259 """
1260 Historizes the current entry for the given scheduling group.
1261 """
1262 if tsExpire is None:
1263 tsExpire = self._oDb.getCurrentTimestamp();
1264 self._oDb.execute('UPDATE SchedGroupMembers\n'
1265 'SET tsExpire = %s\n'
1266 'WHERE idSchedGroup = %s\n'
1267 ' AND idTestGroup = %s\n'
1268 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
1269 , ( tsExpire, oMember.idSchedGroup, oMember.idTestGroup, ));
1270 return True;
1271
1272 #
1273 def _addSchedGroupTestBox(self, uidAuthor, oBoxInGroup, tsEffective = None):
1274 """
1275 addEntry worker for adding a test box to a scheduling group.
1276 """
1277 if tsEffective is None:
1278 tsEffective = self._oDb.getCurrentTimestamp();
1279 self._oDb.execute('INSERT INTO TestBoxesInSchedGroups(\n'
1280 ' idSchedGroup,\n'
1281 ' idTestBox,\n'
1282 ' tsEffective,\n'
1283 ' uidAuthor,\n'
1284 ' iSchedPriority)\n'
1285 'VALUES (%s, %s, %s, %s, %s)\n'
1286 , ( oBoxInGroup.idSchedGroup,
1287 oBoxInGroup.idTestBox,
1288 tsEffective,
1289 uidAuthor,
1290 oBoxInGroup.iSchedPriority, ));
1291 return True;
1292
1293 def _removeSchedGroupTestBox(self, uidAuthor, oBoxInGroup):
1294 """
1295 Removes a testbox from a scheduling group.
1296 """
1297
1298 # Try record who removed it by adding an dummy entry that expires immediately.
1299 (tsCur, tsCurMinusOne) = self._oDb.getCurrentTimestamps();
1300 if oBoxInGroup.tsEffective != tsCur and oBoxInGroup.tsEffective != tsCurMinusOne:
1301 self._historizeSchedGroupTestBox(oBoxInGroup, tsCurMinusOne);
1302 self._addSchedGroupTestBox(uidAuthor, oBoxInGroup, tsCurMinusOne); # lazy bird.
1303 self._historizeSchedGroupTestBox(oBoxInGroup);
1304 else:
1305 self._historizeSchedGroupTestBox(oBoxInGroup);
1306 return True;
1307
1308 def _historizeSchedGroupTestBox(self, oBoxInGroup, tsExpire = None):
1309 """
1310 Historizes the current entry for the given scheduling group.
1311 """
1312 if tsExpire is None:
1313 tsExpire = self._oDb.getCurrentTimestamp();
1314 self._oDb.execute('UPDATE TestBoxesInSchedGroups\n'
1315 'SET tsExpire = %s\n'
1316 'WHERE idSchedGroup = %s\n'
1317 ' AND idTestBox = %s\n'
1318 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
1319 , ( tsExpire, oBoxInGroup.idSchedGroup, oBoxInGroup.idTestBox, ));
1320 return True;
1321
1322
1323
1324#
1325# Unit testing.
1326#
1327
1328# pylint: disable=missing-docstring
1329class SchedGroupMemberDataTestCase(ModelDataBaseTestCase):
1330 def setUp(self):
1331 self.aoSamples = [SchedGroupMemberData(),];
1332
1333class SchedGroupDataTestCase(ModelDataBaseTestCase):
1334 def setUp(self):
1335 self.aoSamples = [SchedGroupData(),];
1336
1337if __name__ == '__main__':
1338 unittest.main();
1339 # not reached.
1340
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