Belle II Software development
LSF.LSFResult Class Reference
Inheritance diagram for LSF.LSFResult:
Result

Public Member Functions

 __init__ (self, job, job_id)
 
 update_status (self)
 
 ready (self)
 
 get_exit_code_from_file (self)
 

Public Attributes

 job_id = job_id
 job id given by LSF
 
 job = job
 Job object for result.
 
 time_to_wait_for_exit_code_file = timedelta(minutes=5)
 After our first attempt to view the exit code file once the job is 'finished', how long should we wait for it to exist before timing out?
 
 exit_code_file_initial_time = None
 Time we started waiting for the exit code file to appear.
 

Static Public Attributes

dict backend_code_to_status
 LSF statuses mapped to Job statuses.
 

Protected Member Functions

 _update_result_status (self, bjobs_output)
 
 _get_status_from_output (self, output)
 

Protected Attributes

bool _is_ready = False
 Quicker way to know if it's ready once it has already been found.
 

Detailed Description

Simple class to help monitor status of jobs submitted by LSF Backend.

You pass in a `Job` object and job id from a bsub command.
When you call the `ready` method it runs bjobs to see whether or not the job has finished.

Definition at line 1677 of file backends.py.

Constructor & Destructor Documentation

◆ __init__()

__init__ ( self,
job,
job_id )
Pass in the job object and the job id to allow the result to do monitoring and perform
post processing of the job.

Definition at line 1693 of file backends.py.

1693 def __init__(self, job, job_id):
1694 """
1695 Pass in the job object and the job id to allow the result to do monitoring and perform
1696 post processing of the job.
1697 """
1698 super().__init__(job)
1699
1700 self.job_id = job_id
1701

Member Function Documentation

◆ _get_status_from_output()

_get_status_from_output ( self,
output )
protected
Get status from output

Definition at line 1756 of file backends.py.

1756 def _get_status_from_output(self, output):
1757 """
1758 Get status from output
1759 """
1760 if output["JOBS"] and "ERROR" in output["JOBS"][0]:
1761 if output["JOBS"][0]["ERROR"] == f"Job <{self.job_id}> is not found":
1762 raise KeyError(f"No job record in the 'output' argument had the 'JOBID'=={self.job_id}")
1763 else:
1764 raise BackendError(f"Unidentified Error during status check for {self.job}: {output}")
1765 else:
1766 for job_info in output["JOBS"]:
1767 if job_info["JOBID"] == self.job_id:
1768 return job_info["STAT"]
1769 else:
1770 raise KeyError(f"No job record in the 'output' argument had the 'JOBID'=={self.job_id}")
1771
1772 @classmethod

◆ _update_result_status()

_update_result_status ( self,
bjobs_output )
protected
Parameters:
    bjobs_output (dict): The JSON output of a previous call to bjobs which we can reuse to find the
        status of this job. Obviously you should only be passing a JSON dict that contains the 'stat' and
        'id' information, otherwise it is useless.

Definition at line 1715 of file backends.py.

1715 def _update_result_status(self, bjobs_output):
1716 """
1717 Parameters:
1718 bjobs_output (dict): The JSON output of a previous call to bjobs which we can reuse to find the
1719 status of this job. Obviously you should only be passing a JSON dict that contains the 'stat' and
1720 'id' information, otherwise it is useless.
1721
1722 """
1723 try:
1724 backend_status = self._get_status_from_output(bjobs_output)
1725 except KeyError:
1726 # If this happens then maybe the job id wasn't in the bjobs_output argument because it finished.
1727 # Instead of failing immediately we try re-running bjobs here explicitly and then fail if it still isn't there.
1728 bjobs_output = LSF.bjobs(output_fields=["stat", "id"], job_id=str(self.job_id))
1729 try:
1730 backend_status = self._get_status_from_output(bjobs_output)
1731 except KeyError:
1732 # If this happened, maybe we're looking at an old finished job. We could fall back to bhist, but it's
1733 # slow and terrible. Instead let's try looking for the exit code file.
1734 try:
1735 exit_code = self.get_exit_code_from_file()
1736 except FileNotFoundError:
1737 waiting_time = datetime.now() - self.exit_code_file_initial_time
1738 if self.time_to_wait_for_exit_code_file > waiting_time:
1739 B2ERROR(f"Exit code file for {self.job} missing and we can't wait longer. Setting exit code to 1.")
1740 exit_code = 1
1741 else:
1742 B2WARNING(f"Exit code file for {self.job} missing, will wait longer.")
1743 return
1744 if exit_code:
1745 backend_status = "EXIT"
1746 else:
1747 backend_status = "FINISHED"
1748 try:
1749 new_job_status = self.backend_code_to_status[backend_status]
1750 except KeyError as err:
1751 raise BackendError(f"Unidentified backend status found for {self.job}: {backend_status}") from err
1752
1753 if new_job_status != self.job.status:
1754 self.job.status = new_job_status
1755

◆ get_exit_code_from_file()

get_exit_code_from_file ( self)
inherited
Read the exit code file to discover the exit status of the job command. Useful fallback if the job is no longer
known to the job database (batch system purged it for example). Since some backends may take time to download
the output files of the job back to the working directory we use a time limit on how long to wait.

Definition at line 908 of file backends.py.

908 def get_exit_code_from_file(self):
909 """
910 Read the exit code file to discover the exit status of the job command. Useful fallback if the job is no longer
911 known to the job database (batch system purged it for example). Since some backends may take time to download
912 the output files of the job back to the working directory we use a time limit on how long to wait.
913 """
914 if not self.exit_code_file_initial_time:
915 self.exit_code_file_initial_time = datetime.now()
916 exit_code_path = Path(self.job.working_dir, Backend.exit_code_file)
917 with open(exit_code_path) as f:
918 exit_code = int(f.read().strip())
919 B2DEBUG(29, f"Exit code from file for {self.job} was {exit_code}")
920 return exit_code
921
922

◆ ready()

ready ( self)
inherited
Returns whether or not this job result is known to be ready. Doesn't actually change the job status. Just changes
the 'readiness' based on the known job status.

Definition at line 887 of file backends.py.

887 def ready(self):
888 """
889 Returns whether or not this job result is known to be ready. Doesn't actually change the job status. Just changes
890 the 'readiness' based on the known job status.
891 """
892 B2DEBUG(29, f"Calling {self.job}.result.ready()")
893 if self._is_ready:
894 return True
895 elif self.job.status in self.job.exit_statuses:
896 self._is_ready = True
897 return True
898 else:
899 return False
900

◆ update_status()

update_status ( self)
Update the job's (or subjobs') status by calling bjobs.

Reimplemented from Result.

Definition at line 1702 of file backends.py.

1702 def update_status(self):
1703 """
1704 Update the job's (or subjobs') status by calling bjobs.
1705 """
1706 B2DEBUG(29, f"Calling {self.job.name}.result.update_status()")
1707 # Get all jobs info and reuse it for each status update to minimise tie spent on this updating.
1708 bjobs_output = LSF.bjobs(output_fields=["stat", "id"])
1709 if self.job.subjobs:
1710 for subjob in self.job.subjobs.values():
1711 subjob.result._update_result_status(bjobs_output)
1712 else:
1713 self._update_result_status(bjobs_output)
1714

Member Data Documentation

◆ _is_ready

bool _is_ready = False
protectedinherited

Quicker way to know if it's ready once it has already been found.

Saves a lot of calls to batch system commands.

Definition at line 880 of file backends.py.

◆ backend_code_to_status

dict backend_code_to_status
static
Initial value:
= {"RUN": "running",
"DONE": "completed",
"FINISHED": "completed",
"EXIT": "failed",
"PEND": "submitted"
}

LSF statuses mapped to Job statuses.

Definition at line 1686 of file backends.py.

◆ exit_code_file_initial_time

exit_code_file_initial_time = None
inherited

Time we started waiting for the exit code file to appear.

Definition at line 885 of file backends.py.

◆ job

job = job
inherited

Job object for result.

Definition at line 878 of file backends.py.

◆ job_id

job_id = job_id

job id given by LSF

Definition at line 1700 of file backends.py.

◆ time_to_wait_for_exit_code_file

time_to_wait_for_exit_code_file = timedelta(minutes=5)
inherited

After our first attempt to view the exit code file once the job is 'finished', how long should we wait for it to exist before timing out?

Definition at line 883 of file backends.py.


The documentation for this class was generated from the following file: