2014-02-04 00:08:37 +00:00
|
|
|
import redis
|
|
|
|
import json
|
2015-09-30 19:23:19 +00:00
|
|
|
import time
|
2014-02-04 00:08:37 +00:00
|
|
|
|
2014-05-09 22:45:11 +00:00
|
|
|
from util.dynamic import import_class
|
2014-09-12 17:13:14 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
|
|
|
|
|
|
|
|
ONE_DAY = timedelta(days=1)
|
2016-06-20 15:51:30 +00:00
|
|
|
SEVEN_DAYS = timedelta(days=7)
|
2014-09-12 17:13:14 +00:00
|
|
|
|
2014-05-09 22:45:11 +00:00
|
|
|
|
2014-03-28 18:20:06 +00:00
|
|
|
class BuildStatusRetrievalError(Exception):
|
|
|
|
pass
|
2014-02-04 00:08:37 +00:00
|
|
|
|
2014-05-09 22:45:11 +00:00
|
|
|
class RedisBuildLogs(object):
|
2014-02-12 23:58:40 +00:00
|
|
|
ERROR = 'error'
|
|
|
|
COMMAND = 'command'
|
|
|
|
PHASE = 'phase'
|
|
|
|
|
2014-10-01 18:16:42 +00:00
|
|
|
def __init__(self, redis_config):
|
2015-10-22 17:24:56 +00:00
|
|
|
args = dict(redis_config)
|
2015-11-06 23:48:47 +00:00
|
|
|
args.update({'socket_connect_timeout': 5, 'socket_timeout': 5})
|
2015-10-22 17:24:56 +00:00
|
|
|
|
|
|
|
self._redis_config = redis_config
|
|
|
|
self._redis = redis.StrictRedis(**args)
|
2014-02-04 00:08:37 +00:00
|
|
|
|
|
|
|
@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.
|
|
|
|
"""
|
2016-11-18 16:47:16 +00:00
|
|
|
pipeline = self._redis.pipeline(transaction=False)
|
|
|
|
pipeline.expire(self._logs_key(build_id), SEVEN_DAYS)
|
|
|
|
pipeline.rpush(self._logs_key(build_id), json.dumps(log_obj))
|
|
|
|
result = pipeline.execute()
|
|
|
|
return result[1]
|
2014-02-04 00:08:37 +00:00
|
|
|
|
2014-08-19 00:34:39 +00:00
|
|
|
def append_log_message(self, build_id, log_message, log_type=None, log_data=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-08-19 00:34:39 +00:00
|
|
|
if log_data:
|
|
|
|
log_obj['data'] = log_data
|
|
|
|
|
2016-06-20 15:51:30 +00:00
|
|
|
return self.append_log_entry(build_id, 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-11-24 21:07:38 +00:00
|
|
|
"""
|
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))
|
2016-06-20 15:20:17 +00:00
|
|
|
except redis.RedisError as re:
|
|
|
|
raise BuildStatusRetrievalError('Cannot retrieve build logs: %s' % re)
|
2014-02-04 00:08:37 +00:00
|
|
|
|
2016-06-20 15:51:30 +00:00
|
|
|
def expire_status(self, build_id):
|
|
|
|
"""
|
|
|
|
Sets the status entry to expire in 1 day.
|
|
|
|
"""
|
|
|
|
self._redis.expire(self._status_key(build_id), ONE_DAY)
|
|
|
|
|
2014-09-12 17:13:14 +00:00
|
|
|
def expire_log_entries(self, build_id):
|
2014-09-11 19:33:10 +00:00
|
|
|
"""
|
2014-09-12 17:13:14 +00:00
|
|
|
Sets the log entry to expire in 1 day.
|
2014-09-11 19:33:10 +00:00
|
|
|
"""
|
2014-09-12 17:13:14 +00:00
|
|
|
self._redis.expire(self._logs_key(build_id), ONE_DAY)
|
2014-09-11 19:33:10 +00:00
|
|
|
|
2017-03-17 14:35:47 +00:00
|
|
|
def delete_log_entries(self, build_id):
|
|
|
|
"""
|
|
|
|
Delete the log entry
|
|
|
|
"""
|
|
|
|
self._redis.delete(self._logs_key(build_id))
|
|
|
|
|
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.
|
|
|
|
"""
|
2016-06-20 15:51:30 +00:00
|
|
|
self._redis.set(self._status_key(build_id), json.dumps(status_obj), ex=SEVEN_DAYS)
|
2014-02-04 00:08:37 +00:00
|
|
|
|
|
|
|
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))
|
2016-06-20 15:20:17 +00:00
|
|
|
except redis.RedisError as re:
|
|
|
|
raise BuildStatusRetrievalError('Cannot retrieve build status: %s' % re)
|
2014-03-25 17:29:06 +00:00
|
|
|
|
2014-02-04 00:08:37 +00:00
|
|
|
return json.loads(fetched) if fetched else None
|
2014-05-09 22:45:11 +00:00
|
|
|
|
2015-09-30 19:23:19 +00:00
|
|
|
@staticmethod
|
|
|
|
def _health_key():
|
|
|
|
return '_health'
|
|
|
|
|
2014-05-13 22:53:42 +00:00
|
|
|
def check_health(self):
|
|
|
|
try:
|
2015-10-22 17:24:56 +00:00
|
|
|
args = dict(self._redis_config)
|
|
|
|
args.update({'socket_connect_timeout': 1, 'socket_timeout': 1})
|
|
|
|
|
|
|
|
connection = redis.StrictRedis(**args)
|
2017-11-10 20:46:09 +00:00
|
|
|
if not connection.ping():
|
2017-05-11 04:05:14 +00:00
|
|
|
return (False, 'Could not ping redis')
|
2015-09-30 19:23:19 +00:00
|
|
|
|
|
|
|
# Ensure we can write and read a key.
|
2015-10-22 17:24:56 +00:00
|
|
|
connection.set(self._health_key(), time.time())
|
|
|
|
connection.get(self._health_key())
|
2017-05-11 04:05:14 +00:00
|
|
|
return (True, None)
|
|
|
|
except redis.RedisError as re:
|
|
|
|
return (False, 'Could not connect to redis: %s' % re.message)
|
2014-05-13 22:53:42 +00:00
|
|
|
|
|
|
|
|
2014-05-09 22:45:11 +00:00
|
|
|
class BuildLogs(object):
|
|
|
|
def __init__(self, app=None):
|
|
|
|
self.app = app
|
|
|
|
if app is not None:
|
|
|
|
self.state = self.init_app(app)
|
|
|
|
else:
|
|
|
|
self.state = None
|
|
|
|
|
|
|
|
def init_app(self, app):
|
2014-10-01 18:16:42 +00:00
|
|
|
buildlogs_config = app.config.get('BUILDLOGS_REDIS')
|
|
|
|
if not buildlogs_config:
|
|
|
|
# This is the old key name.
|
|
|
|
buildlogs_config = {
|
|
|
|
'host': app.config.get('BUILDLOGS_REDIS_HOSTNAME')
|
|
|
|
}
|
|
|
|
|
2014-05-09 22:45:11 +00:00
|
|
|
buildlogs_options = app.config.get('BUILDLOGS_OPTIONS', [])
|
|
|
|
buildlogs_import = app.config.get('BUILDLOGS_MODULE_AND_CLASS', None)
|
|
|
|
|
|
|
|
if buildlogs_import is None:
|
|
|
|
klass = RedisBuildLogs
|
|
|
|
else:
|
|
|
|
klass = import_class(buildlogs_import[0], buildlogs_import[1])
|
|
|
|
|
2014-10-01 18:16:42 +00:00
|
|
|
buildlogs = klass(buildlogs_config, *buildlogs_options)
|
2014-05-09 22:45:11 +00:00
|
|
|
|
|
|
|
# register extension with app
|
|
|
|
app.extensions = getattr(app, 'extensions', {})
|
|
|
|
app.extensions['buildlogs'] = buildlogs
|
|
|
|
return buildlogs
|
|
|
|
|
|
|
|
def __getattr__(self, name):
|
2014-08-19 00:34:39 +00:00
|
|
|
return getattr(self.state, name, None)
|