@@ -109,7 +109,8 @@ class EasyConfig(object):
109
109
Class which handles loading, reading, validation of easyconfigs
110
110
"""
111
111
112
- def __init__ (self , path , extra_options = None , build_specs = None , validate = True , hidden = None , rawtxt = None ):
112
+ def __init__ (self , path , extra_options = None , build_specs = None , validate = True , hidden = None , rawtxt = None ,
113
+ auto_convert_value_types = True ):
113
114
"""
114
115
initialize an easyconfig.
115
116
@param path: path to easyconfig file to be parsed (ignored if rawtxt is specified)
@@ -118,6 +119,8 @@ def __init__(self, path, extra_options=None, build_specs=None, validate=True, hi
118
119
@param validate: indicates whether validation should be performed (note: combined with 'validate' build option)
119
120
@param hidden: indicate whether corresponding module file should be installed hidden ('.'-prefixed)
120
121
@param rawtxt: raw contents of easyconfig file
122
+ @param auto_convert_value_types: indicates wether types of easyconfig values should be automatically converted
123
+ in case they are wrong
121
124
"""
122
125
self .template_values = None
123
126
self .enable_templating = True # a boolean to control templating
@@ -184,7 +187,8 @@ def __init__(self, path, extra_options=None, build_specs=None, validate=True, hi
184
187
185
188
# parse easyconfig file
186
189
self .build_specs = build_specs
187
- self .parser = EasyConfigParser (filename = self .path , rawcontent = self .rawtxt )
190
+ self .parser = EasyConfigParser (filename = self .path , rawcontent = self .rawtxt ,
191
+ auto_convert_value_types = auto_convert_value_types )
188
192
self .parse ()
189
193
190
194
# handle allowed system dependencies
@@ -214,6 +218,7 @@ def __init__(self, path, extra_options=None, build_specs=None, validate=True, hi
214
218
self .short_mod_name = mns .det_short_module_name (self )
215
219
self .mod_subdir = mns .det_module_subdir (self )
216
220
221
+ self .software_license = None
217
222
218
223
def copy (self ):
219
224
"""
@@ -335,16 +340,17 @@ def validate(self, check_osdeps=True):
335
340
336
341
def validate_license (self ):
337
342
"""Validate the license"""
338
- lic = self . _config ['software_license' ][ 0 ]
343
+ lic = self ['software_license' ]
339
344
if lic is None :
340
345
# when mandatory, remove this possibility
341
346
if 'software_license' in self .mandatory :
342
- raise EasyBuildError ("License is mandatory, but 'software_license' is undefined" )
343
- elif not isinstance (lic , License ):
344
- raise EasyBuildError ('License %s has to be a License subclass instance, found classname %s.' ,
345
- lic , lic .__class__ .__name__ )
346
- elif not lic .name in EASYCONFIG_LICENSES_DICT :
347
- raise EasyBuildError ('Invalid license %s (classname: %s).' , lic .name , lic .__class__ .__name__ )
347
+ raise EasyBuildError ("Software license is mandatory, but 'software_license' is undefined" )
348
+ elif lic in EASYCONFIG_LICENSES_DICT :
349
+ # create License instance
350
+ self .software_license = EASYCONFIG_LICENSES_DICT [lic ]()
351
+ else :
352
+ known_licenses = ', ' .join (sorted (EASYCONFIG_LICENSES_DICT .keys ()))
353
+ raise EasyBuildError ("Invalid license %s (known licenses: %s)" , lic , known_licenses )
348
354
349
355
# TODO, when GROUP_SOURCE and/or GROUP_BINARY is True
350
356
# check the owner of source / binary (must match 'group' parameter from easyconfig)
@@ -478,7 +484,21 @@ def toolchain(self):
478
484
returns the Toolchain used
479
485
"""
480
486
if self ._toolchain is None :
481
- self ._toolchain = get_toolchain (self ['toolchain' ], self ['toolchainopts' ], mns = ActiveMNS ())
487
+ # provide list of (direct) toolchain dependencies (name & version), if easyconfig can be found for toolchain
488
+ tcdeps = None
489
+ tcname , tcversion = self ['toolchain' ]['name' ], self ['toolchain' ]['version' ]
490
+ if tcname != DUMMY_TOOLCHAIN_NAME :
491
+ tc_ecfile = robot_find_easyconfig (tcname , tcversion )
492
+ if tc_ecfile is None :
493
+ self .log .debug ("No easyconfig found for toolchain %s version %s, can't determine dependencies" ,
494
+ tcname , tcversion )
495
+ else :
496
+ self .log .debug ("Found easyconfig for toolchain %s version %s: %s" , tcname , tcversion , tc_ecfile )
497
+ tc_ec = process_easyconfig (tc_ecfile )[0 ]
498
+ tcdeps = tc_ec ['dependencies' ]
499
+ self .log .debug ("Toolchain dependencies based on easyconfig: %s" , tcdeps )
500
+
501
+ self ._toolchain = get_toolchain (self ['toolchain' ], self ['toolchainopts' ], mns = ActiveMNS (), tcdeps = tcdeps )
482
502
tc_dict = self ._toolchain .as_dict ()
483
503
self .log .debug ("Initialized toolchain: %s (opts: %s)" % (tc_dict , self ['toolchainopts' ]))
484
504
return self ._toolchain
@@ -1008,28 +1028,33 @@ def create_paths(path, name, version):
1008
1028
1009
1029
def robot_find_easyconfig (name , version ):
1010
1030
"""
1011
- Find an easyconfig for module in path
1031
+ Find an easyconfig for module in path, returns (absolute) path to easyconfig file (or None, if none is found).
1012
1032
"""
1013
1033
key = (name , version )
1014
1034
if key in _easyconfig_files_cache :
1015
1035
_log .debug ("Obtained easyconfig path from cache for %s: %s" % (key , _easyconfig_files_cache [key ]))
1016
1036
return _easyconfig_files_cache [key ]
1037
+
1017
1038
paths = build_option ('robot_path' )
1018
- if not paths :
1019
- raise EasyBuildError ( "No robot path specified, which is required when looking for easyconfigs (use --robot)" )
1020
- if not isinstance (paths , (list , tuple )):
1039
+ if paths is None :
1040
+ paths = []
1041
+ elif not isinstance (paths , (list , tuple )):
1021
1042
paths = [paths ]
1022
- # candidate easyconfig paths
1043
+
1044
+ res = None
1023
1045
for path in paths :
1024
1046
easyconfigs_paths = create_paths (path , name , version )
1025
1047
for easyconfig_path in easyconfigs_paths :
1026
1048
_log .debug ("Checking easyconfig path %s" % easyconfig_path )
1027
1049
if os .path .isfile (easyconfig_path ):
1028
1050
_log .debug ("Found easyconfig file for name %s, version %s at %s" % (name , version , easyconfig_path ))
1029
1051
_easyconfig_files_cache [key ] = os .path .abspath (easyconfig_path )
1030
- return _easyconfig_files_cache [key ]
1052
+ res = _easyconfig_files_cache [key ]
1053
+ break
1054
+ if res :
1055
+ break
1031
1056
1032
- return None
1057
+ return res
1033
1058
1034
1059
1035
1060
class ActiveMNS (object ):
@@ -1091,7 +1116,22 @@ def _det_module_name_with(self, mns_method, ec, force_visible=False):
1091
1116
- string representing module name has length > 0
1092
1117
- module name only contains printable characters (string.printable, except carriage-control chars)
1093
1118
"""
1094
- mod_name = mns_method (self .check_ec_type (ec ))
1119
+ ec = self .check_ec_type (ec )
1120
+
1121
+ # replace software name with desired replacement (if specified)
1122
+ orig_name = None
1123
+ if ec .get ('modaltsoftname' , None ):
1124
+ orig_name = ec ['name' ]
1125
+ ec ['name' ] = ec ['modaltsoftname' ]
1126
+ self .log .info ("Replaced software name '%s' with '%s' when determining module name" , orig_name , ec ['name' ])
1127
+ else :
1128
+ self .log .debug ("No alternative software name specified to determine module name with" )
1129
+
1130
+ mod_name = mns_method (ec )
1131
+
1132
+ # restore original software name if it was tampered with
1133
+ if orig_name is not None :
1134
+ ec ['name' ] = orig_name
1095
1135
1096
1136
if not is_valid_module_name (mod_name ):
1097
1137
raise EasyBuildError ("%s is not a valid module name" , str (mod_name ))
@@ -1134,7 +1174,7 @@ def det_short_module_name(self, ec, force_visible=False):
1134
1174
self .log .debug ("Obtained valid short module name %s" % mod_name )
1135
1175
1136
1176
# sanity check: obtained module name should pass the 'is_short_modname_for' check
1137
- if not self .is_short_modname_for (mod_name , ec ['name' ]):
1177
+ if not self .is_short_modname_for (mod_name , ec . get ( 'modaltsoftname' , None ) or ec ['name' ]):
1138
1178
raise EasyBuildError ("is_short_modname_for('%s', '%s') for active module naming scheme returns False" ,
1139
1179
mod_name , ec ['name' ])
1140
1180
return mod_name
0 commit comments