blob: eecb577da47cb8622cf2d5f4470a89d00e33c29b [file] [log] [blame]
# <koder>: order imports in usual way
from urlparse import urlparse
from flask import Flask, render_template, url_for, request, g, make_response
from flask_bootstrap import Bootstrap
from config import TEST_PATH
from report import build_vertical_bar, build_lines_chart
from storage_api import builds_list, collect_builds, create_measurement
from logging import getLogger, INFO
import json
import os.path
import math
from web_app.keystone import KeystoneAuth
app = Flask(__name__)
Bootstrap(app)
def get_resource_as_string(name, charset='utf-8'):
with app.open_resource(name) as f:
return f.read().decode(charset)
app.jinja_env.globals['get_resource_as_string'] = get_resource_as_string
def total_lab_info(data):
# <koder>: give 'd' meaningful name
d = {}
d['nodes_count'] = len(data['nodes'])
d['total_memory'] = 0
d['total_disk'] = 0
d['processor_count'] = 0
for node in data['nodes']:
d['total_memory'] += node['memory']['total']
d['processor_count'] += len(node['processors'])
for disk in node['disks']:
d['total_disk'] += disk['size']
to_gb = lambda x: x / (1024 ** 3)
d['total_memory'] = format(to_gb(d['total_memory']), ',d')
d['total_disk'] = format(to_gb(d['total_disk']), ',d')
return d
def collect_lab_data(meta):
u = urlparse(meta['__meta__'])
cred = {"username": "admin", "password": "admin", "tenant_name": "admin"}
keystone = KeystoneAuth(root_url=meta['__meta__'], creds=cred, admin_node_ip=u.hostname)
lab_info = keystone.do(method='get', path="")
nodes = []
result = {}
for node in lab_info:
# <koder>: give p,i,d,... vars meaningful names
d = {}
d['name'] = node['name']
p = []
i = []
disks = []
devices = []
for processor in node['meta']['cpu']['spec']:
p.append(processor)
for iface in node['meta']['interfaces']:
i.append(iface)
m = node['meta']['memory'].copy()
for disk in node['meta']['disks']:
disks.append(disk)
d['memory'] = m
d['disks'] = disks
d['devices'] = devices
d['interfaces'] = i
d['processors'] = p
nodes.append(d)
result['nodes'] = nodes
result['name'] = 'Perf-1 Env'
return result
def merge_builds(b1, b2):
d = {}
for pair in b2.items():
if pair[0] in b1 and type(pair[1]) is list:
b1[pair[0]].extend(pair[1])
else:
b1[pair[0]] = pair[1]
@app.route("/", methods=['GET', 'POST'])
def index():
data = builds_list()
return render_template("index.html", tests=data)
@app.route("/images/<image_name>")
def get_image(image_name):
with open("static/images/" + image_name, 'rb') as f:
image_binary = f.read()
response = make_response(image_binary)
response.headers['Content-Type'] = 'image/png'
response.headers['Content-Disposition'] = 'attachment; filename=img.png'
return response
@app.route("/tests/<test_name>", methods=['GET'])
def render_test(test_name):
tests = []
header_keys = ['build_id', 'iso_md5', 'type', 'date']
table = [[]]
builds = collect_builds()
# <koder>: rename
l = filter(lambda x: x['name'] == test_name, builds)
if l[0]['type'] == 'GA':
builds = filter(lambda x: x['type'] == 'GA', builds)
else:
l.extend(filter(lambda x: x['type'] in ['GA', 'master'] and x not in l, builds))
builds = l
results = {}
# <koder>: magik ip? fixme
meta = {"__meta__": "http://172.16.52.112:8000/api/nodes"}
data = collect_lab_data(meta)
lab_meta = total_lab_info(data)
for build in builds:
# <koder>: don't use name 'type'
type = build['type']
results[type] = create_measurement(build)
urls = build_vertical_bar(results) + build_lines_chart(results)
urls = [url_for("get_image", image_name=os.path.basename(url)) if not url.startswith('http') else url for url in urls]
if len(tests) > 0:
sorted_keys = sorted(tests[0].keys())
for key in sorted_keys:
if key not in header_keys:
header_keys.append(key)
for test in tests:
row = []
for header in header_keys:
if isinstance(test[header], list):
# <koder>: make a named constant from unichr(0x00B1)
# <koder>: use format in this line
row.append(str(test[header][0]) + unichr(0x00B1) + str(test[header][1]))
else:
row.append(test[header])
table.append(row)
return render_template("test.html", urls=urls, table_url=url_for('render_table', test_name=test_name),
index_url=url_for('index'), lab_meta=lab_meta)
@app.route("/tests/table/<test_name>/")
def render_table(test_name):
builds = collect_builds()
l = filter(lambda x: x['name'] == test_name, builds)
if l[0]['type'] == 'GA':
builds = filter(lambda x: x['type'] == 'GA', builds)
else:
l.extend(filter(lambda x: x['type'] in ['GA', 'master'] and x not in l, builds))
builds = l
header_keys = ['build_id', 'iso_md5', 'type' ,'date']
table = [[]]
meta = {"__meta__": "http://172.16.52.112:8000/api/nodes"}
data = collect_lab_data(meta)
if len(builds) > 0:
sorted_keys = sorted(builds[0].keys())
for key in sorted_keys:
if key not in header_keys:
header_keys.append(key)
for test in builds:
row = []
for header in header_keys:
if isinstance(test[header], list):
row.append(str(test[header][0]) + unichr(0x00B1) + str(test[header][1]))
else:
row.append(test[header])
table.append(row)
return render_template("table.html", headers=header_keys, table=table,
back_url=url_for('render_test', test_name=test_name), lab=data)
@app.route("/api/tests/<test_name>", methods=['POST'])
def add_test(test_name):
test = json.loads(request.data)
file_name = TEST_PATH + '/' + 'storage' + ".json"
if not os.path.exists(file_name):
with open(file_name, "w+") as f:
f.write(json.dumps([]))
builds = collect_builds()
res = None
for b in builds:
if b['name'] == test['name']:
res = b
break
if res is None:
builds.append(test)
else:
merge_builds(res, test)
with open(TEST_PATH + '/' + 'storage' + ".json", 'w+') as f:
f.write(json.dumps(builds))
return "Created", 201
@app.route("/api/tests", methods=['GET'])
def get_all_tests():
return json.dumps(collect_builds())
@app.route("/api/tests/<test_name>", methods=['GET'])
def get_test(test_name):
builds = collect_builds()
for build in builds:
if build["type"] == test_name:
return json.dumps(build)
return "Not Found", 404
if __name__ == "__main__":
logger = getLogger("logger")
app.logger.setLevel(INFO)
app.logger.addHandler(logger)
app.run(host='0.0.0.0', debug=True)