@@ -709,6 +709,21 @@ def disable_hook(cls, hook_name):
709709 extra_resources = fields .TypedField ('extra_resources' ,
710710 typ .Dict [str , typ .Dict [str , object ]])
711711
712+ #: .. versionadded:: 3.3
713+ #:
714+ #: Always build the source code for this test locally. If set to
715+ #: :class:`False`, ReFrame will spawn a build job on the partition where
716+ #: the test will run. Setting this to :class:`False` is useful when
717+ #: cross-compilation is not supported on the system where ReFrame is run.
718+ #: Normally, ReFrame will mark the test as a failure if the spawned job
719+ #: exits with a non-zero exit code. However, certain scheduler backends,
720+ #: such as the ``squeue`` do not set it. In such cases, it is the user's
721+ #: responsibility to check whether the build phase failed by adding an
722+ #: appropriate sanity check.
723+ #:
724+ #: :type: boolean : :default: :class:`True`
725+ build_locally = fields .TypedField ('build_locally' , bool )
726+
712727 def __new__ (cls , * args , ** kwargs ):
713728 obj = super ().__new__ (cls )
714729
@@ -727,7 +742,12 @@ def __new__(cls, *args, **kwargs):
727742 if osext .is_interactive ():
728743 prefix = os .getcwd ()
729744 else :
730- prefix = os .path .abspath (os .path .dirname (inspect .getfile (cls )))
745+ try :
746+ prefix = cls ._rfm_pinned_prefix
747+ except AttributeError :
748+ prefix = os .path .abspath (
749+ os .path .dirname (inspect .getfile (cls ))
750+ )
731751
732752 obj ._rfm_init (name , prefix )
733753 return obj
@@ -736,10 +756,17 @@ def __init__(self):
736756 pass
737757
738758 @classmethod
739- def __init_subclass__ (cls , * , special = False , ** kwargs ):
759+ def __init_subclass__ (cls , * , special = False , pin_prefix = False , ** kwargs ):
740760 super ().__init_subclass__ (** kwargs )
741761 cls ._rfm_special_test = special
742762
763+ # Insert the prefix to pin the test to if the test lives in a test
764+ # library with resources in it.
765+ if pin_prefix :
766+ cls ._rfm_pinned_prefix = os .path .abspath (
767+ os .path .dirname (inspect .getfile (cls ))
768+ )
769+
743770 def _rfm_init (self , name = None , prefix = None ):
744771 if name is not None :
745772 self .name = name
@@ -777,6 +804,7 @@ def _rfm_init(self, name=None, prefix=None):
777804
778805 # True only if check is to be run locally
779806 self .local = False
807+ self .build_locally = True
780808
781809 # Static directories of the regression check
782810 self ._prefix = os .path .abspath (prefix )
@@ -1038,29 +1066,29 @@ def _setup_paths(self):
10381066 except OSError as e :
10391067 raise PipelineError ('failed to set up paths' ) from e
10401068
1041- def _setup_job (self , ** job_opts ):
1069+ def _setup_job (self , name , force_local = False , ** job_opts ):
10421070 '''Setup the job related to this check.'''
10431071
1044- if self . local :
1072+ if force_local :
10451073 scheduler = getscheduler ('local' )()
10461074 launcher = getlauncher ('local' )()
10471075 else :
10481076 scheduler = self ._current_partition .scheduler
10491077 launcher = self ._current_partition .launcher_type ()
10501078
10511079 self .logger .debug (
1052- f'Setting up run job descriptor '
1080+ f'Setting up job { name !r } '
10531081 f'(scheduler: { scheduler .registered_name !r} , '
10541082 f'launcher: { launcher .registered_name !r} )'
10551083 )
1056- self . _job = Job .create (scheduler ,
1057- launcher ,
1058- name = 'rfm_%s_job' % self . name ,
1059- workdir = self ._stagedir ,
1060- max_pending_time = self .max_pending_time ,
1061- sched_access = self ._current_partition .access ,
1062- sched_exclusive_access = self .exclusive_access ,
1063- ** job_opts )
1084+ return Job .create (scheduler ,
1085+ launcher ,
1086+ name = name ,
1087+ workdir = self ._stagedir ,
1088+ max_pending_time = self .max_pending_time ,
1089+ sched_access = self ._current_partition .access ,
1090+ sched_exclusive_access = self .exclusive_access ,
1091+ ** job_opts )
10641092
10651093 def _setup_perf_logging (self ):
10661094 self ._perf_logger = logging .getperflogger (self )
@@ -1089,7 +1117,12 @@ def setup(self, partition, environ, **job_opts):
10891117 self ._current_partition = partition
10901118 self ._current_environ = environ
10911119 self ._setup_paths ()
1092- self ._setup_job (** job_opts )
1120+ self ._job = self ._setup_job (f'rfm_{ self .name } _job' ,
1121+ self .local ,
1122+ ** job_opts )
1123+ self ._build_job = self ._setup_job (f'rfm_{ self .name } _build' ,
1124+ self .local or self .build_locally ,
1125+ ** job_opts )
10931126
10941127 def _copy_to_stagedir (self , path ):
10951128 self .logger .debug (f'Copying { path } to stage directory' )
@@ -1191,10 +1224,6 @@ def compile(self):
11911224 environs = [self ._current_partition .local_env , self ._current_environ ,
11921225 user_environ , self ._cdt_environ ]
11931226
1194- self ._build_job = Job .create (getscheduler ('local' )(),
1195- launcher = getlauncher ('local' )(),
1196- name = 'rfm_%s_build' % self .name ,
1197- workdir = self ._stagedir )
11981227 with osext .change_dir (self ._stagedir ):
11991228 try :
12001229 self ._build_job .prepare (
@@ -1225,8 +1254,8 @@ def compile_wait(self):
12251254 '''
12261255 self ._build_job .wait ()
12271256
1228- # FIXME: this check is not reliable for certain scheduler backends
1229- if self ._build_job .exitcode != 0 :
1257+ # We raise a BuildError when we an exit code and it is non zero
1258+ if self ._build_job .exitcode :
12301259 raise BuildError (self ._build_job .stdout , self ._build_job .stderr )
12311260
12321261 @_run_hooks ('pre_run' )
@@ -1760,6 +1789,20 @@ class RunOnlyRegressionTest(RegressionTest, special=True):
17601789 module.
17611790 '''
17621791
1792+ @_run_hooks ()
1793+ def setup (self , partition , environ , ** job_opts ):
1794+ '''The setup stage of the regression test pipeline.
1795+
1796+ Similar to the :func:`RegressionTest.setup`, except that no build job
1797+ is created for this test.
1798+ '''
1799+ self ._current_partition = partition
1800+ self ._current_environ = environ
1801+ self ._setup_paths ()
1802+ self ._job = self ._setup_job (f'rfm_{ self .name } _job' ,
1803+ self .local ,
1804+ ** job_opts )
1805+
17631806 def compile (self ):
17641807 '''The compilation phase of the regression test pipeline.
17651808
@@ -1802,21 +1845,20 @@ class CompileOnlyRegressionTest(RegressionTest, special=True):
18021845 module.
18031846 '''
18041847
1805- def _rfm_init (self , * args , ** kwargs ):
1806- super ()._rfm_init (* args , ** kwargs )
1807- self .local = True
1808-
18091848 @_run_hooks ()
18101849 def setup (self , partition , environ , ** job_opts ):
18111850 '''The setup stage of the regression test pipeline.
18121851
1813- Similar to the :func:`RegressionTest.setup`, except that no job
1814- descriptor is set up for this test.
1852+ Similar to the :func:`RegressionTest.setup`, except that no run job
1853+ is created for this test.
18151854 '''
18161855 # No need to setup the job for compile-only checks
18171856 self ._current_partition = partition
18181857 self ._current_environ = environ
18191858 self ._setup_paths ()
1859+ self ._build_job = self ._setup_job (f'rfm_{ self .name } _build' ,
1860+ self .local or self .build_locally ,
1861+ ** job_opts )
18201862
18211863 @property
18221864 @sn .sanity_function
0 commit comments