From a327f31fe382c71ad9a8fea1895d2e66ef396ac5 Mon Sep 17 00:00:00 2001
From: Edward Moyse <edward.moyse@cern.ch>
Date: Fri, 6 Nov 2020 10:33:49 +0100
Subject: [PATCH] Better handle wrapping of private tools - adds quite a lot of
 extra debug output - and some more checks, such as checking the types being
 compared (can be different if configuration is different) - handle the bug
 that you cannot create a private tool if a public tool with the same name
 exists - skip if the property value is none

---
 .../python/ComponentAccumulator.py            | 55 ++++++++++++++++---
 1 file changed, 47 insertions(+), 8 deletions(-)

diff --git a/Control/AthenaConfiguration/python/ComponentAccumulator.py b/Control/AthenaConfiguration/python/ComponentAccumulator.py
index 6550974705c..ca46dbd5dd8 100644
--- a/Control/AthenaConfiguration/python/ComponentAccumulator.py
+++ b/Control/AthenaConfiguration/python/ComponentAccumulator.py
@@ -933,6 +933,11 @@ def conf2toConfigurable( comp, indent="", suppressDupes=False ):
     def __areSettingsSame( existingConfigurableInstance, newConf2Instance, indent="" ):
         _log.debug( "%sChecking if settings are the same %s %s",
                     indent, existingConfigurableInstance.getFullName(), newConf2Instance.getFullJobOptName() )
+        if (existingConfigurableInstance.getType() != newConf2Instance.__cpp_type__): 
+            raise ConfigurationError("Old/new ({} | {}) cpp types are not the same for ({} | {}) !".format(
+                                    existingConfigurableInstance.getType(),newConf2Instance.__cpp_type__, 
+                                    existingConfigurableInstance.getFullName(), newConf2Instance.getFullJobOptName() ) )
+
         alreadySetProperties = existingConfigurableInstance.getValuedProperties().copy()
         _log.debug( "Existing properties: %s", alreadySetProperties )
         _log.debug( "New properties: %s", newConf2Instance._properties )
@@ -944,18 +949,35 @@ def conf2toConfigurable( comp, indent="", suppressDupes=False ):
                               indent, compName(newConf2Instance), pname, compName(pvalue) )
                 continue
             propType = newConf2Instance._descriptors[pname].cpp_type
-            _log.debug( "%sComparing type: %s for: %s", indent, propType, pname )
+            _log.debug( "%sComparing type: %s for: %s in: %s", indent, propType, pname, existingConfigurableInstance.getFullJobOptName() )
             if  "PrivateToolHandleArray" in  propType:
                 toolDict = {_.getName(): _ for _ in alreadySetProperties[pname]}
                 _log.debug('Private tool properties? %s', toolDict)
                 newCdict = {_.getName() : _ for _ in pvalue}
                 oldCset = set(toolDict); newCset = set(newCdict)
                 _log.debug('Private tool property names? %s %s', oldCset, newCset)
+                if ( not (oldCset == newCset) ):
+                    _log.warning('%s PrivateToolHandleArray %s  does not have the same named components',indent, pname )
+                    _log.warning('%s Old (conf1) %s for %s',indent, oldCset, existingConfigurableInstance.getFullJobOptName())
+                    _log.warning('%s New (conf2) %s for %s',indent, newCset, newConf2Instance.getFullJobOptName())
+                    _log.warning('%s Will try to merge them, but this might go wrong!',indent)
                 for oldC in oldCset & newCset:
                     __areSettingsSame( toolDict[oldC], newCdict[oldC], __indent(indent))
+                # And now just the new properties in conf2 (the stuff just in conf1 is already in the objec)
                 for newC in newCset-oldCset:
-                    # clone new config to old array
-                    alreadySetProperties[pname].append(conf2toConfigurable(newCdict[newC]))
+                    className = newCdict[newC].getFullJobOptName().split( "/" )[0]
+                    _log.debug('%s %s not in oldconfig. Will try to create conf1 instance using this className: %s, and merge.',indent, newC, className)
+                    configurableClass = __findConfigurableClass( className )
+                    # Do not create with existing name, or it will try to get an existing public tool, if available
+                    # (and public tools cannot be added to a PrivateToolHandleArray)
+                    instance = configurableClass( newC + className + str(len(indent)) ) 
+
+                    # Now give it the correct name
+                    instance._name = newCdict[newC].name
+
+                    __setProperties( instance, newCdict[newC], __indent( indent ) )
+                    _log.debug('%s will now add %s to array.',indent, instance) 
+                    alreadySetProperties[pname].append(instance)
             elif "PublicToolHandleArray" in propType:
                 toolSet = {_.getName() for _ in alreadySetProperties[pname]}
                 _log.debug('Public tool handle array properties? %s %s', toolSet, pvalue)
@@ -976,6 +998,9 @@ def conf2toConfigurable( comp, indent="", suppressDupes=False ):
                                  "skipping deeper checks, configuration may be incorrect",
                                  indent, propType, newConf2Instance.name, pname, pvalue)
                 else:
+                    if pvalue is None:
+                        _log.debug("%sThe property value for %s of %s is None. Skipping.", indent, pname, newConf2Instance.name )
+                        continue
                     _log.debug( "%sSome kind of handle and, object type %s existing %s",
                                 indent, type(pvalue), type(existingVal) )
                     __areSettingsSame( existingVal, pvalue, indent)
@@ -985,22 +1010,36 @@ def conf2toConfigurable( comp, indent="", suppressDupes=False ):
 
                 if pname not in alreadySetProperties:
                     _log.info( "%sAdding property: %s for %s", indent, pname, newConf2Instance.getName() )
-                    setattr(existingConfigurableInstance, pname, pvalue)
+                    try:
+                        setattr(existingConfigurableInstance, pname, pvalue)
+                    except AttributeError:
+                        _log.info("%s Could not set attribute. Type of existingConfigurableInstance %s.",indent, type(existingConfigurableInstance) )
+                        raise    
                 elif alreadySetProperties[pname] != pvalue:
                     _log.info( "%sMerging property: %s for %s", indent, pname, newConf2Instance.getName() )
                     # create surrogate
                     clone = newConf2Instance.getInstance("Clone")
                     setattr(clone, pname, alreadySetProperties[pname])
-                    updatedPropValue = __listHelperToList(newConf2Instance._descriptors[pname].semantics.merge( getattr(newConf2Instance, pname), getattr(clone, pname)))
-                        
-                    setattr(existingConfigurable, pname, updatedPropValue)
+                    try:
+                        updatedPropValue = __listHelperToList(newConf2Instance._descriptors[pname].semantics.merge( getattr(newConf2Instance, pname), getattr(clone, pname)))
+                    except ValueError:
+                        _log.warning( "Failed merging new config value (%s) and old config value (%s) for (%s) property of %s (%s) old (new). Will take value from NEW configuration, but this should be checked!",
+                            getattr(newConf2Instance, pname),getattr(clone, pname),pname,existingConfigurableInstance.getFullJobOptName() ,newConf2Instance.getFullJobOptName() )
+                        updatedPropValue=getattr(newConf2Instance, pname)
+                        # Necessary because default value is returned for lists... see e.g.:
+                        # https://gitlab.cern.ch/atlas/athena/-/blob/630b82d0540fff24c2a1da6716efc0adb53727c9/Control/AthenaCommon/python/PropertyProxy.py#L109
+                    _log.debug("existingConfigurable.name: %s, pname: %s, updatedPropValue: %s", existingConfigurableInstance.name(), pname, updatedPropValue )
+
+                    setattr(existingConfigurableInstance, pname, updatedPropValue)
                     del clone
                     _log.info("%s invoked GaudiConf2 semantics to merge the %s and the %s to %s "
                               "for property %s of %s",
                               indent, alreadySetProperties[pname], pvalue, pname,
                               updatedPropValue, existingConfigurable.getFullName())
-
+    
+    _log.debug( "%s Conf2 Full name: %s ", indent, comp.getFullJobOptName() )
     existingConfigurable = __alreadyConfigured( comp.name )
+ 
     if existingConfigurable: # if configurable exists we try to merge with it
         _log.debug( "%sPre-existing configurable %s was found, checking if has the same properties", indent, comp.getName() )
         __areSettingsSame( existingConfigurable, comp )
-- 
GitLab