/**
* Copyright (c) 2017 Google Inc. All Rights Reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License"); you
* may not use this file except in compliance with the License. You may
* obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
* implied. See the License for the specific language governing
* permissions and limitations under the License.
*/
(function($, moment) {
/**
* Display the log links in a modal window.
* @param linkList A list of [name, url] tuples representing log links.
*/
function showLinks(container, linkList) {
if (!linkList || linkList.length == 0) return;
var logCollection = $('<ul class="collection"></ul>');
var entries = linkList.reduce(function(acc, entry) {
if (!entry || entry.length == 0) return acc;
var link = '<a href="' + entry[1] + '"';
link += 'class="collection-item">' + entry[0] + '</li>';
return acc + link;
}, '');
logCollection.html(entries);
if (container.find('#info-modal').length == 0) {
var modal =
$('<div id="info-modal" class="modal modal-fixed-footer"></div>');
var content = $('<div class="modal-content"></div>');
content.append('<h4>Links</h4>');
content.append('<div class="info-container"></div>');
content.appendTo(modal);
var footer = $('<div class="modal-footer"></div>');
footer.append('<a class="btn-flat modal-close">Close</a></div>');
footer.appendTo(modal);
modal.appendTo(container);
}
var infoContainer = $('#info-modal>.modal-content>.info-container');
infoContainer.empty();
logCollection.appendTo(infoContainer);
$('#info-modal').modal({dismissible: true});
$('#info-modal').modal('open');
}
/**
* Get the nickname for a test case result.
*
* Removes the result prefix and suffix, extracting only the result name.
*
* @param testCaseResult The string name of a VtsReportMessage.TestCaseResult.
* @returns the string nickname of the result.
*/
function getNickname(testCaseResult) {
return testCaseResult.replace('TEST_CASE_RESULT_', '')
.replace('_RESULT', '')
.trim()
.toLowerCase();
}
/**
* Get the badge color from ratio value.
*
* @param the percentage value.
* @returns the string of color for the badge.
*/
function getBadgeColor(ratio) {
var color = "orange";
if (ratio <= 20) {
color = "red";
} else if (ratio >= 70) {
color = "green";
}
return color;
}
/**
* Get the rounded value.
*
* @param the percentage value.
* @returns the rounded value from percentage value.
*/
function getRoundValue(ratio) {
return Math.round(ratio * 1000) / 10;
}
/**
* Display test data in the body beneath a test run's metadata.
* @param container The jquery object in which to insert the test metadata.
* @param data The json object containing the columns to display.
* @param lineHeight The height of each list element.
*/
function displayTestDetails(container, data, lineHeight) {
var nCol = data.length;
var width = 's' + (12 / nCol);
test = container;
var maxLines = 0;
data.forEach(function(column, index) {
if (column.data == undefined || column.name == undefined) {
return;
}
var classes = 'col test-col grey lighten-5 ' + width;
if (index != nCol - 1) {
classes += ' bordered';
}
if (index == 0) {
classes += ' left-most';
}
if (index == nCol - 1) {
classes += ' right-most';
}
var colContainer = $('<div class="' + classes + '"></div>');
var col = $('<div class="test-case-container"></div>');
colContainer.appendTo(container);
var count = column.data.length;
var head = $('<h5 class="test-result-label white"></h5>')
.text(getNickname(column.name))
.appendTo(colContainer)
.css('text-transform', 'capitalize');
$('<div class="indicator right center"></div>')
.text(count)
.addClass(column.name)
.appendTo(head);
col.appendTo(colContainer);
var list = $('<ul></ul>').appendTo(col);
column.data.forEach(function(testCase) {
$('<li></li>')
.text(testCase)
.addClass('test-case')
.css('font-size', lineHeight - 2)
.css('line-height', lineHeight + 'px')
.appendTo(list);
});
if (count > maxLines) {
maxLines = count;
}
});
var containers = container.find('.test-case-container');
containers.height(maxLines * lineHeight);
}
/**
* Click handler for displaying test run details.
* @param e The click event.
*/
function testRunClick(e) {
var header = $(this);
var icon = header.find('.material-icons.expand-arrow');
var container = header.parent().find('.test-results');
var test = header.attr('test');
var time = header.attr('time');
var url = '/api/test_run?test=' + test + '×tamp=' + time;
if (header.parent().hasClass('active')) {
header.parent().removeClass('active');
header.removeClass('active');
icon.removeClass('rotate');
header.siblings('.collapsible-body').stop(true, false).slideUp({
duration: 100,
easing: 'easeOutQuart',
queue: false,
complete: function() {
header.css('height', '');
}
});
} else {
container.empty();
header.parent().addClass('active');
header.addClass('active');
header.addClass('disabled');
icon.addClass('rotate');
$.get(url)
.done(function(data) {
displayTestDetails(container, data, 16);
header.siblings('.collapsible-body').stop(true, false).slideDown({
duration: 100,
easing: 'easeOutQuart',
queue: false,
complete: function() {
header.css('height', '');
}
});
})
.fail(function() {
icon.removeClass('rotate');
})
.always(function() {
header.removeClass('disabled');
});
}
}
/**
* Append a clickable indicator link to the container.
* @param container The jquery object to append the indicator to.
* @param content The text to display in the indicator.
* @param classes Additional space-delimited classes to add to the indicator.
* @param click The click handler to assign to the indicator.
* @returns The jquery object for the indicator.
*/
function createClickableIndicator(container, content, classes, click) {
var link = $('<span></span>');
link.addClass('indicator badge padded hoverable waves-effect');
link.addClass(classes);
link.css('color', 'white');
link.css('margin-left', '1px');
link.append(content);
link.appendTo(container);
link.click(click);
return link;
}
function displayTestMetadata(container, metadataList, showTestNames = false) {
var popout = $('<ul></ul>');
popout.attr('data-collapsible', 'expandable');
popout.addClass('collapsible popout test-runs');
popout.appendTo(container);
popout.unbind();
metadataList.forEach(function(metadata) {
var li = $('<li class="test-run-container"></li>');
li.appendTo(popout);
var div = $('<div></div>');
var test = metadata.testRun.testName;
var startTime = metadata.testRun.startTimestamp;
var endTime = metadata.testRun.endTimestamp;
div.attr('test', test);
div.attr('time', startTime);
div.addClass('collapsible-header test-run');
div.appendTo(li);
div.unbind().click(testRunClick);
var span = $('<span></span>');
span.addClass('test-run-metadata');
span.appendTo(div);
span.click(function() {
return false;
});
if (showTestNames) {
$('<span class="test-run-label"></span>').text(test).appendTo(span);
span.append('<br>');
}
if (metadata.deviceInfo) {
$('<b></b>').text(metadata.deviceInfo).appendTo(span);
span.append('<br>');
}
if (metadata.abiInfo) {
$('<b></b>').text('ABI: ').appendTo(span)
span.append(metadata.abiInfo).append('<br>');
}
$('<b></b>').text('VTS Build: ').appendTo(span)
span.append(metadata.testRun.testBuildId).append('<br>');
$('<b></b>').text('Host: ').appendTo(span)
span.append(metadata.testRun.hostName).append('<br>');
var timeString =
(moment().renderTime(startTime, false) + ' - ' +
moment().renderTime(endTime, true) + ' (' +
moment().renderDuration(endTime - startTime) + ')');
span.append(timeString);
var indicator = $('<span></span>');
var color = metadata.testRun.failCount > 0 ? 'red' : 'green';
indicator.addClass('indicator badge ' + color);
indicator.css('color', 'white');
indicator.append(
metadata.testRun.passCount + '/' +
(metadata.testRun.passCount + metadata.testRun.failCount));
indicator.appendTo(div);
if (metadata.testRun.coveredLineCount != undefined &&
metadata.testRun.totalLineCount != undefined) {
var url = ('/show_coverage?testName=' + test + '&startTime=' + startTime);
var covered = metadata.testRun.coveredLineCount;
var total = metadata.testRun.totalLineCount;
var covPct = getRoundValue(covered / total);
var color = getBadgeColor(covPct);
var coverage =
('Coverage: ' + covered + '/' + total + ' (' + covPct + '%)');
createClickableIndicator(div, coverage, color, function(evt) {
window.location.href = url;
return false;
});
}
if (metadata.testRun.coveredApiCount != undefined &&
metadata.testRun.totalApiCount != undefined) {
var covered = metadata.testRun.coveredApiCount;
var total = metadata.testRun.totalApiCount;
var covPct = getRoundValue(covered / total);
var color = getBadgeColor(covPct);
var apiCoverage = ('API Coverage: ' + covered + '/' + total + ' (' + covPct + '%)');
createClickableIndicator(div, apiCoverage, color, function(evt) {
$('#apiCoverageModal')
.data('urlSafeKeyList', metadata.testRun.apiCoverageKeyList);
$('#apiCoverageModal').modal('open');
return false;
});
}
if (metadata.testRun.logLinks != undefined) {
createClickableIndicator(div, 'Links', 'grey lighten-1', function() {
showLinks(popout, metadata.testRun.logLinks);
return false;
});
}
if ($('#coverageModalGraph').length) {
createClickableIndicator(div, 'Graph', 'grey lighten-1', function(evt) {
$('#coverageModalGraph').data('testname', test);
$('#coverageModalGraph').modal('open');
$(evt.target).removeClass('grey');
$(evt.target).addClass('blue');
return false;
});
}
var expand = $('<i></i>');
expand.addClass('material-icons expand-arrow')
expand.text('expand_more');
expand.appendTo(div);
var body = $('<div></div>')
.addClass('collapsible-body test-results row')
.appendTo(li);
if (metadata.testDetails != undefined) {
expand.addClass('rotate');
li.addClass('active');
div.addClass('active');
displayTestDetails(body, metadata.testDetails, 16);
div.siblings('.collapsible-body').stop(true, false).slideDown({
duration: 0,
queue: false,
complete: function() {
div.css('height', '');
}
});
}
});
}
/**
* Display test metadata in a vertical popout.
* @param container The jquery object in which to insert the test metadata.
* @param metadataList The list of metadata objects to render on the display.
* @param showTestNames True to label each entry with the test module name.
*/
$.fn.showTests = function(metadataList, showTestNames = false) {
displayTestMetadata($(this), metadataList, showTestNames);
}
})(jQuery, moment);