-
Notifications
You must be signed in to change notification settings - Fork 84
/
Copy pathPatchOrderGenerator.py
executable file
·781 lines (741 loc) · 34 KB
/
PatchOrderGenerator.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
#!/usr/bin/env python
# Generate Patch Order Sequence
#
# python PatchOrderGenerator.py <path_to_patches_dir>
#
# This script reads all the Patch files(*.KID/*.KIDs)
# and info file (*.TXT(s)/*,txt) under the input directory recursively
# and generate the Patch order via patch dependency
#
#---------------------------------------------------------------------------
# Copyright 2012-2019 The Open Source Electronic Health Record Alliance
# Copyright 2024 Sam Habiel. Python3.12 changes.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#---------------------------------------------------------------------------
from __future__ import print_function
from past.builtins import cmp
from builtins import str
from builtins import range
from builtins import object
import os
import sys
import re
import glob
import csv
from datetime import datetime
# append this module in the sys.path at run time
curDir = os.path.dirname(os.path.abspath(__file__))
if curDir not in sys.path:
sys.path.append(curDir)
PATCH_IGNORED_DIRS = ('Packages', 'Uncategorized', 'MultiBuilds')
VALID_CSV_ORDER_FILE_FIELDS = [
'INSTALLED', 'VERIFY_DT', 'STATUS', 'SEQ#',
'LABELED_AS', 'CATEGORY', 'PRODUCT_NAME'
]
""" enums """
KIDS_BUILD_FILE_TYPE_KIDS = 0
KIDS_BUILD_FILE_TYPE_HEADER = 1
KIDS_BUILD_FILE_TYPE_SHA1 = 2
from LoggerManager import logger, initConsoleLogging
from KIDSBuildParser import KIDSBuildParser
from PatchInfoParser import PatchInfoParser
from PatchInfoParser import convertToInstallName
from PatchInfoParser import dirNameToInstallName, PatchInfo
from PatchInfoParser import setPatchInfoFromInstallName
from ConvertToExternalData import readSha1SumFromSha1File
from ConvertToExternalData import isValidKIDSBuildSuffix
from ConvertToExternalData import isValidKIDSBuildHeaderSuffix
from ConvertToExternalData import isValidKIDSBuildSha1Suffix
from ConvertToExternalData import isValidPatchInfoSuffix
from ConvertToExternalData import isValidPatchInfoSha1Suffix
from ConvertToExternalData import isValidCSVSuffix
from ConvertToExternalData import isValidPatchRelatedFiles
from ConvertToExternalData import isValidGlobalFileSuffix
from ConvertToExternalData import isValidGlobalSha1Suffix
from ConvertToExternalData import isValidPythonSuffix
from KIDSAssociatedFilesMapping import getAssociatedInstallName
class CycleException(Exception):
pass
"""
This class will generate a Patch order based on input
patch directory
"""
class PatchOrderGenerator(object):
def __init__(self):
self._kidsInstallNameDict = dict() # the install name -> kids files
self._kidsDepBuildDict = dict() # install name -> [dependency build]
self._multiBuildDict = dict() # kids file -> [install names]
self._kidsBuildFileDict = dict() # all the kids files name->[path,sha1path]
self._kidsInstallNameSha1Dict = dict() # install name -> sha1
self._kidsInfoFileList = [] # all kids info file under vista patches dir
self._csvOrderFileList = [] # all csv order file under vista patches dir
self._globalFilesSet = set() # all global file under vista patches dir
self._patchInfoDict = dict() #install name -> patchInfo
self._missKidsBuildDict = dict() # install name -> patchInfo without Kids
self._missKidsInfoSet = set() # kids build without info file
self._patchOrderCSVDict = dict() # csv File -> list of patches in order
self._patchOrder = [] # list of install name in order
self._informationalKidsSet = set() # a list of kids that are informational
self._notInstalledKidsSet = set() # a list of kids that are not installed
self._patchDependencyDict = dict() # the dependency dict of all patches
self._invalidInfoFileSet = set() # invalid txt file
self._csvDepDict = dict() # installName => installName based on csvFile
self._installNameSeqMap = dict() # installName => seqNo, patch in order
self._pythonScriptList = [] # all the python script files
def generatePatchOrder(self, patchReposDir, installName=None):
return self.generatePatchOrderTopologic(patchReposDir, installName)
""" generate a patch seqence order by topologic sort """
def generatePatchOrderTopologic(self, patchDir, installName=None):
self.analyzeVistAPatchDir(patchDir)
if installName:
if installName not in self._patchInfoDict:
raise Exception("Could not find patch for %s" % installName)
self.__updatePatchDependency__((installName is None))
self.__generatePatchDependencyGraph__()
try:
self.__topologicSort__(installName)
except CycleException as e:
errorMessage = "Failed to sort patches: %s" % e
logger.error(errorMessage)
return []
logger.info("After topologic sort %d" % len(self._patchOrder))
return self._patchOrder
""" analyze VistA patch Dir generate data structure """
def analyzeVistAPatchDir(self, patchDir):
assert os.path.exists(patchDir)
self.__getAllKIDSBuildInfoAndOtherFileList__(patchDir)
self.__parseAllKIDSBuildFilesList__()
self.__parseAllKIDSInfoFilesList__()
self.__generateMissKIDSInfoSet__()
self.__addMissKIDSInfoPatch__()
self.__handlePatchAssociatedFiles__()
self.__updateCustomInstaller__()
self.__updateMultiBuildPatchInfo__()
self.__getPatchOrderDependencyByCSVFiles__()
""" Some getter function to return result """
""" @ return all patchInfoDict install => patchInfo"""
def getPatchInfoDict(self):
return self._patchInfoDict
""" @return all invalid KIDS info files set"""
def getInvalidInfoFiles(self):
return self._invalidInfoFileSet
""" @return Info file without a KIDS build as install => patchInfo"""
def getNoKidsBuildInfoDict(self):
return self._missKidsBuildDict
""" print the final order list """
def printPatchOrderList(self):
printPatchOrderList(self._patchOrder)
def __addKidsBuildFileToDict__(self, fileName, absPath, fileType):
if fileName not in self._kidsBuildFileDict:
self._kidsBuildFileDict[fileName] = [None, None]
if ( fileType == KIDS_BUILD_FILE_TYPE_KIDS or
fileType == KIDS_BUILD_FILE_TYPE_HEADER ):
filePath = self._kidsBuildFileDict[fileName][0]
if self._kidsBuildFileDict[fileName][0] != None:
logger.error("Duplicated KIDS file path %s : %s" % (filePath, absPath))
else:
self._kidsBuildFileDict[fileName][0] = absPath
return
if fileType == KIDS_BUILD_FILE_TYPE_SHA1:
sha1File = self._kidsBuildFileDict[fileName][1]
if self._kidsBuildFileDict[fileName][1] != None:
logger.error("Duplicated KIDS Sha1 File %s : %s" % (sha1File, absPath))
else:
self._kidsBuildFileDict[fileName][1] = absPath
""" walk through the dir to find all KIDS build and info file and others """
def __getAllKIDSBuildInfoAndOtherFileList__(self, patchDir):
assert os.path.exists(patchDir)
absPatchDir = os.path.abspath(patchDir)
for (root, dirs, files) in os.walk(absPatchDir):
lastDir = os.path.split(root)[-1]
for fileName in files:
absFilename = os.path.join(root, fileName)
if not isValidPatchRelatedFiles(absFilename, True):
continue
""" Handle KIDS build files """
if isValidKIDSBuildSuffix(fileName):
logger.debug("Adding %s KIDS file to dict" % absFilename)
self.__addKidsBuildFileToDict__(fileName, absFilename,
KIDS_BUILD_FILE_TYPE_KIDS)
continue
""" Handle KIDS build HEADER files """
if isValidKIDSBuildHeaderSuffix(fileName):
logger.debug("Adding %s KIDS header to dict" % absFilename)
kidsFileName = fileName[0:fileName.rfind('.')]
self.__addKidsBuildFileToDict__(kidsFileName, absFilename,
KIDS_BUILD_FILE_TYPE_HEADER)
continue
""" Handle KIDS build Sha1 files """
if isValidKIDSBuildSha1Suffix(fileName):
logger.debug("Adding %s KIDS info to dict" % absFilename)
kidsFileName = fileName[0:fileName.rfind('.')]
self.__addKidsBuildFileToDict__(kidsFileName, absFilename,
KIDS_BUILD_FILE_TYPE_SHA1)
continue
""" Handle KIDS Info/Sha1 files """
if ( isValidPatchInfoSuffix(fileName) or
isValidPatchInfoSha1Suffix(fileName) ):
self._kidsInfoFileList.append(absFilename)
continue
""" Handle Global/Sha1 Files """
if ( isValidGlobalFileSuffix(fileName) or
isValidGlobalSha1Suffix(fileName) ):
logger.debug("Adding %s Global files to list" % absFilename)
self._globalFilesSet.add(absFilename)
continue
""" handle all csv files """
if isValidCSVSuffix(fileName):
if isValidOrderCSVFile(absFilename):
self._csvOrderFileList.append(absFilename)
continue
""" Handle .py files """
if isValidPythonSuffix(fileName):
logger.debug("Adding %s python script to list" % absFilename)
self._pythonScriptList.append(absFilename)
continue
logger.info("Total # of KIDS Builds are %d" % len(self._kidsBuildFileDict))
logger.info("Total # of KIDS Info are %d" % len(self._kidsInfoFileList))
logger.info("Total # of Global files are %d" % len(self._globalFilesSet))
logger.info("Total # of Python files are %d" % len(self._pythonScriptList))
logger.info("Total # of CSV files are %d" % len(self._csvOrderFileList))
""" parse all the KIDS files, update kidsInstallNameDict, multibuildDict """
def __parseAllKIDSBuildFilesList__(self):
for basename in self._kidsBuildFileDict:
kidsFile, sha1Path = self._kidsBuildFileDict[basename]
if kidsFile == None:
logger.error("No KIDS file available for name %s" % basename)
continue
installNameList, seqNo, kidsBuilds = None, None, None
if isValidKIDSBuildHeaderSuffix(kidsFile):
from KIDSBuildParser import loadMetaDataFromJSON
#continue
installNameList, seqNo, kidsBuilds = loadMetaDataFromJSON(kidsFile)
else:
kidsParser = KIDSBuildParser(None)
kidsParser.unregisterSectionHandler(KIDSBuildParser.ROUTINE_SECTION)
kidsParser.parseKIDSBuild(kidsFile)
installNameList = kidsParser.installNameList
logger.debug("install name list is %s" % installNameList)
seqNo = kidsParser.seqNo
kidsBuilds = kidsParser.kidsBuilds
if len(installNameList) > 1:
if not self._multiBuildDict.get(kidsFile):
self._multiBuildDict[kidsFile] = installNameList
else:
assert self._multiBuildDict[kidsFile] == installNameList
elif seqNo:
if installNameList[0] not in self._installNameSeqMap:
self._installNameSeqMap[installNameList[0]] = seqNo
else:
logger.error("Duplicated KIDS build file %s" % kidsFile)
for installName in installNameList:
if installName in self._kidsInstallNameDict:
logger.warning("%s is already in the dict %s" % (installName, kidsFile))
logger.debug("Added installName %s, file %s" % (installName, kidsFile))
self._kidsInstallNameDict[installName] = os.path.normpath(kidsFile)
""" handle KIDS sha1 file Path """
if sha1Path:
if installName in self._kidsInstallNameSha1Dict:
logger.warning("%s is already in the dict %s" % (installName, sha1Path))
self._kidsInstallNameSha1Dict[installName] = sha1Path
""" update kids dependency """
if installName in self._kidsDepBuildDict:
logger.warning("%s already has the dep map %s" %
(installName, self._kidsDepBuildDict[installName]))
if kidsBuilds:
for kidsBuild in kidsBuilds:
if kidsBuild.installName == installName:
depList = kidsBuild.dependencyList
if depList:
self._kidsDepBuildDict[installName] = set([x[0] for x in
depList])
logger.info("%s: %s" % (installName,
self._kidsDepBuildDict[installName]))
logger.debug("%s" % sorted(self._kidsInstallNameDict.keys()))
logger.info("Total # of install name %d" % len(self._kidsInstallNameDict))
""" parse all the KIDS info files, update patchInfoDict, missKidsBuildDict"""
def __parseAllKIDSInfoFilesList__(self):
kidsParser = PatchInfoParser()
for kidsInfoFile in self._kidsInfoFileList:
patchInfo = kidsParser.parseKIDSInfoFile(kidsInfoFile)
if not patchInfo:
logger.debug("invalid kids info file %s" % kidsInfoFile)
self._invalidInfoFileSet.add(kidsInfoFile)
continue
""" only add to list for info that is related to a Patch"""
installName = patchInfo.installName
if installName not in self._kidsInstallNameDict:
logger.warning("no KIDS file related to %s (%s)" % (installName, kidsInfoFile))
if installName in self._missKidsBuildDict:
logger.warning("duplicated kids install name")
if kidsInfoFile != self._missKidsBuildDict[installName].kidsInfoPath:
logger.warning("duplicated kids info file name %s" % kidsInfoFile)
continue
self._missKidsBuildDict[installName] = patchInfo
continue
patchInfo.kidsFilePath = self._kidsInstallNameDict[installName]
assert patchInfo.kidsFilePath
""" update PatchInfo kidsSha1 and kidsSha1Path """
if installName in self._kidsInstallNameSha1Dict:
sha1Path = self._kidsInstallNameSha1Dict[installName]
patchInfo.kidsSha1Path = sha1Path
patchInfo.kidsSha1 = readSha1SumFromSha1File(sha1Path)
if installName in self._patchInfoDict:
logger.warning("duplicated installName %s, %s, %s" %
(installName, self._patchInfoDict[installName],
kidsInfoFile))
""" merge the dependency if needed, also
put extra dependency into optional set """
if installName in self._kidsDepBuildDict:
infoDepSet = set()
kidsDepSet = set()
if patchInfo.depKIDSBuild:
infoDepSet = patchInfo.depKIDSBuild
if self._kidsDepBuildDict[installName]:
kidsDepSet = self._kidsDepBuildDict[installName]
diffSet = kidsDepSet ^ infoDepSet
if len(diffSet):
logger.info("Merging kids dependencies %s" % installName)
logger.debug("kids build set is %s" % kidsDepSet)
logger.debug("info build set is %s" % infoDepSet)
logger.warning("difference set: %s" % diffSet)
patchInfo.depKIDSBuild = infoDepSet | kidsDepSet
patchInfo.optionalDepSet = infoDepSet - kidsDepSet
else:
patchInfo.depKIDSBuild = infoDepSet
self._patchInfoDict[installName] = patchInfo
""" update multiBuild KIDS patch info"""
def __updateMultiBuildPatchInfo__(self):
patchList = self._patchInfoDict
for installList in self._multiBuildDict.values():
for installName in installList:
patchInfo = patchList[installName]
patchInfo.isMultiBuilds = True
patchInfo.multiBuildsList = installList
""" update multiBuild KIDS files dependencies """
def __updateMultiBuildDependencies__(self):
patchList = self._patchInfoDict
for installList in self._multiBuildDict.values():
logger.info("Multi-Buids KIDS install List: %s" % (installList))
firstPatch = patchList[installList[0]]
firstPatch.otherKidsInfoList = []
if firstPatch.csvDepPatch is None:
"""
If primary build install name is not specified in the csv file
will fall back to use dependency specified in the first
secondary build
"""
secondPatch = patchList[installList[1]]
if secondPatch.csvDepPatch != firstPatch.installName:
logger.info("Assign first patch CSV Dep %s" % firstPatch.installName)
firstPatch.csvDepPatch = secondPatch.csvDepPatch
for index in range(1,len(installList)):
nextPatchInfo = patchList[installList[index]]
""" just to make sure the first one has all the dependencies """
firstPatch.depKIDSBuild.update(nextPatchInfo.depKIDSBuild)
firstPatch.optionalDepSet.update(nextPatchInfo.optionalDepSet)
firstPatch.otherKidsInfoList.append([nextPatchInfo.kidsInfoPath,
nextPatchInfo.kidsInfoSha1])
prevInstallName = installList[index - 1]
# Removing May 2019: Causes cyclical dependency - J.Snyder
#if prevInstallName not in nextPatchInfo.depKIDSBuild:
# nextPatchInfo.depKIDSBuild.add(prevInstallName)
#del patchList[installList[index]] #remove the other patch from the list
logger.debug("%s:%s" % (nextPatchInfo.installName, nextPatchInfo.depKIDSBuild))
""" remove the self dependencies of the first patch """
firstPatch.depKIDSBuild.difference_update(installList)
logger.debug("%s:%s" % (firstPatch.installName, firstPatch.depKIDSBuild))
""" update the csvDepPatch based on csv file based dependencies """
def __updateCSVDependencies__(self):
for patchInfo in self._patchInfoDict.values():
installName = patchInfo.installName
if installName in self._csvDepDict:
patchInfo.csvDepPatch = self._csvDepDict[installName]
def __updatePatchDependency__(self, updCSVDep=True):
if updCSVDep:
""" update the dependencies based on csv files """
self.__updateCSVDependencies__()
""" update the dependencies based on patch Sequenece # """
self.__updateSeqNoDependencies__()
""" update the dependencies for multi-build KIDS files """
self.__updateMultiBuildDependencies__()
def __updateSeqNoDependencies__(self):
namespaceVerSeq = dict()
patchInfoDict = self._patchInfoDict
for patchInfo in patchInfoDict.values():
""" generate dependencies map based on seq # """
namespace = patchInfo.namespace
version = patchInfo.version
seqNo = patchInfo.seqNo
installName = patchInfo.installName
if namespace and version:
if not seqNo:
continue
if namespace not in namespaceVerSeq:
namespaceVerSeq[namespace] = dict()
if version not in namespaceVerSeq[namespace]:
namespaceVerSeq[namespace][version] = []
namespaceVerSeq[namespace][version].append((int(seqNo),
installName))
""" add dependencies based on SEQ # """
for versionDict in namespaceVerSeq.values():
for seqList in versionDict.values():
if len(seqList) < 2:
continue
else:
# sorted list by sequence #
seqOrder = sorted(seqList, key=lambda item: item[0])
for idx in range(len(seqOrder)-1,0,-1):
installName = seqOrder[idx][1]
patchInfoDict[installName].depKIDSBuild.add(seqOrder[idx-1][1])
""" now generate the dependency graph """
def __generatePatchDependencyGraph__(self):
depDict = self._patchDependencyDict
namespaceVerSeq = dict()
for patchInfo in self._patchInfoDict.values():
installName = patchInfo.installName
if installName not in depDict:
depDict[installName] = set()
if patchInfo.depKIDSBuild:
depDict[installName].update(patchInfo.depKIDSBuild)
""" combine csv dependencies """
if patchInfo.csvDepPatch:
if installName not in depDict:
depDict[installName] = set()
if patchInfo.csvDepPatch in self._patchInfoDict:
depDict[installName].add(patchInfo.csvDepPatch)
""" generate self._missKidsInfoSet """
def __generateMissKIDSInfoSet__(self):
patchInstallNameSet = set(x for x in self._patchInfoDict)
kidsInstallNameSet = set(self._kidsInstallNameDict.keys())
self._missKidsInfoSet = kidsInstallNameSet.difference(patchInstallNameSet)
logger.info("Missing KIDS Info set %s" % self._missKidsInfoSet)
""" add missing info Patch """
def __addMissKIDSInfoPatch__(self):
for kidsInstallName in self._missKidsInfoSet:
logger.debug("Installation Name: %s, does not have info file, %s" %
(kidsInstallName, self._kidsInstallNameDict[kidsInstallName]))
patchInfo = PatchInfo()
patchInfo.installName = kidsInstallName
setPatchInfoFromInstallName(kidsInstallName, patchInfo)
if kidsInstallName in self._kidsInstallNameSha1Dict:
sha1Path = self._kidsInstallNameSha1Dict[kidsInstallName]
patchInfo.kidsSha1Path = sha1Path
patchInfo.kidsSha1 = readSha1SumFromSha1File(sha1Path)
if kidsInstallName in self._installNameSeqMap:
patchInfo.seqNo = self._installNameSeqMap[kidsInstallName]
patchInfo.kidsFilePath = self._kidsInstallNameDict[kidsInstallName]
if kidsInstallName in self._kidsDepBuildDict:
logger.info("update the Missing Info KIDS depencency %s" %
kidsInstallName)
patchInfo.depKIDSBuild = self._kidsDepBuildDict[kidsInstallName]
self._patchInfoDict[kidsInstallName] = patchInfo
""" update the associated files for patchInfo """
def __handlePatchAssociatedFiles__(self):
""" handle the info files first """
""" first by name assiciation """
patchInfoList = list(self._patchInfoDict.values())
#handle the associated files for missingKIDSBuild info
patchInfoList.extend(list(self._missKidsBuildDict.values()))
for patchInfo in patchInfoList:
infoPath = patchInfo.kidsInfoPath
if infoPath:
infoName = os.path.basename(infoPath)
associateSet = set()
for infoFile in self._invalidInfoFileSet:
infoFileName = os.path.basename(infoFile)
if infoFileName.startswith(infoName[:infoName.rfind('.')]):
patchInfo.addToAssociatedInfoList(infoFile)
associateSet.add(infoFile)
continue
self._invalidInfoFileSet.difference_update(associateSet)
""" second by mapping association """
associateSet = set()
for infoFile in self._invalidInfoFileSet:
installName = getAssociatedInstallName(infoFile)
if installName:
if installName in self._patchInfoDict:
patchInfo = self._patchInfoDict[installName]
#handle the associated files for missingKIDSBuild info
elif installName in self._missKidsBuildDict:
patchInfo = self._missKidsBuildDict[installName]
else:
continue
patchInfo.addToAssociatedInfoList(infoFile)
associateSet.add(infoFile)
self._invalidInfoFileSet.difference_update(associateSet)
""" handle global files """
associateSet = set()
for globalFile in self._globalFilesSet:
installName = getAssociatedInstallName(globalFile)
if installName and installName in self._patchInfoDict:
patchInfo = self._patchInfoDict[installName]
patchInfo.addToAssociatedGlobalList(globalFile)
associateSet.add(globalFile)
self._globalFilesSet.difference_update(associateSet)
logger.info("Total # of leftover info files: %s" %
len(self._invalidInfoFileSet))
logger.debug(self._invalidInfoFileSet)
logger.info("Total # of leftover global files: %s" %
len(self._globalFilesSet))
logger.debug(self._globalFilesSet)
""" update PatchInfo custom installer """
def __updateCustomInstaller__(self):
for pythonScript in self._pythonScriptList:
installName = os.path.basename(pythonScript)
installName = dirNameToInstallName(installName[:installName.rfind('.')])
if installName in self._patchInfoDict:
patchInfo = self._patchInfoDict[installName]
patchInfo.hasCustomInstaller = True
if patchInfo.customInstallerPath:
logger.warning("Duplicated installer for %s: [%s:%s]" % (
installName, patchInfo.customInstallerPath,
pythonScript))
logger.info("%s: custom installer %s" % (pythonScript, installName))
self._patchInfoDict[installName].customInstallerPath = pythonScript
""" get all the patch order dependency by csv files """
def __getPatchOrderDependencyByCSVFiles__(self):
for csvOrderFile in self._csvOrderFileList:
self.__getPatchOrderListByCSV__(csvOrderFile)
self.__buildPatchOrderDependencyByCSV__(csvOrderFile)
sortedPatchList = self.__sortCSVDependencyList__()
""" build csvDepDict based on csv File """
def __buildPatchOrderDependencyByCSV__(self, orderCSV):
patchOrderList = self._patchOrderCSVDict[orderCSV]
if patchOrderList is None: return
""" some sanity check """
outPatchList = []
multiBuildSet = set()
for patchOrder in patchOrderList:
installName = patchOrder[0]
if installName not in self._patchInfoDict:
if (installName not in self._informationalKidsSet and
installName not in self._kidsInstallNameDict):
logger.warning("No KIDS file found for %s" % str(patchOrder))
continue
patchInfo = self._patchInfoDict[installName]
patchInfo.verifiedDate = patchOrder[2]
""" check the seq no """
seqNo = patchOrder[1]
if len(seqNo) > 0:
""" check the seq no match the parsing result """
if patchInfo.seqNo is not None:
if int(seqNo) != int(patchInfo.seqNo):
logger.error("SeqNo mismatch for %s, from csv: %s, info %s" %
(installName, seqNo, patchInfo.seqNo))
else:
logger.info("Add seqNo %s for %s" % (seqNo, installName))
patchInfo.seqNo = seqNo
""" handle the multi-build patch """
if patchInfo.installName in multiBuildSet:
logger.info("%s is already part of the multiBuild" % installName)
continue
if patchInfo.isMultiBuilds:
patchList = [self._patchInfoDict[x] for x in patchInfo.multiBuildsList]
for patchInfo in patchList:
patchInfo.verifiedDate = patchOrder[2]
outPatchList.extend(patchList)
multiBuildSet.update(patchInfo.multiBuildsList)
else:
outPatchList.append(patchInfo)
""" update the order list to include only patch info """
self._patchOrderCSVDict[orderCSV] = outPatchList
def __sortCSVDependencyList__(self):
""" Utility methods to sort the CSV file based dependency """
outOrderList = []
""" sort the csv file by the first entry's verification date """
csvFileOrder = sorted(list(self._patchOrderCSVDict.keys()),
key=lambda
item: self._patchOrderCSVDict[item][0].verifiedDate)
for csvFile in csvFileOrder:
outOrderList.extend(self._patchOrderCSVDict[csvFile])
# Removing May 2019: Causes cyclical dependency - J.Snyder
#for idx in range(len(outOrderList)-1, 0, -1):
# installName = outOrderList[idx].installName
# prevInstallName = outOrderList[idx-1].installName
# #if not outOrderList[idx].kidsFilePath == outOrderList[idx-1].kidsFilePath:
# self._csvDepDict[installName] = prevInstallName
def _removeNotInstalledKIDSBuild(self, installName):
patchInfo = self._patchInfoDict.get(installName)
if not patchInfo: return
listToRemove = [installName]
if patchInfo.isMultiBuilds:
listToRemove = patchInfo.multiBuildsList
self._multiBuildDict.pop(patchInfo.kidsFilePath, None)
for install in listToRemove:
logger.info("Removing %s" % install)
self._kidsInstallNameDict.pop(install, None)
self._patchInfoDict.pop(install, None)
""" parse the order csv file and generate an ordered list of install name """
def __getPatchOrderListByCSV__(self, orderCSV):
"""INSTALLED,VERIFY_DT,STATUS,SEQ#,LABELED_AS,CATEGORY,PRODUCT_NAME"""
assert os.path.exists(orderCSV)
logger.info("Parsing file: %s" % orderCSV)
if orderCSV not in self._patchOrderCSVDict:
self._patchOrderCSVDict[orderCSV] = []
patchOrderList = self._patchOrderCSVDict[orderCSV]
result = csv.DictReader(open(orderCSV, 'r'))
installNameSet = set() # to check possible duplicates entry
for row in result:
installName = convertToInstallName(row['LABELED_AS'].strip())
if installName in installNameSet:
logger.error("Ignore duplicate installName %s" % installName)
continue
installNameSet.add(installName)
if row['INSTALLED'].strip() != "TRUE":
self._notInstalledKidsSet.add(installName)
if installName in self._kidsInstallNameDict:
logger.error("Uninstalled patch %s found in %s: %s" %
(installName, self._kidsInstallNameDict[installName],
row))
self._removeNotInstalledKIDSBuild(installName)
logger.debug("Ignore uninstalled patch %s" % row)
continue
try:
verifiedTime = datetime.strptime(row['VERIFY_DT'], "%d-%b-%y")
except ValueError as ex:
verifiedTime = datetime.strptime(row['VERIFY_DT'], "%Y-%m-%d")
""" check the seq # field """
seqNo = row['SEQ#'].strip()
if len(seqNo) > 0:
try: int(seqNo)
except: seqNo = ""
patchOrderList.append((installName, seqNo, verifiedTime))
if re.match("^Informational$", row['CATEGORY'].strip(), re.IGNORECASE):
logger.debug("patch is informational %s " % row)
self._informationalKidsSet.add(installName)
""" generate a sequence of patches that need to be applied by
using topologic sort algorithm.
If installName is provided, will only generated the order WRT.
"""
def __topologicSort__(self, installName=None):
patchDict = self._patchInfoDict
depDict = self._patchDependencyDict
result = topologicSort(depDict, installName)
self._patchOrder = [self._patchInfoDict[x] for x in result if x in patchDict]
self._checkMultiBuildsOrder()
def _checkMultiBuildsOrder(self):
""" make sure that all the multi-build are grouped together """
multiDict = dict()
for index in range(len(self._patchOrder)):
patchInfo = self._patchOrder[index]
if patchInfo.isMultiBuilds:
if patchInfo.kidsFilePath not in multiDict:
multiDict[patchInfo.kidsFilePath] = index
if ( multiDict[patchInfo.kidsFilePath] != index and
multiDict[patchInfo.kidsFilePath] != index - 1 ):
logger.error("Patch out of order %s" % patchInfo)
multiDict[patchInfo.kidsFilePath] = index
""" compare function for PatchInfo objects """
def comparePatchInfo(one, two):
assert isinstance(one, PatchInfo)
assert isinstance(two, PatchInfo)
if (one.package == two.package and
(one.version != None and two.version != None) and
float(one.version) == float(two.version)):
if one.seqNo and two.seqNo:
return cmp(int(one.seqNo), int(two.seqNo))
if one.seqNo: return 1
if two.seqNo: return -1
return 0
if one.rundate and two.rundate:
return cmp(one.rundate, two.rundate)
if one.rundate:
return -1
return 1
""" topologic sort the DAG graph """
def topologicSort(depDict, item=None):
initSet = set()
if item:
initSet.add(item)
else:
initSet = set(depDict.keys())
visitSet = set() # store all node that are already visited
tempStack = [] # mark the temp list
result = []
while len(initSet) > 0:
item = initSet.pop()
visitNode(item, depDict, visitSet, tempStack, result)
initSet.difference_update(visitSet)
return result
def visitNode(nodeName, depDict, visitSet, tempStack, result):
if nodeName in visitSet: # already visited, just return
return
if nodeName in tempStack: # there is a cycle in DAG
index = tempStack.index(nodeName)
logger.error("This is a cycle among these items:\n" +
'\n'.join(repr(x) for x in tempStack[index:]))
raise CycleException("DAG is NOT acyclic")
tempStack.append(nodeName)
for item in depDict.get(nodeName,[]):
visitNode(item, depDict, visitSet, tempStack, result)
""" remove from tempStach """
item = tempStack.pop()
assert item == nodeName
visitSet.add(nodeName)
result.append(nodeName)
""" Utility function to print result of a ordered patch list """
def printPatchOrderList(patchOrderList):
for x in patchOrderList:
print(({"Name" : x.installName},
{"Seq#" : x.seqNo},
{"KIDS" : os.path.basename(x.kidsFilePath)},
{"CSVDep" : x.csvDepPatch},
))
""" Utility function to check if the csv file is indeed in valid format """
def isValidOrderCSVFile(patchesCSV):
assert os.path.exists(patchesCSV)
validFields = VALID_CSV_ORDER_FILE_FIELDS
patches_csv = csv.DictReader(open(patchesCSV, 'r'))
patchCSVHeader = patches_csv.fieldnames
if (patchCSVHeader is None or
len(patchCSVHeader) < len(validFields)):
return False
fieldSet = set(patchCSVHeader)
if fieldSet.issuperset(validFields):
return True
return False
""" generate an output file that can be plotted by graphviz """
def generateDependencyGraph(depDict, outputFile):
with open(outputFile, 'w') as output:
output.write("digraph dependency_graph {\n")
# set graph prop
output.write("\tgraph [nodesep=\"0.35\",\n\t\transsep=\"0.55\"\n\t];\n")
# set the node shape to be box
output.write("\tnode [fontsize=14,\n\t\tshape=box\n\t];\n")
# set the edge label and size props
output.write("\tedge [fontsize=12];\n")
for depNode in depDict:
for item in depDict[depNode]:
output.write("\t\"%s\" -> \"%s\";\n" % (depNode,
item))
output.write("}\n")
###########################################################################
####### """ Testing code section """
###########################################################################
def testGeneratePatchOrder():
import logging
initConsoleLogging(logging.INFO)
patchOrderGen = PatchOrderGenerator()
if len(sys.argv) <= 1:
sys.stderr.write("Specify patch directory")
sys.exit(-1)
result = []
if len(sys.argv) == 2:
result = patchOrderGen.generatePatchOrder(sys.argv[1])
else:
result = patchOrderGen.generatePatchOrder(sys.argv[1], sys.argv[2])
printPatchOrderList(result)
if __name__ == '__main__':
testGeneratePatchOrder()