X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/8f9f0dece977ccf5a778f3b3bd2379375e723c18..e0f940ec72a9ed000185196f8d01419302b3cb59:/services/nodemanager/arvnodeman/computenode/__init__.py diff --git a/services/nodemanager/arvnodeman/computenode/__init__.py b/services/nodemanager/arvnodeman/computenode/__init__.py index 4955992faa..6e46bc0f4c 100644 --- a/services/nodemanager/arvnodeman/computenode/__init__.py +++ b/services/nodemanager/arvnodeman/computenode/__init__.py @@ -2,20 +2,47 @@ from __future__ import absolute_import, print_function +import calendar import itertools +import re import time +ARVADOS_TIMEFMT = '%Y-%m-%dT%H:%M:%SZ' +ARVADOS_TIMESUBSEC_RE = re.compile(r'(\.\d+)Z$') + def arvados_node_fqdn(arvados_node, default_hostname='dynamic.compute'): hostname = arvados_node.get('hostname') or default_hostname return '{}.{}'.format(hostname, arvados_node['domain']) def arvados_node_mtime(node): - return time.mktime(time.strptime(node['modified_at'] + 'UTC', - '%Y-%m-%dT%H:%M:%SZ%Z')) - time.timezone + return arvados_timestamp(node['modified_at']) + +def arvados_timestamp(timestr): + subsec_match = ARVADOS_TIMESUBSEC_RE.search(timestr) + if subsec_match is None: + subsecs = .0 + else: + subsecs = float(subsec_match.group(1)) + timestr = timestr[:subsec_match.start()] + 'Z' + return calendar.timegm(time.strptime(timestr + 'UTC', + ARVADOS_TIMEFMT + '%Z')) def timestamp_fresh(timestamp, fresh_time): return (time.time() - timestamp) < fresh_time +def arvados_node_missing(arvados_node, fresh_time): + """Indicate if cloud node corresponding to the arvados + node is "missing". + + If True, this means the node has not pinged the API server within the timeout + period. If False, the ping is up to date. If the node has never pinged, + returns None. + """ + if arvados_node["last_ping_at"] is None: + return None + else: + return not timestamp_fresh(arvados_timestamp(arvados_node["last_ping_at"]), fresh_time) + class ShutdownTimer(object): """Keep track of a cloud node's shutdown windows.