Skip to content

Commit bd10ce1

Browse files
committed
expand(): correct leading zeros issue
Problem: As noted in LLNL#32, the expand() function doesn't correctly handle hostlists with leading zeros in the ranges. Rework and restructure the expand() function to correctly handle ranges of leading zeros, particularly when there are multiple ranges. Add a unit test using the example provided in LLNL#32 to confirm it is working as expected.
1 parent 8ace7a6 commit bd10ce1

File tree

2 files changed

+30
-52
lines changed

2 files changed

+30
-52
lines changed

hostlist/hostlist.py

+24-52
Original file line numberDiff line numberDiff line change
@@ -73,67 +73,39 @@ def expand(nodelist):
7373
:return: The expanded hostlist string.
7474
"""
7575
node_list = nodelist.split(", ")
76-
# print node_list
77-
7876
result_hostlist = []
77+
7978
for node in node_list:
8079
nodelist_match = r"(\w+-?)\[((,?[0-9]+-?,?-?){0,})\](.*)?"
81-
if re.search(nodelist_match, node):
82-
match = re.search(nodelist_match, node)
80+
match = re.search(nodelist_match, node)
8381

84-
# holds the ranges of nodes as a string
85-
# now we can manipulate the string and cast it to a list of numbers
86-
oldstr = str(match.group(2))
87-
left_br = oldstr.replace("[", "")
88-
right_br = left_br.replace("]", "")
89-
num_list = right_br.split(',')
82+
if match:
83+
prefix = match.group(1) # hostname prefix
84+
ranges_str = match.group(2) # the range of numbers within brackets
85+
suffix = match.group(4) or "" # optional suffix
9086

91-
# if the node numbers contain leading zeros, store them to be
92-
# prepended in the final list
93-
final_list = []
94-
lead_zeros = 0
95-
lead_zeros_str = ''
96-
for elem in num_list:
97-
# if it is a range of numbers, break it by the hyphen and
98-
# create a list
99-
#
100-
# will then be merged with final list
101-
if '-' in elem:
102-
tmp_list = elem.replace("-", ",").split(",")
103-
104-
for digit in tmp_list[0]:
105-
if digit == '0':
106-
lead_zeros = lead_zeros + 1
107-
lead_zeros_str = lead_zeros_str + '0'
108-
109-
rng_list = range(int(tmp_list[0]), int(tmp_list[1]) + 1)
110-
final_list.extend(rng_list)
111-
else:
112-
final_list.append(int(elem))
113-
114-
# put final list in ascending order and append cluster name to
115-
# each node number
116-
final_list.sort()
117-
118-
# prepend leading zeros to numbers required
119-
hostlist_tmp = []
120-
for elem in final_list:
121-
if ((lead_zeros > 0) and (len(str(elem)) <= len(lead_zeros_str))):
122-
hostlist_tmp.append(str(elem).zfill(lead_zeros + 1))
87+
final_hostlist = []
88+
ranges = ranges_str.split(',')
89+
90+
for rng in ranges:
91+
if '-' in rng:
92+
start, end = rng.split('-')
93+
# handle leading zeros
94+
width = len(start)
95+
start, end = int(start), int(end)
96+
for i in range(start, end + 1):
97+
final_hostlist.append(f"{prefix}{str(i).zfill(width)}{suffix}")
12398
else:
124-
hostlist_tmp.append(str(elem))
99+
final_hostlist.append(f"{prefix}{rng.zfill(len(rng))}{suffix}")
125100

126-
# append hostname to the node numbers
127-
hostlist_no_suffix = []
128-
for elem in hostlist_tmp:
129-
hostlist_no_suffix.append(match.group(1) + elem)
101+
# append sorted hostnames
102+
result_hostlist.extend(final_hostlist)
130103

131-
# append suffix to hostlist if there is one
132-
final_hostlist = []
133-
for elem in hostlist_no_suffix:
134-
final_hostlist.append(elem + match.group(4))
104+
# sort the entire hostlist numerically based on the numeric part
105+
def numeric_key(hostname):
106+
return int(re.search(r'(\d+)', hostname).group(1))
135107

136-
result_hostlist.append('%s' % ','.join(map(str, final_hostlist)))
108+
result_hostlist.sort(key=numeric_key)
137109

138110
return ','.join(result_hostlist)
139111

hostlist/unittest_hostlist.py

+6
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,12 @@ def test_expand(self):
2020
test = hl.expand('quartz[4-8]')
2121
self.assertEqual(test, expected)
2222

23+
# expand() with the prompt in issue#32
24+
def test_expand_leading_zeros(self):
25+
expected = 's02p017,s02p029,s02p030'
26+
test = hl.expand('s02p[017,029-030]')
27+
self.assertEqual(test, expected)
28+
2329
# expand() will also return correctly with
2430
# multiple sets of ranges
2531
def test_expand_multi_range(self):

0 commit comments

Comments
 (0)