-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnodestatus.py
executable file
·105 lines (95 loc) · 3.49 KB
/
nodestatus.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
#!/home/dmick/v/bin/python3
import argparse
import datetime
import os
import jenkins
import requests
import sys
import time
jenkins_user=os.environ.get('JENKINS_USER')
jenkins_token=os.environ.get('JENKINS_TOKEN')
j=jenkins.Jenkins('https://jenkins.ceph.com', jenkins_user, jenkins_token)
def parse_args():
ap = argparse.ArgumentParser()
ap.add_argument('-o', '--offline', action='store_true', help='Show only offline nodes')
ap.add_argument('nodes', nargs='*')
return ap.parse_args()
def shortname(s):
if '+' not in s:
return s
return s[s.index('+')+1:]
def main():
args=parse_args()
print("getting nodelist...", end='', file=sys.stderr)
nodes = j.get_nodes()
newnodes = list()
nodes = [n for n in nodes if shortname(n['name']) in args.nodes]
print(f'{len(nodes)} nodes found', file=sys.stderr)
nodetojob = dict()
if not args.offline:
print("getting active builds...", end='', file=sys.stderr, flush=True)
for node in nodes:
name=node['name']
if 'Built' in name:
name='(master)'
nodeinfo = j.get_node_info(name)
nodetojob[name] = dict()
nodetojob[name]['tags'] = \
[t['name'] for t in nodeinfo['assignedLabels'] if '+' not in t['name']]
# node_info(depth=2) is too expensive. Get just the running jobs.
if nodeinfo['offline']:
temp = "temporarily " if nodeinfo['temporarilyOffline'] else ""
cause = ""
if temp:
r = requests.get(f'https://jenkins.ceph.com/computer/{name}/api/json?tree=offlineCause[description,timestamp]')
cause = r.json()['offlineCause']
desc = cause.get('description', '??')
ts = cause.get('timestamp', None)
if ts:
ts = datetime.datetime.fromtimestamp(ts/1000).strftime('%b %d')
hostname = name[name.find('+')+1:]
if temp:
nodetojob[name]['offline'] = f'{temp}offline: {ts} {desc}'
else:
nodetojob[name]['offline'] = 'offline'
continue
print(".", end='', file=sys.stderr, flush=True)
req = f'https://jenkins.ceph.com/computer/{name}/api/json?tree=executors[currentExecutable[url],busyExecutors,idleExecutors]'
start = time.time()
resp = requests.get(req)
end = time.time()
resp.raise_for_status()
builds=resp.json()
for b in builds['executors']:
ce = b['currentExecutable']
if not ce:
continue
if not 'builds' in nodetojob[name]:
nodetojob[name]['builds'] = list()
nodetojob[name]['builds'].append(ce['url'])
if args.offline:
return 0
print(file=sys.stderr)
idlecnt = busycnt = offlinecnt = 0
for k,v in nodetojob.items():
name = k
# chop off IP addr
name = shortname(name)
if 'offline' in v:
print(f'{name}: {v["offline"]} ({",".join(v["tags"])})')
offlinecnt += 1
continue
if 'builds' in v:
buildstr = f'{len(v["builds"])} active builds'
busycnt += 1
else:
buildstr = 'idle'
idlecnt += 1
print(f'{name}: {buildstr} ({",".join(v["tags"])})')
if 'builds' in v:
for b in v['builds']:
print(f'{b}')
print()
print(f'Idle: {idlecnt} Busy: {busycnt} Offline: {offlinecnt}')
if __name__ == '__main__':
sys.exit(main())