Coverage for mockslurm/mock_scancel.py: 63%
53 statements
« prev ^ index » next coverage.py v7.6.10, created at 2025-01-17 10:19 +0000
« prev ^ index » next coverage.py v7.6.10, created at 2025-01-17 10:19 +0000
1#!/usr/bin/env python
2"""Implement a mock of the scancel command of slurm.
4The jobs to cancel are found by querying the database for job IDs or job names,
5filtering the jobs that are already completed (return code != default value),
6then the signal to transmit (default SIGKILL) is send.
7"""
9import argparse
10import getpass
11import os
12import signal
14import numpy as np
15from mockslurm.process_db import (
16 JobState,
17 find_db_file,
18 get_db,
19 get_db_file_handle,
20 get_filtered_DB_mask,
21 update_db_value,
22)
23from mockslurm.utils import filter_dict_from_args
26def main():
27 parser = argparse.ArgumentParser(
28 description="Slurm scancel mock", formatter_class=argparse.ArgumentDefaultsHelpFormatter
29 )
30 user_group = parser.add_mutually_exclusive_group()
31 parser.add_argument(
32 "--account",
33 "-A",
34 type=str,
35 dest="ACCOUNT",
36 help="Restrict the scancel operation to jobs under this charge account",
37 )
38 parser.add_argument(
39 "--jobname", "-n", type=str, dest="NAME", help="Restrict the scancel operation to jobs with this job name"
40 )
41 user_group.add_argument(
42 "--me",
43 action="store_true",
44 dest="me",
45 help="Restrict the scancel operation to jobs owned by the current account",
46 )
47 parser.add_argument(
48 "--nodelist",
49 "-w",
50 type=str,
51 dest="NODELIST",
52 help="Cancel any jobs using any of the given hosts. "
53 "The list may be specified as a comma-separated list of hosts, a range of hosts "
54 "(host[1-5,7,...] for example)",
55 )
56 parser.add_argument(
57 "--partition",
58 "-P",
59 type=str,
60 dest="PARTITION",
61 help="Restrict the scancel operation to jobs in this partition",
62 )
63 parser.add_argument(
64 "--quiet",
65 "-Q",
66 type=str,
67 dest="quiet",
68 help="Do not report an error if the specified job is already completed",
69 )
70 parser.add_argument(
71 "--reservation",
72 "-R",
73 type=str,
74 dest="RESERVATION",
75 help="Restrict the scancel operation to jobs with this reservation name",
76 )
77 parser.add_argument(
78 "--signal",
79 "-s",
80 type=str,
81 dest="signal",
82 default="SIGKILL",
83 help="The name or number of the signal to send. If this option is not used the specified job or step will be terminated",
84 )
85 parser.add_argument(
86 "--user",
87 "-u",
88 type=str,
89 dest="USER",
90 help="Restrict the scancel operation to jobs owned by the given user",
91 )
92 parser.add_argument("jobids", type=int, nargs="*", help="The Slurm job ID to be signaled")
93 args = parser.parse_args()
94 if ( 94 ↛ 104line 94 didn't jump to line 104 because the condition on line 94 was never true
95 not args.jobids
96 and args.ACCOUNT is None
97 and args.NAME is None
98 and args.me is False
99 and args.NODELIST is None
100 and args.PARTITION is None
101 and args.RESERVATION is None
102 and args.USER is None
103 ):
104 print("scancel: error: No job identification provided")
105 exit(1)
107 if args.jobids: 107 ↛ 108line 107 didn't jump to line 108 because the condition on line 107 was never true
108 for id in args.jobids:
109 if id < 0:
110 print("scancel: error: Invalid job id {}".format(id))
111 exit(1)
113 if args.me is not False: 113 ↛ 114line 113 didn't jump to line 114 because the condition on line 113 was never true
114 args.ACCOUNT = getpass.getuser()
116 if args.signal in [signal.name for signal in signal.Signals]: 116 ↛ 119line 116 didn't jump to line 119 because the condition on line 116 was always true
117 args.signal = signal.Signals[args.signal]
118 else:
119 try:
120 args.signal = int(args.signal)
121 except:
122 print("Unknown job signal: {}".format(args.signal))
123 exit(1)
125 # Transform the arguments values into a map {field: value, field2: [values], etc...} for filtering DB
126 field_filter_values = filter_dict_from_args(args)
128 with get_db_file_handle(find_db_file()) as db_file:
129 db = get_db(db_file)
130 if db.shape[0] == 0: # db is empty: no jobs to cancel 130 ↛ 131line 130 didn't jump to line 131 because the condition on line 130 was never true
131 exit(0)
133 # Get mask to select DB rows
134 mask = get_filtered_DB_mask(db_file, field_filter_values)
135 # filter job IDs if some were specified
136 if args.jobids: 136 ↛ 137line 136 didn't jump to line 137 because the condition on line 136 was never true
137 mask[args.jobids] = True
139 job_indices = np.nonzero(mask)[0]
140 # Send signal
141 for idx, job in zip(job_indices, db[mask]):
142 if job["STATE"] == JobState.RUNNING: # job is started, we can kill it 142 ↛ 143line 142 didn't jump to line 143 because the condition on line 142 was never true
143 os.kill(job["PID"], args.signal)
144 # set STATE to stopped immediately, actual exit STATE will be updated with exit code
145 # in mock of sbatch if job was running
146 # TODO: if signal wasn't meant to kill job, STATE is wrong ?
147 update_db_value(db_file, idx, key="STATE", value=JobState.CANCELLED)
150if __name__ == "__main__": 150 ↛ 151line 150 didn't jump to line 151 because the condition on line 150 was never true
151 main()