2014-02-04 00:08:37 +00:00
|
|
|
import redis
|
|
|
|
import json
|
|
|
|
|
2014-03-28 18:20:06 +00:00
|
|
|
class BuildStatusRetrievalError(Exception):
|
|
|
|
pass
|
2014-02-04 00:08:37 +00:00
|
|
|
|
|
|
|
class BuildLogs(object):
|
2014-02-12 23:58:40 +00:00
|
|
|
ERROR = 'error'
|
|
|
|
COMMAND = 'command'
|
|
|
|
PHASE = 'phase'
|
|
|
|
|
2014-02-04 00:08:37 +00:00
|
|
|
def __init__(self, redis_host):
|
|
|
|
self._redis = redis.StrictRedis(host=redis_host)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _logs_key(build_id):
|
|
|
|
return 'builds/%s/logs' % build_id
|
|
|
|
|
|
|
|
def append_log_entry(self, build_id, log_obj):
|
|
|
|
"""
|
|
|
|
Appends the serialized form of log_obj to the end of the log entry list
|
|
|
|
and returns the new length of the list.
|
|
|
|
"""
|
|
|
|
return self._redis.rpush(self._logs_key(build_id), json.dumps(log_obj))
|
|
|
|
|
2014-02-12 23:58:40 +00:00
|
|
|
def append_log_message(self, build_id, log_message, log_type=None):
|
2014-02-04 00:08:37 +00:00
|
|
|
"""
|
|
|
|
Wraps the message in an envelope and push it to the end of the log entry
|
2014-02-11 00:12:43 +00:00
|
|
|
list and returns the index at which it was inserted.
|
2014-02-04 00:08:37 +00:00
|
|
|
"""
|
|
|
|
log_obj = {
|
|
|
|
'message': log_message
|
|
|
|
}
|
2014-02-11 00:12:43 +00:00
|
|
|
|
2014-02-12 23:58:40 +00:00
|
|
|
if log_type:
|
|
|
|
log_obj['type'] = log_type
|
2014-02-11 00:12:43 +00:00
|
|
|
|
2014-02-12 23:58:40 +00:00
|
|
|
return self._redis.rpush(self._logs_key(build_id), json.dumps(log_obj)) - 1
|
2014-02-04 00:08:37 +00:00
|
|
|
|
2014-02-12 23:58:40 +00:00
|
|
|
def get_log_entries(self, build_id, start_index):
|
2014-02-04 00:08:37 +00:00
|
|
|
"""
|
|
|
|
Returns a tuple of the current length of the list and an iterable of the
|
2014-02-12 23:58:40 +00:00
|
|
|
requested log entries.
|
2014-02-04 00:08:37 +00:00
|
|
|
"""
|
2014-03-25 17:29:06 +00:00
|
|
|
try:
|
|
|
|
llen = self._redis.llen(self._logs_key(build_id))
|
|
|
|
log_entries = self._redis.lrange(self._logs_key(build_id), start_index, -1)
|
|
|
|
return (llen, (json.loads(entry) for entry in log_entries))
|
|
|
|
except redis.ConnectionError:
|
2014-03-28 18:20:06 +00:00
|
|
|
raise BuildStatusRetrievalError('Cannot retrieve build logs')
|
2014-02-04 00:08:37 +00:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def _status_key(build_id):
|
|
|
|
return 'builds/%s/status' % build_id
|
|
|
|
|
|
|
|
def set_status(self, build_id, status_obj):
|
|
|
|
"""
|
|
|
|
Sets the status key for this build to json serialized form of the supplied
|
|
|
|
obj.
|
|
|
|
"""
|
|
|
|
self._redis.set(self._status_key(build_id), json.dumps(status_obj))
|
|
|
|
|
|
|
|
def get_status(self, build_id):
|
|
|
|
"""
|
|
|
|
Loads the status information for the specified build id.
|
|
|
|
"""
|
2014-03-25 17:29:06 +00:00
|
|
|
try:
|
|
|
|
fetched = self._redis.get(self._status_key(build_id))
|
|
|
|
except redis.ConnectionError:
|
2014-03-28 18:20:06 +00:00
|
|
|
raise BuildStatusRetrievalError('Cannot retrieve build status')
|
2014-03-25 17:29:06 +00:00
|
|
|
|
2014-02-04 00:08:37 +00:00
|
|
|
return json.loads(fetched) if fetched else None
|