VirtualBox

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

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

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 21.7 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: buildsource.py 106061 2024-09-16 14:03:52Z vboxsync $
3
4"""
5Test Manager - Build Sources.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2024 Oracle and/or its affiliates.
11
12This file is part of VirtualBox base platform packages, as
13available from https://www.virtualbox.org.
14
15This program is free software; you can redistribute it and/or
16modify it under the terms of the GNU General Public License
17as published by the Free Software Foundation, in version 3 of the
18License.
19
20This program is distributed in the hope that it will be useful, but
21WITHOUT ANY WARRANTY; without even the implied warranty of
22MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
23General Public License for more details.
24
25You should have received a copy of the GNU General Public License
26along with this program; if not, see <https://www.gnu.org/licenses>.
27
28The contents of this file may alternatively be used under the terms
29of the Common Development and Distribution License Version 1.0
30(CDDL), a copy of it is provided in the "COPYING.CDDL" file included
31in the VirtualBox distribution, in which case the provisions of the
32CDDL are applicable instead of those of the GPL.
33
34You may elect to license modified versions of this file under the
35terms and conditions of either the GPL or the CDDL or both.
36
37SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
38"""
39__version__ = "$Revision: 106061 $"
40
41
42# Standard python imports.
43import unittest;
44
45# Validation Kit imports.
46from common import utils;
47from testmanager.core.base import ModelDataBase, ModelDataBaseTestCase, ModelLogicBase, TMRowAlreadyExists, \
48 TMRowInUse, TMInvalidData, TMRowNotFound;
49from testmanager.core import coreconsts;
50
51
52class BuildSourceData(ModelDataBase):
53 """
54 A build source.
55 """
56
57 ksIdAttr = 'idBuildSrc';
58
59 ksParam_idBuildSrc = 'BuildSource_idBuildSrc';
60 ksParam_tsEffective = 'BuildSource_tsEffective';
61 ksParam_tsExpire = 'BuildSource_tsExpire';
62 ksParam_uidAuthor = 'BuildSource_uidAuthor';
63 ksParam_sName = 'BuildSource_sName';
64 ksParam_sDescription = 'BuildSource_sDescription';
65 ksParam_sProduct = 'BuildSource_sProduct';
66 ksParam_sBranch = 'BuildSource_sBranch';
67 ksParam_asTypes = 'BuildSource_asTypes';
68 ksParam_asOsArches = 'BuildSource_asOsArches';
69 ksParam_iFirstRevision = 'BuildSource_iFirstRevision';
70 ksParam_iLastRevision = 'BuildSource_iLastRevision';
71 ksParam_cSecMaxAge = 'BuildSource_cSecMaxAge';
72
73 kasAllowNullAttributes = [ 'idBuildSrc', 'tsEffective', 'tsExpire', 'uidAuthor', 'sDescription', 'asTypes',
74 'asOsArches', 'iFirstRevision', 'iLastRevision', 'cSecMaxAge' ];
75
76 def __init__(self):
77 ModelDataBase.__init__(self);
78
79 #
80 # Initialize with defaults.
81 # See the database for explanations of each of these fields.
82 #
83 self.idBuildSrc = None;
84 self.tsEffective = None;
85 self.tsExpire = None;
86 self.uidAuthor = None;
87 self.sName = None;
88 self.sDescription = None;
89 self.sProduct = None;
90 self.sBranch = None;
91 self.asTypes = None;
92 self.asOsArches = None;
93 self.iFirstRevision = None;
94 self.iLastRevision = None;
95 self.cSecMaxAge = None;
96
97 def initFromDbRow(self, aoRow):
98 """
99 Re-initializes the object from a SELECT * FROM BuildSources row.
100 Returns self. Raises exception if aoRow is None.
101 """
102 if aoRow is None:
103 raise TMRowNotFound('Build source not found.');
104
105 self.idBuildSrc = aoRow[0];
106 self.tsEffective = aoRow[1];
107 self.tsExpire = aoRow[2];
108 self.uidAuthor = aoRow[3];
109 self.sName = aoRow[4];
110 self.sDescription = aoRow[5];
111 self.sProduct = aoRow[6];
112 self.sBranch = aoRow[7];
113 self.asTypes = aoRow[8];
114 self.asOsArches = aoRow[9];
115 self.iFirstRevision = aoRow[10];
116 self.iLastRevision = aoRow[11];
117 self.cSecMaxAge = aoRow[12];
118 return self;
119
120 def initFromDbWithId(self, oDb, idBuildSrc, tsNow = None, sPeriodBack = None):
121 """
122 Initialize from the database, given the ID of a row.
123 """
124 oDb.execute(self.formatSimpleNowAndPeriodQuery(oDb,
125 'SELECT *\n'
126 'FROM BuildSources\n'
127 'WHERE idBuildSrc = %s\n'
128 , ( idBuildSrc,), tsNow, sPeriodBack));
129 aoRow = oDb.fetchOne()
130 if aoRow is None:
131 raise TMRowNotFound('idBuildSrc=%s not found (tsNow=%s sPeriodBack=%s)' % (idBuildSrc, tsNow, sPeriodBack,));
132 return self.initFromDbRow(aoRow);
133
134 def _validateAndConvertAttribute(self, sAttr, sParam, oValue, aoNilValues, fAllowNull, oDb):
135 # Handle asType and asOsArches specially.
136 if sAttr == 'sType':
137 (oNewValue, sError) = ModelDataBase._validateAndConvertAttribute(self, sAttr, sParam, oValue,
138 aoNilValues, fAllowNull, oDb);
139 if sError is None:
140 if not self.asTypes:
141 oNewValue = None;
142 else:
143 for sType in oNewValue:
144 if len(sType) < 2 or sType.lower() != sType:
145 if sError is None: sError = '';
146 else: sError += ', ';
147 sError += 'invalid value "%s"' % (sType,);
148
149 elif sAttr == 'asOsArches':
150 (oNewValue, sError) = self.validateListOfStr(oValue, aoNilValues = aoNilValues, fAllowNull = fAllowNull,
151 asValidValues = coreconsts.g_kasOsDotCpusAll);
152 if sError is not None and oNewValue is not None:
153 oNewValue = sorted(oNewValue); # Must be sorted!
154
155 elif sAttr == 'cSecMaxAge' and oValue not in aoNilValues: # Allow human readable interval formats.
156 (oNewValue, sError) = utils.parseIntervalSeconds(oValue);
157 else:
158 return ModelDataBase._validateAndConvertAttribute(self, sAttr, sParam, oValue, aoNilValues, fAllowNull, oDb);
159
160 return (oNewValue, sError);
161
162class BuildSourceLogic(ModelLogicBase): # pylint: disable=too-few-public-methods
163 """
164 Build source database logic.
165 """
166
167 def __init__(self, oDb):
168 ModelLogicBase.__init__(self, oDb)
169 self.dCache = None;
170
171 #
172 # Standard methods.
173 #
174
175 def fetchForListing(self, iStart, cMaxRows, tsNow, aiSortColumns = None):
176 """
177 Fetches build sources.
178
179 Returns an array (list) of BuildSourceData items, empty list if none.
180 Raises exception on error.
181 """
182 _ = aiSortColumns;
183
184 if tsNow is None:
185 self._oDb.execute('SELECT *\n'
186 'FROM BuildSources\n'
187 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
188 'ORDER BY idBuildSrc DESC\n'
189 'LIMIT %s OFFSET %s\n'
190 , (cMaxRows, iStart,));
191 else:
192 self._oDb.execute('SELECT *\n'
193 'FROM BuildSources\n'
194 'WHERE tsExpire > %s\n'
195 ' AND tsEffective <= %s\n'
196 'ORDER BY idBuildSrc DESC\n'
197 'LIMIT %s OFFSET %s\n'
198 , (tsNow, tsNow, cMaxRows, iStart,));
199
200 aoRows = []
201 for aoRow in self._oDb.fetchAll():
202 aoRows.append(BuildSourceData().initFromDbRow(aoRow))
203 return aoRows
204
205 def fetchForCombo(self):
206 """Fetch data which is aimed to be passed to HTML form"""
207 self._oDb.execute('SELECT idBuildSrc, sName, sProduct\n'
208 'FROM BuildSources\n'
209 'WHERE tsExpire = \'infinity\'::TIMESTAMP\n'
210 'ORDER BY idBuildSrc DESC\n')
211 asRet = self._oDb.fetchAll();
212 asRet.insert(0, (-1, 'None', 'None'));
213 return asRet;
214
215
216 def addEntry(self, oData, uidAuthor, fCommit = False):
217 """
218 Add a new build source to the database.
219 """
220
221 #
222 # Validate the input.
223 #
224 dErrors = oData.validateAndConvert(self._oDb, oData.ksValidateFor_Add);
225 if dErrors:
226 raise TMInvalidData('addEntry invalid input: %s' % (dErrors,));
227 self._assertUnique(oData, None);
228
229 #
230 # Add it.
231 #
232 self._oDb.execute('INSERT INTO BuildSources (\n'
233 ' uidAuthor,\n'
234 ' sName,\n'
235 ' sDescription,\n'
236 ' sProduct,\n'
237 ' sBranch,\n'
238 ' asTypes,\n'
239 ' asOsArches,\n'
240 ' iFirstRevision,\n'
241 ' iLastRevision,\n'
242 ' cSecMaxAge)\n'
243 'VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s, %s)\n'
244 , ( uidAuthor,
245 oData.sName,
246 oData.sDescription,
247 oData.sProduct,
248 oData.sBranch,
249 oData.asTypes,
250 oData.asOsArches,
251 oData.iFirstRevision,
252 oData.iLastRevision,
253 oData.cSecMaxAge, ));
254
255 self._oDb.maybeCommit(fCommit);
256 return True;
257
258 def editEntry(self, oData, uidAuthor, fCommit = False):
259 """
260 Modifies a build source.
261 """
262
263 #
264 # Validate the input and read the old entry.
265 #
266 dErrors = oData.validateAndConvert(self._oDb, oData.ksValidateFor_Edit);
267 if dErrors:
268 raise TMInvalidData('addEntry invalid input: %s' % (dErrors,));
269 self._assertUnique(oData, oData.idBuildSrc);
270 oOldData = BuildSourceData().initFromDbWithId(self._oDb, oData.idBuildSrc);
271
272 #
273 # Make the changes (if something actually changed).
274 #
275 if not oData.isEqualEx(oOldData, [ 'tsEffective', 'tsExpire', 'uidAuthor', ]):
276 self._historizeBuildSource(oData.idBuildSrc);
277 self._oDb.execute('INSERT INTO BuildSources (\n'
278 ' uidAuthor,\n'
279 ' idBuildSrc,\n'
280 ' sName,\n'
281 ' sDescription,\n'
282 ' sProduct,\n'
283 ' sBranch,\n'
284 ' asTypes,\n'
285 ' asOsArches,\n'
286 ' iFirstRevision,\n'
287 ' iLastRevision,\n'
288 ' cSecMaxAge)\n'
289 'VALUES (%s, %s, %s, %s, %s, %s, %s, %s, %s, %s, %s)\n'
290 , ( uidAuthor,
291 oData.idBuildSrc,
292 oData.sName,
293 oData.sDescription,
294 oData.sProduct,
295 oData.sBranch,
296 oData.asTypes,
297 oData.asOsArches,
298 oData.iFirstRevision,
299 oData.iLastRevision,
300 oData.cSecMaxAge, ));
301 self._oDb.maybeCommit(fCommit);
302 return True;
303
304 def removeEntry(self, uidAuthor, idBuildSrc, fCascade = False, fCommit = False):
305 """
306 Deletes a build sources.
307 """
308
309 #
310 # Check cascading.
311 #
312 if fCascade is not True:
313 self._oDb.execute('SELECT idSchedGroup, sName\n'
314 'FROM SchedGroups\n'
315 'WHERE idBuildSrc = %s\n'
316 ' OR idBuildSrcTestSuite = %s\n'
317 , (idBuildSrc, idBuildSrc,));
318 if self._oDb.getRowCount() > 0:
319 asGroups = [];
320 for aoRow in self._oDb.fetchAll():
321 asGroups.append('%s (#%d)' % (aoRow[1], aoRow[0]));
322 raise TMRowInUse('Build source #%d is used by one or more scheduling groups: %s'
323 % (idBuildSrc, ', '.join(asGroups),));
324 else:
325 self._oDb.execute('UPDATE SchedGroups\n'
326 'SET idBuildSrc = NULL\n'
327 'WHERE idBuildSrc = %s'
328 , ( idBuildSrc,));
329 self._oDb.execute('UPDATE SchedGroups\n'
330 'SET idBuildSrcTestSuite = NULL\n'
331 'WHERE idBuildSrcTestSuite = %s'
332 , ( idBuildSrc,));
333
334 #
335 # Do the job.
336 #
337 self._historizeBuildSource(idBuildSrc, None);
338 _ = uidAuthor; ## @todo record deleter.
339
340 self._oDb.maybeCommit(fCommit);
341 return True;
342
343 def cachedLookup(self, idBuildSrc):
344 """
345 Looks up the most recent BuildSourceData object for idBuildSrc
346 via an object cache.
347
348 Returns a shared BuildSourceData object. None if not found.
349 Raises exception on DB error.
350 """
351 if self.dCache is None:
352 self.dCache = self._oDb.getCache('BuildSourceData');
353 oEntry = self.dCache.get(idBuildSrc, None);
354 if oEntry is None:
355 self._oDb.execute('SELECT *\n'
356 'FROM BuildSources\n'
357 'WHERE idBuildSrc = %s\n'
358 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
359 , (idBuildSrc, ));
360 if self._oDb.getRowCount() == 0:
361 # Maybe it was deleted, try get the last entry.
362 self._oDb.execute('SELECT *\n'
363 'FROM BuildSources\n'
364 'WHERE idBuildSrc = %s\n'
365 'ORDER BY tsExpire DESC\n'
366 'LIMIT 1\n'
367 , (idBuildSrc, ));
368 elif self._oDb.getRowCount() > 1:
369 raise self._oDb.integrityException('%s infinity rows for %s' % (self._oDb.getRowCount(), idBuildSrc));
370
371 if self._oDb.getRowCount() == 1:
372 aaoRow = self._oDb.fetchOne();
373 oEntry = BuildSourceData();
374 oEntry.initFromDbRow(aaoRow);
375 self.dCache[idBuildSrc] = oEntry;
376 return oEntry;
377
378 #
379 # Other methods.
380 #
381
382 def openBuildCursor(self, oBuildSource, sOs, sCpuArch, tsNow):
383 """
384 Opens a cursor (SELECT) using the criteria found in the build source
385 and the given OS.CPUARCH.
386
387 Returns database cursor. May raise exception on bad input or logic error.
388
389 Used by SchedulerBase.
390 """
391
392 oCursor = self._oDb.openCursor();
393
394 #
395 # Construct the extra conditionals.
396 #
397 sExtraConditions = '';
398
399 # Types
400 if oBuildSource.asTypes is not None and oBuildSource.asTypes:
401 if len(oBuildSource.asTypes) == 1:
402 sExtraConditions += oCursor.formatBindArgs(' AND BuildCategories.sType = %s', (oBuildSource.asTypes[0],));
403 else:
404 sExtraConditions += oCursor.formatBindArgs(' AND BuildCategories.sType IN (%s', (oBuildSource.asTypes[0],))
405 for i in range(1, len(oBuildSource.asTypes) - 1):
406 sExtraConditions += oCursor.formatBindArgs(', %s', (oBuildSource.asTypes[i],));
407 sExtraConditions += oCursor.formatBindArgs(', %s)\n', (oBuildSource.asTypes[-1],));
408
409 # BuildSource OSes.ARCHes. (Paranoia: use a dictionary to avoid duplicate values.)
410 if oBuildSource.asOsArches is not None and oBuildSource.asOsArches:
411 sExtraConditions += oCursor.formatBindArgs(' AND BuildCategories.asOsArches && %s', (oBuildSource.asOsArches,));
412
413 # TestBox OSes.ARCHes. (Paranoia: use a dictionary to avoid duplicate values.)
414 dOsDotArches = {};
415 dOsDotArches[sOs + '.' + sCpuArch] = 1;
416 dOsDotArches[sOs + '.' + coreconsts.g_ksCpuArchAgnostic] = 1;
417 dOsDotArches[coreconsts.g_ksOsAgnostic + '.' + sCpuArch] = 1;
418 dOsDotArches[coreconsts.g_ksOsDotArchAgnostic] = 1;
419 sExtraConditions += oCursor.formatBindArgs(' AND BuildCategories.asOsArches && %s', (list(dOsDotArches.keys()),));
420
421 # Revision range.
422 if oBuildSource.iFirstRevision is not None:
423 sExtraConditions += oCursor.formatBindArgs(' AND Builds.iRevision >= %s\n', (oBuildSource.iFirstRevision,));
424 if oBuildSource.iLastRevision is not None:
425 sExtraConditions += oCursor.formatBindArgs(' AND Builds.iRevision <= %s\n', (oBuildSource.iLastRevision,));
426
427 # Max age.
428 if oBuildSource.cSecMaxAge is not None:
429 sExtraConditions += oCursor.formatBindArgs(' AND Builds.tsCreated >= (%s - \'%s seconds\'::INTERVAL)\n',
430 (tsNow, oBuildSource.cSecMaxAge,));
431
432 #
433 # Execute the query.
434 #
435 oCursor.execute('SELECT Builds.*, BuildCategories.*,\n'
436 ' EXISTS( SELECT tsExpire\n'
437 ' FROM BuildBlacklist\n'
438 ' WHERE BuildBlacklist.tsExpire = \'infinity\'::TIMESTAMP\n'
439 ' AND BuildBlacklist.sProduct = %s\n'
440 ' AND BuildBlacklist.sBranch = %s\n'
441 ' AND BuildBlacklist.iFirstRevision <= Builds.iRevision\n'
442 ' AND BuildBlacklist.iLastRevision >= Builds.iRevision ) AS fMaybeBlacklisted\n'
443 'FROM Builds, BuildCategories\n'
444 'WHERE Builds.idBuildCategory = BuildCategories.idBuildCategory\n'
445 ' AND Builds.tsExpire = \'infinity\'::TIMESTAMP\n'
446 ' AND Builds.tsEffective <= %s\n'
447 ' AND Builds.fBinariesDeleted is FALSE\n'
448 ' AND BuildCategories.sProduct = %s\n'
449 ' AND BuildCategories.sBranch = %s\n'
450 + sExtraConditions +
451 'ORDER BY Builds.idBuild DESC\n'
452 'LIMIT 256\n'
453 , ( oBuildSource.sProduct, oBuildSource.sBranch,
454 tsNow, oBuildSource.sProduct, oBuildSource.sBranch,));
455
456 return oCursor;
457
458
459 def getById(self, idBuildSrc):
460 """Get Build Source data by idBuildSrc"""
461
462 self._oDb.execute('SELECT *\n'
463 'FROM BuildSources\n'
464 'WHERE tsExpire = \'infinity\'::timestamp\n'
465 ' AND idBuildSrc = %s;', (idBuildSrc,))
466 aRows = self._oDb.fetchAll()
467 if len(aRows) not in (0, 1):
468 raise self._oDb.integrityException(
469 'Found more than one build sources with the same credentials. Database structure is corrupted.')
470 try:
471 return BuildSourceData().initFromDbRow(aRows[0])
472 except IndexError:
473 return None
474
475 #
476 # Internal helpers.
477 #
478
479 def _assertUnique(self, oData, idBuildSrcIgnore):
480 """ Checks that the build source name is unique, raises exception if it isn't. """
481 self._oDb.execute('SELECT idBuildSrc\n'
482 'FROM BuildSources\n'
483 'WHERE sName = %s\n'
484 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
485 + ('' if idBuildSrcIgnore is None else ' AND idBuildSrc <> %d\n' % (idBuildSrcIgnore,))
486 , ( oData.sName, ))
487 if self._oDb.getRowCount() > 0:
488 raise TMRowAlreadyExists('A build source with name "%s" already exist.' % (oData.sName,));
489 return True;
490
491
492 def _historizeBuildSource(self, idBuildSrc, tsExpire = None):
493 """ Historizes the current build source entry. """
494 if tsExpire is None:
495 self._oDb.execute('UPDATE BuildSources\n'
496 'SET tsExpire = CURRENT_TIMESTAMP\n'
497 'WHERE idBuildSrc = %s\n'
498 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
499 , ( idBuildSrc, ));
500 else:
501 self._oDb.execute('UPDATE BuildSources\n'
502 'SET tsExpire = %s\n'
503 'WHERE idBuildSrc = %s\n'
504 ' AND tsExpire = \'infinity\'::TIMESTAMP\n'
505 , ( tsExpire, idBuildSrc, ));
506 return True;
507
508
509
510
511
512#
513# Unit testing.
514#
515
516# pylint: disable=missing-docstring
517class BuildSourceDataTestCase(ModelDataBaseTestCase):
518 def setUp(self):
519 self.aoSamples = [BuildSourceData(),];
520
521if __name__ == '__main__':
522 unittest.main();
523 # not reached.
524
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