Fixed bugs and improved logging in config recommendation tasks/views
This commit is contained in:
parent
3786714093
commit
9393fb7aca
|
@ -338,7 +338,6 @@ def get_result(max_time_sec=180, interval_sec=5, upload_code=None):
|
|||
url = dconf.WEBSITE_URL + '/query_and_get/' + upload_code
|
||||
elapsed = 0
|
||||
response_dict = None
|
||||
response = ''
|
||||
rout = ''
|
||||
|
||||
while elapsed <= max_time_sec:
|
||||
|
@ -347,8 +346,9 @@ def get_result(max_time_sec=180, interval_sec=5, upload_code=None):
|
|||
assert response != 'null'
|
||||
rout = json.dumps(response, indent=4) if isinstance(response, dict) else response
|
||||
|
||||
LOG.debug('%s [status code: %d, content_type: %s, elapsed: %ds]', rout,
|
||||
rsp.status_code, rsp.headers.get('Content-Type', ''), elapsed)
|
||||
LOG.debug('%s\n\n[status code: %d, type(response): %s, elapsed: %ds, %s]', rout,
|
||||
rsp.status_code, type(response), elapsed,
|
||||
', '.join(['{}: {}'.format(k, v) for k, v in rsp.headers.items()]))
|
||||
|
||||
if rsp.status_code == 200:
|
||||
# Success
|
||||
|
@ -368,12 +368,12 @@ def get_result(max_time_sec=180, interval_sec=5, upload_code=None):
|
|||
|
||||
elif rsp.status_code == 500:
|
||||
# Failure
|
||||
if len(response) > 5000:
|
||||
with open('error.html', 'w') as f:
|
||||
f.write(response)
|
||||
msg = "Saved HTML error to 'error.html'."
|
||||
else:
|
||||
msg = rout
|
||||
if isinstance(response, str):
|
||||
savepath = os.path.join(dconf.LOG_DIR, 'error.html')
|
||||
with open(savepath, 'w') as f:
|
||||
f.write(response)
|
||||
msg = "Saved HTML error to '{}'.".format(os.path.relpath(savepath))
|
||||
raise Exception(
|
||||
"Failed to download the next config.\nStatus code: {}\nMessage: {}\n".format(
|
||||
rsp.status_code, msg))
|
||||
|
@ -385,12 +385,12 @@ def get_result(max_time_sec=180, interval_sec=5, upload_code=None):
|
|||
if not response_dict:
|
||||
assert elapsed > max_time_sec, \
|
||||
'response={} but elapsed={}s <= max_time={}s'.format(
|
||||
response, elapsed, max_time_sec)
|
||||
rout, elapsed, max_time_sec)
|
||||
raise Exception(
|
||||
'Failed to download the next config in {}s: {} (elapsed: {}s)'.format(
|
||||
max_time_sec, rout, elapsed))
|
||||
|
||||
LOG.info('Downloaded the next config in %ds: %s', elapsed, rout)
|
||||
LOG.info('Downloaded the next config in %ds', elapsed)
|
||||
|
||||
return response_dict
|
||||
|
||||
|
@ -792,3 +792,5 @@ def integration_tests():
|
|||
upload_code='ottertuneTestTuningGPR')
|
||||
response = get_result(upload_code='ottertuneTestTuningGPR')
|
||||
assert response['status'] == 'good'
|
||||
|
||||
LOG.info("\n\nIntegration Tests: PASSED!!\n")
|
||||
|
|
|
@ -83,58 +83,56 @@ class TaskUtilTest(TestCase):
|
|||
# FIXME: Actually setup celery tasks instead of a dummy class?
|
||||
test_tasks = []
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks)
|
||||
self.assertTrue(status is None and num_complete == 0)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks, 1)
|
||||
self.assertTrue(status is 'UNAVAILABLE' and num_complete == 0)
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks, 0)
|
||||
self.assertTrue(status is 'UNAVAILABLE' and num_complete == 0)
|
||||
|
||||
test_tasks2 = [VarType() for i in range(5)]
|
||||
for task in test_tasks2:
|
||||
task.status = "SUCCESS"
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks2)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks2, 5)
|
||||
self.assertTrue(status == "SUCCESS" and num_complete == 5)
|
||||
|
||||
test_tasks3 = test_tasks2
|
||||
test_tasks3[3].status = "FAILURE"
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks3)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks3, 5)
|
||||
self.assertTrue(status == "FAILURE" and num_complete == 3)
|
||||
|
||||
test_tasks4 = test_tasks3
|
||||
test_tasks4[2].status = "REVOKED"
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks4)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks4, 5)
|
||||
self.assertTrue(status == "REVOKED" and num_complete == 2)
|
||||
|
||||
test_tasks5 = test_tasks4
|
||||
test_tasks5[1].status = "RETRY"
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks5)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks5, 5)
|
||||
self.assertTrue(status == "RETRY" and num_complete == 1)
|
||||
|
||||
test_tasks6 = [VarType() for i in range(10)]
|
||||
for i, task in enumerate(test_tasks6):
|
||||
task.status = "PENDING" if i % 2 == 0 else "SUCCESS"
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks6)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks6, 10)
|
||||
self.assertTrue(status == "PENDING" and num_complete == 5)
|
||||
|
||||
test_tasks7 = test_tasks6
|
||||
test_tasks7[9].status = "STARTED"
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks7)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks7, 10)
|
||||
self.assertTrue(status == "STARTED" and num_complete == 4)
|
||||
|
||||
test_tasks8 = test_tasks7
|
||||
test_tasks8[9].status = "RECEIVED"
|
||||
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks8)
|
||||
(status, num_complete) = TaskUtil.get_task_status(test_tasks8, 10)
|
||||
self.assertTrue(status == "RECEIVED" and num_complete == 4)
|
||||
|
||||
with self.assertRaises(Exception):
|
||||
test_tasks9 = [VarType() for i in range(1)]
|
||||
test_tasks9[0].status = "attemped"
|
||||
TaskUtil.get_task_status(test_tasks9)
|
||||
|
||||
|
||||
class DataUtilTest(TestCase):
|
||||
|
||||
|
|
|
@ -28,7 +28,7 @@ from analysis.constraints import ParamConstraintHelper
|
|||
from website.models import (PipelineData, PipelineRun, Result, Workload, KnobCatalog, SessionKnob,
|
||||
MetricCatalog)
|
||||
from website import db
|
||||
from website.types import PipelineTaskType, AlgorithmType
|
||||
from website.types import PipelineTaskType, AlgorithmType, VarType
|
||||
from website.utils import DataUtil, JSONUtil
|
||||
from website.settings import IMPORTANT_KNOB_NUMBER, NUM_SAMPLES, TOP_NUM_CONFIG # pylint: disable=no-name-in-module
|
||||
from website.settings import (USE_GPFLOW, DEFAULT_LENGTH_SCALE, DEFAULT_MAGNITUDE,
|
||||
|
@ -47,7 +47,6 @@ from website.settings import (USE_GPFLOW, DEFAULT_LENGTH_SCALE, DEFAULT_MAGNITUD
|
|||
GPR_MODEL_NAME, ENABLE_DUMMY_ENCODER, DNN_GD_ITER)
|
||||
|
||||
from website.settings import INIT_FLIP_PROB, FLIP_PROB_DECAY
|
||||
from website.types import VarType
|
||||
|
||||
|
||||
LOG = get_task_logger(__name__)
|
||||
|
@ -116,32 +115,59 @@ class ConfigurationRecommendation(UpdateTask): # pylint: disable=abstract-metho
|
|||
|
||||
def clean_knob_data(knob_matrix, knob_labels, session):
|
||||
# Makes sure that all knobs in the dbms are included in the knob_matrix and knob_labels
|
||||
knob_cat = SessionKnob.objects.get_knobs_for_session(session)
|
||||
knob_cat = [knob["name"] for knob in knob_cat if knob["tunable"]]
|
||||
matrix = np.array(knob_matrix)
|
||||
missing_columns = set(knob_cat) - set(knob_labels)
|
||||
unused_columns = set(knob_labels) - set(knob_cat)
|
||||
LOG.debug("clean_knob_data added %d knobs and removed %d knobs.", len(missing_columns),
|
||||
len(unused_columns))
|
||||
# If columns are missing from the matrix
|
||||
if missing_columns:
|
||||
for knob in missing_columns:
|
||||
knob_object = KnobCatalog.objects.get(dbms=session.dbms, name=knob, tunable=True)
|
||||
index = knob_cat.index(knob)
|
||||
knob_matrix = np.array(knob_matrix)
|
||||
session_knobs = SessionKnob.objects.get_knobs_for_session(session)
|
||||
knob_cat = [k['name'] for k in session_knobs]
|
||||
|
||||
if knob_cat == knob_labels:
|
||||
# Nothing to do!
|
||||
return knob_matrix, knob_labels
|
||||
|
||||
LOG.info("session_knobs: %s, knob_labels: %s, missing: %s, extra: %s", len(knob_cat),
|
||||
len(knob_labels), len(set(knob_cat) - set(knob_labels)),
|
||||
len(set(knob_labels) - set(knob_cat)))
|
||||
|
||||
nrows = knob_matrix.shape[0] # pylint: disable=unsubscriptable-object
|
||||
new_labels = []
|
||||
new_columns = []
|
||||
|
||||
for knob in session_knobs:
|
||||
knob_name = knob['name']
|
||||
if knob_name not in knob_labels:
|
||||
# Add missing column initialized to knob's default value
|
||||
default_val = knob['default']
|
||||
try:
|
||||
default_val = float(knob_object.default)
|
||||
if knob['vartype'] == VarType.ENUM:
|
||||
default_val = knob['enumvals'].split(',').index(default_val)
|
||||
elif knob['vartype'] == VarType.BOOL:
|
||||
default_val = str(default_val).lower() in ("on", "true", "yes", "0")
|
||||
else:
|
||||
default_val = float(default_val)
|
||||
except ValueError:
|
||||
default_val = 0
|
||||
matrix = np.insert(matrix, index, default_val, axis=1)
|
||||
knob_labels.insert(index, knob)
|
||||
# If they are useless columns in the matrix
|
||||
if unused_columns:
|
||||
indexes = [i for i, n in enumerate(knob_labels) if n in unused_columns]
|
||||
# Delete unused columns
|
||||
matrix = np.delete(matrix, indexes, 1)
|
||||
for i in sorted(indexes, reverse=True):
|
||||
del knob_labels[i]
|
||||
return matrix, knob_labels
|
||||
LOG.exception("Error parsing knob default: %s=%s", knob_name, default_val)
|
||||
new_col = np.ones((nrows, 1), dtype=float) * default_val
|
||||
new_lab = knob_name
|
||||
else:
|
||||
index = knob_labels.index(knob_name)
|
||||
new_col = knob_matrix[:, index]
|
||||
new_lab = knob_labels[index]
|
||||
|
||||
new_labels.append(new_lab)
|
||||
new_columns.append(new_col)
|
||||
|
||||
new_matrix = np.hstack(new_columns).reshape(nrows, -1)
|
||||
LOG.debug("Cleaned matrix: %s, knobs (%s): %s", new_matrix.shape,
|
||||
len(new_labels), new_labels)
|
||||
|
||||
assert new_labels == knob_cat, \
|
||||
"Expected knobs: {}\nActual knobs: {}\n".format(
|
||||
knob_cat, new_labels)
|
||||
assert new_matrix.shape == (nrows, len(knob_cat)), \
|
||||
"Expected shape: {}, Actual shape: {}".format(
|
||||
(nrows, len(knob_cat)), new_matrix.shape)
|
||||
|
||||
return new_matrix, new_labels
|
||||
|
||||
|
||||
def clean_metric_data(metric_matrix, metric_labels, session):
|
||||
|
@ -438,8 +464,7 @@ def configuration_recommendation_ddpg(result_info): # pylint: disable=invalid-n
|
|||
metric_data = metric_data.flatten()
|
||||
metric_scalar = MinMaxScaler().fit(metric_data.reshape(1, -1))
|
||||
normalized_metric_data = metric_scalar.transform(metric_data.reshape(1, -1))[0]
|
||||
cleaned_knob_data = clean_knob_data(agg_data['X_matrix'], agg_data['X_columnlabels'],
|
||||
session)
|
||||
cleaned_knob_data = clean_knob_data(agg_data['X_matrix'], agg_data['X_columnlabels'], session)
|
||||
knob_labels = np.array(cleaned_knob_data[1]).flatten()
|
||||
knob_num = len(knob_labels)
|
||||
metric_num = len(metric_data)
|
||||
|
@ -834,7 +859,6 @@ def map_workload(map_workload_input):
|
|||
knob_data["data"], knob_data["columnlabels"] = clean_knob_data(knob_data["data"],
|
||||
knob_data["columnlabels"],
|
||||
newest_result.session)
|
||||
|
||||
metric_data = load_data_helper(pipeline_data, unique_workload, PipelineTaskType.METRIC_DATA)
|
||||
X_matrix = np.array(knob_data["data"])
|
||||
y_matrix = np.array(metric_data["data"])
|
||||
|
|
|
@ -12,9 +12,13 @@
|
|||
<td style="width: 50%"><div class="text-right">{{ labels.id }}</div></td>
|
||||
<td style="width: 50%">{{ result.pk }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td><div class="text-right">{{ labels.creation_time }}</div></td>
|
||||
<td>{{ result.creation_time }}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td><div class="text-right">{{ labels.session }}</div></td>
|
||||
<td>{{ result.session.name }}</td>
|
||||
<td><a href="{% url 'session' project_id session_id %}">{{ result.session.name }}</a></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td><div class="text-right">{{ labels.workload }}</div></td>
|
||||
|
@ -28,10 +32,6 @@
|
|||
<td><div class="text-top text-right">{{ labels.metric_data }}</div></td>
|
||||
<td><a href="{% url 'metric_data' project_id session_id result.metric_data.pk %}">{{ result.metric_data.name }}</a></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td><div class="text-right">{{ labels.creation_time }}</div></td>
|
||||
<td>{{ result.creation_time }}</td>
|
||||
</tr>
|
||||
{% for key, value in default_metrics.items %}
|
||||
<tr>
|
||||
<td><div class="text-right">{{ metric_meta|get_item:key|get_attr:"pprint" }}</div></td>
|
||||
|
@ -47,8 +47,16 @@
|
|||
{% endif %}
|
||||
{% if next_conf_available %}
|
||||
<tr>
|
||||
<td><div class="text-right">{{ labels.next_conf_available }}</div></td>
|
||||
<td><a href="/get_result_data_file/?id={{ result.pk }}&type=next_conf">Download</a></td>
|
||||
<td>
|
||||
<div class="text-right">
|
||||
{{ labels.next_conf }}<br>
|
||||
<button type="submit" style="margin-top:15px;" class="btn btn-default btn-lg alert alert-info"
|
||||
onclick="location.href='/get_result_data_file/?id={{ result.pk }}&type=next_conf'">
|
||||
<span class="glyphicon glyphicon-download-alt" aria-hidden="true"></span>
|
||||
</button>
|
||||
</div>
|
||||
</td>
|
||||
<td><pre style="display:inline-block">{{ next_conf }}</pre></td>
|
||||
</tr>
|
||||
{% endif %}
|
||||
</tbody>
|
||||
|
|
|
@ -82,11 +82,8 @@ class TaskUtil(object):
|
|||
return TaskMeta.objects.filter(task_id__in=task_ids).order_by(preserved)
|
||||
|
||||
@staticmethod
|
||||
def get_task_status(tasks):
|
||||
if not tasks:
|
||||
return None, 0
|
||||
|
||||
overall_status = 'SUCCESS'
|
||||
def get_task_status(tasks, num_tasks):
|
||||
overall_status = 'UNAVAILABLE'
|
||||
num_completed = 0
|
||||
for task in tasks:
|
||||
status = task.status
|
||||
|
@ -101,6 +98,9 @@ class TaskUtil(object):
|
|||
task.id, status, task.task_id)
|
||||
overall_status = status
|
||||
|
||||
if num_tasks > 0 and num_tasks == num_completed:
|
||||
overall_status = 'SUCCESS'
|
||||
|
||||
return overall_status, num_completed
|
||||
|
||||
|
||||
|
@ -484,3 +484,13 @@ def delete_user(username):
|
|||
deleted = False
|
||||
|
||||
return delete_info, deleted
|
||||
|
||||
|
||||
def model_to_dict2(m, exclude=None):
|
||||
exclude = exclude or []
|
||||
d = {}
|
||||
for f in m._meta.fields: # pylint: disable=protected-access
|
||||
fname = f.name
|
||||
if fname not in exclude:
|
||||
d[fname] = getattr(m, fname, None)
|
||||
return d
|
||||
|
|
|
@ -398,18 +398,33 @@ def result_view(request, project_id, session_id, result_id):
|
|||
# default_metrics = {mname: metric_data[mname] * metric_meta[mname].scale
|
||||
# for mname in default_metrics}
|
||||
|
||||
status = None
|
||||
if target.task_ids is not None:
|
||||
tasks = TaskUtil.get_tasks(target.task_ids)
|
||||
status, _ = TaskUtil.get_task_status(tasks)
|
||||
if status is None:
|
||||
status = 'UNAVAILABLE'
|
||||
task_ids = [t for t in (target.task_ids or '').split(',') if t.strip() != '']
|
||||
tasks = TaskUtil.get_tasks(task_ids)
|
||||
status, _ = TaskUtil.get_task_status(tasks, len(task_ids))
|
||||
|
||||
next_conf_available = True if status == 'SUCCESS' else False
|
||||
next_conf = ''
|
||||
cfg = target.next_configuration
|
||||
LOG.debug("status: %s, next_conf_available: %s, next_conf: %s, type: %s",
|
||||
status, next_conf_available, cfg, type(cfg))
|
||||
|
||||
if next_conf_available:
|
||||
try:
|
||||
cfg = JSONUtil.loads(cfg)['recommendation']
|
||||
kwidth = max(len(k) for k in cfg.keys())
|
||||
vwidth = max(len(str(v)) for v in cfg.values())
|
||||
next_conf = ''
|
||||
for k, v in cfg.items():
|
||||
next_conf += '{: <{kwidth}} = {: >{vwidth}}\n'.format(
|
||||
k, v, kwidth=kwidth, vwidth=vwidth)
|
||||
except Exception as e: # pylint: disable=broad-except
|
||||
LOG.exception("Failed to format the next config (type=%s): %s.\n\n%s\n",
|
||||
type(cfg), cfg, e)
|
||||
|
||||
form_labels = Result.get_labels()
|
||||
form_labels.update(LabelUtil.style_labels({
|
||||
'status': 'status',
|
||||
'next_conf_available': 'next configuration'
|
||||
'next_conf': 'next configuration',
|
||||
}))
|
||||
form_labels['title'] = 'Result Info'
|
||||
context = {
|
||||
|
@ -417,6 +432,7 @@ def result_view(request, project_id, session_id, result_id):
|
|||
'metric_meta': metric_meta,
|
||||
'status': status,
|
||||
'next_conf_available': next_conf_available,
|
||||
'next_conf': next_conf,
|
||||
'labels': form_labels,
|
||||
'project_id': project_id,
|
||||
'session_id': session_id
|
||||
|
@ -843,14 +859,15 @@ def download_debug_info(request, project_id, session_id): # pylint: disable=unu
|
|||
def tuner_status_view(request, project_id, session_id, result_id): # pylint: disable=unused-argument
|
||||
res = Result.objects.get(pk=result_id)
|
||||
|
||||
tasks = TaskUtil.get_tasks(res.task_ids)
|
||||
task_ids = [t for t in (res.task_ids or '').split(',') if t.strip() != '']
|
||||
tasks = TaskUtil.get_tasks(task_ids)
|
||||
|
||||
overall_status, num_completed = TaskUtil.get_task_status(tasks)
|
||||
if overall_status in ['PENDING', 'RECEIVED', 'STARTED', None]:
|
||||
overall_status, num_completed = TaskUtil.get_task_status(tasks, len(task_ids))
|
||||
if overall_status in ['PENDING', 'RECEIVED', 'STARTED', 'UNAVAILABLE']:
|
||||
completion_time = 'N/A'
|
||||
total_runtime = 'N/A'
|
||||
else:
|
||||
completion_time = tasks[-1].date_done
|
||||
completion_time = tasks.reverse()[0].date_done
|
||||
total_runtime = (completion_time - res.creation_time).total_seconds()
|
||||
total_runtime = '{0:.2f} seconds'.format(total_runtime)
|
||||
|
||||
|
@ -1066,7 +1083,6 @@ def get_timeline_data(request):
|
|||
|
||||
# get the lastest result
|
||||
def give_result(request, upload_code): # pylint: disable=unused-argument
|
||||
|
||||
try:
|
||||
session = Session.objects.get(upload_code=upload_code)
|
||||
except Session.DoesNotExist:
|
||||
|
@ -1074,17 +1090,19 @@ def give_result(request, upload_code): # pylint: disable=unused-argument
|
|||
return HttpResponse("Invalid upload code: " + upload_code, status=400)
|
||||
|
||||
latest_result = Result.objects.filter(session=session).latest('creation_time')
|
||||
tasks = TaskUtil.get_tasks(latest_result.task_ids)
|
||||
overall_status, num_completed = TaskUtil.get_task_status(tasks)
|
||||
response = {
|
||||
'celery_status': overall_status,
|
||||
'result_id': latest_result.pk,
|
||||
'message': '',
|
||||
'errors': [],
|
||||
}
|
||||
task_ids = [t for t in (latest_result.task_ids or '').split(',') if t.strip() != '']
|
||||
tasks = TaskUtil.get_tasks(task_ids)
|
||||
overall_status, num_completed = TaskUtil.get_task_status(tasks, len(task_ids))
|
||||
next_config = latest_result.next_configuration
|
||||
|
||||
LOG.debug("result_id: %s, overall_status: %s, tasks_completed: %s/%s, "
|
||||
"next_config: %s\n", latest_result.pk, overall_status, num_completed,
|
||||
len(task_ids), next_config)
|
||||
|
||||
response = dict(celery_status=overall_status, result_id=latest_result.pk, message='', errors=[])
|
||||
if overall_status == 'SUCCESS':
|
||||
response.update(JSONUtil.loads(latest_result.next_configuration),
|
||||
next_config = JSONUtil.loads(next_config)
|
||||
response.update(next_config,
|
||||
message='Celery successfully recommended the next configuration')
|
||||
status_code = 200
|
||||
|
||||
|
|
Loading…
Reference in New Issue