pullrequests.js
550 lines
| 17.6 KiB
| application/javascript
|
JavascriptLexer
r2487 | // # Copyright (C) 2010-2018 RhodeCode GmbH | ||
r1 | // # | ||
// # This program is free software: you can redistribute it and/or modify | |||
// # it under the terms of the GNU Affero General Public License, version 3 | |||
// # (only), as published by the Free Software Foundation. | |||
// # | |||
// # This program is distributed in the hope that it will be useful, | |||
// # but WITHOUT ANY WARRANTY; without even the implied warranty of | |||
// # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |||
// # GNU General Public License for more details. | |||
// # | |||
// # You should have received a copy of the GNU Affero General Public License | |||
// # along with this program. If not, see <http://www.gnu.org/licenses/>. | |||
// # | |||
// # This program is dual-licensed. If you wish to learn more about the | |||
// # RhodeCode Enterprise Edition, including its added features, Support services, | |||
// # and proprietary license terms, please see https://rhodecode.com/licenses/ | |||
r1769 | |||
var prButtonLockChecks = { | |||
'compare': false, | |||
'reviewers': false | |||
}; | |||
r1 | /** | ||
r1769 | * lock button until all checks and loads are made. E.g reviewer calculation | ||
* should prevent from submitting a PR | |||
* @param lockEnabled | |||
* @param msg | |||
* @param scope | |||
r1 | */ | ||
r1769 | var prButtonLock = function(lockEnabled, msg, scope) { | ||
scope = scope || 'all'; | |||
if (scope == 'all'){ | |||
prButtonLockChecks['compare'] = !lockEnabled; | |||
prButtonLockChecks['reviewers'] = !lockEnabled; | |||
} else if (scope == 'compare') { | |||
prButtonLockChecks['compare'] = !lockEnabled; | |||
} else if (scope == 'reviewers'){ | |||
prButtonLockChecks['reviewers'] = !lockEnabled; | |||
} | |||
var checksMeet = prButtonLockChecks.compare && prButtonLockChecks.reviewers; | |||
if (lockEnabled) { | |||
r2806 | $('#pr_submit').attr('disabled', 'disabled'); | ||
r1769 | } | ||
else if (checksMeet) { | |||
r2806 | $('#pr_submit').removeAttr('disabled'); | ||
r1769 | } | ||
r1 | |||
r1769 | if (msg) { | ||
$('#pr_open_message').html(msg); | |||
} | |||
}; | |||
r1 | |||
r1769 | |||
/** | |||
Generate Title and Description for a PullRequest. | |||
In case of 1 commits, the title and description is that one commit | |||
in case of multiple commits, we iterate on them with max N number of commits, | |||
and build description in a form | |||
- commitN | |||
- commitN+1 | |||
... | |||
Title is then constructed from branch names, or other references, | |||
replacing '-' and '_' into spaces | |||
* @param sourceRef | |||
* @param elements | |||
* @param limit | |||
* @returns {*[]} | |||
*/ | |||
var getTitleAndDescription = function(sourceRef, elements, limit) { | |||
var title = ''; | |||
var desc = ''; | |||
$.each($(elements).get().reverse().slice(0, limit), function(idx, value) { | |||
var rawMessage = $(value).find('td.td-description .message').data('messageRaw'); | |||
desc += '- ' + rawMessage.split('\n')[0].replace(/\n+$/, "") + '\n'; | |||
}); | |||
// only 1 commit, use commit message as title | |||
if (elements.length === 1) { | |||
title = $(elements[0]).find('td.td-description .message').data('messageRaw').split('\n')[0]; | |||
} | |||
else { | |||
// use reference name | |||
title = sourceRef.replace(/-/g, ' ').replace(/_/g, ' ').capitalizeFirstLetter(); | |||
} | |||
return [title, desc] | |||
r1 | }; | ||
r1769 | |||
ReviewersController = function () { | |||
var self = this; | |||
this.$reviewRulesContainer = $('#review_rules'); | |||
this.$rulesList = this.$reviewRulesContainer.find('.pr-reviewer-rules'); | |||
this.forbidReviewUsers = undefined; | |||
this.$reviewMembers = $('#review_members'); | |||
this.currentRequest = null; | |||
this.defaultForbidReviewUsers = function() { | |||
return [ | |||
{'username': 'default', | |||
'user_id': templateContext.default_user.user_id} | |||
]; | |||
}; | |||
this.hideReviewRules = function() { | |||
self.$reviewRulesContainer.hide(); | |||
}; | |||
this.showReviewRules = function() { | |||
self.$reviewRulesContainer.show(); | |||
}; | |||
this.addRule = function(ruleText) { | |||
self.showReviewRules(); | |||
return '<div>- {0}</div>'.format(ruleText) | |||
}; | |||
this.loadReviewRules = function(data) { | |||
// reset forbidden Users | |||
this.forbidReviewUsers = self.defaultForbidReviewUsers(); | |||
// reset state of review rules | |||
self.$rulesList.html(''); | |||
if (!data || data.rules === undefined || $.isEmptyObject(data.rules)) { | |||
// default rule, case for older repo that don't have any rules stored | |||
self.$rulesList.append( | |||
self.addRule( | |||
_gettext('All reviewers must vote.')) | |||
); | |||
return self.forbidReviewUsers | |||
} | |||
if (data.rules.voting !== undefined) { | |||
r2484 | if (data.rules.voting < 0) { | ||
r1769 | self.$rulesList.append( | ||
self.addRule( | |||
r2484 | _gettext('All individual reviewers must vote.')) | ||
r1769 | ) | ||
} else if (data.rules.voting === 1) { | |||
self.$rulesList.append( | |||
self.addRule( | |||
_gettext('At least {0} reviewer must vote.').format(data.rules.voting)) | |||
) | |||
} else { | |||
self.$rulesList.append( | |||
self.addRule( | |||
_gettext('At least {0} reviewers must vote.').format(data.rules.voting)) | |||
) | |||
} | |||
} | |||
r2484 | |||
if (data.rules.voting_groups !== undefined) { | |||
$.each(data.rules.voting_groups, function(index, rule_data) { | |||
self.$rulesList.append( | |||
self.addRule(rule_data.text) | |||
) | |||
}); | |||
} | |||
r1769 | if (data.rules.use_code_authors_for_review) { | ||
self.$rulesList.append( | |||
self.addRule( | |||
_gettext('Reviewers picked from source code changes.')) | |||
) | |||
} | |||
if (data.rules.forbid_adding_reviewers) { | |||
$('#add_reviewer_input').remove(); | |||
self.$rulesList.append( | |||
self.addRule( | |||
_gettext('Adding new reviewers is forbidden.')) | |||
) | |||
} | |||
if (data.rules.forbid_author_to_review) { | |||
self.forbidReviewUsers.push(data.rules_data.pr_author); | |||
self.$rulesList.append( | |||
self.addRule( | |||
_gettext('Author is not allowed to be a reviewer.')) | |||
) | |||
} | |||
r1787 | if (data.rules.forbid_commit_author_to_review) { | ||
if (data.rules_data.forbidden_users) { | |||
$.each(data.rules_data.forbidden_users, function(index, member_data) { | |||
self.forbidReviewUsers.push(member_data) | |||
}); | |||
} | |||
self.$rulesList.append( | |||
self.addRule( | |||
_gettext('Commit Authors are not allowed to be a reviewer.')) | |||
) | |||
} | |||
r1769 | return self.forbidReviewUsers | ||
}; | |||
this.loadDefaultReviewers = function(sourceRepo, sourceRef, targetRepo, targetRef) { | |||
if (self.currentRequest) { | |||
// make sure we cleanup old running requests before triggering this | |||
// again | |||
self.currentRequest.abort(); | |||
r821 | } | ||
r1769 | |||
$('.calculate-reviewers').show(); | |||
// reset reviewer members | |||
self.$reviewMembers.empty(); | |||
prButtonLock(true, null, 'reviewers'); | |||
$('#user').hide(); // hide user autocomplete before load | |||
r2555 | if (sourceRef.length !== 3 || targetRef.length !== 3) { | ||
// don't load defaults in case we're missing some refs... | |||
$('.calculate-reviewers').hide(); | |||
return | |||
} | |||
r1769 | var url = pyroutes.url('repo_default_reviewers_data', | ||
{ | |||
'repo_name': templateContext.repo_name, | |||
'source_repo': sourceRepo, | |||
'source_ref': sourceRef[2], | |||
'target_repo': targetRepo, | |||
'target_ref': targetRef[2] | |||
}); | |||
self.currentRequest = $.get(url) | |||
.done(function(data) { | |||
self.currentRequest = null; | |||
// review rules | |||
self.loadReviewRules(data); | |||
for (var i = 0; i < data.reviewers.length; i++) { | |||
var reviewer = data.reviewers[i]; | |||
self.addReviewMember( | |||
r2484 | reviewer, reviewer.reasons, reviewer.mandatory); | ||
r1769 | } | ||
$('.calculate-reviewers').hide(); | |||
prButtonLock(false, null, 'reviewers'); | |||
$('#user').show(); // show user autocomplete after load | |||
}); | |||
}; | |||
// check those, refactor | |||
this.removeReviewMember = function(reviewer_id, mark_delete) { | |||
var reviewer = $('#reviewer_{0}'.format(reviewer_id)); | |||
if(typeof(mark_delete) === undefined){ | |||
mark_delete = false; | |||
} | |||
if(mark_delete === true){ | |||
if (reviewer){ | |||
// now delete the input | |||
$('#reviewer_{0} input'.format(reviewer_id)).remove(); | |||
// mark as to-delete | |||
var obj = $('#reviewer_{0}_name'.format(reviewer_id)); | |||
obj.addClass('to-delete'); | |||
obj.css({"text-decoration":"line-through", "opacity": 0.5}); | |||
} | |||
} | |||
else{ | |||
$('#reviewer_{0}'.format(reviewer_id)).remove(); | |||
} | |||
}; | |||
r2484 | this.reviewMemberEntry = function() { | ||
r1769 | |||
r2484 | }; | ||
this.addReviewMember = function(reviewer_obj, reasons, mandatory) { | |||
r1769 | var members = self.$reviewMembers.get(0); | ||
r2484 | var id = reviewer_obj.user_id; | ||
var username = reviewer_obj.username; | |||
r1769 | var reasons = reasons || []; | ||
var mandatory = mandatory || false; | |||
r873 | |||
r2484 | // register IDS to check if we don't have this ID already in | ||
var currentIds = []; | |||
r1769 | var _els = self.$reviewMembers.find('li').toArray(); | ||
for (el in _els){ | |||
r2484 | currentIds.push(_els[el].id) | ||
r1769 | } | ||
var userAllowedReview = function(userId) { | |||
var allowed = true; | |||
$.each(self.forbidReviewUsers, function(index, member_data) { | |||
if (parseInt(userId) === member_data['user_id']) { | |||
allowed = false; | |||
return false // breaks the loop | |||
} | |||
}); | |||
return allowed | |||
}; | |||
var userAllowed = userAllowedReview(id); | |||
if (!userAllowed){ | |||
r2484 | alert(_gettext('User `{0}` not allowed to be a reviewer').format(username)); | ||
} else { | |||
// only add if it's not there | |||
var alreadyReviewer = currentIds.indexOf('reviewer_'+id) != -1; | |||
r1769 | |||
r2484 | if (alreadyReviewer) { | ||
alert(_gettext('User `{0}` already in reviewers').format(username)); | |||
} else { | |||
members.innerHTML += renderTemplate('reviewMemberEntry', { | |||
'member': reviewer_obj, | |||
'mandatory': mandatory, | |||
'allowed_to_update': true, | |||
'review_status': 'not_reviewed', | |||
'review_status_label': _gettext('Not Reviewed'), | |||
r2504 | 'reasons': reasons, | ||
'create': true | |||
r2484 | }); | ||
} | |||
r1769 | } | ||
}; | |||
this.updateReviewers = function(repo_name, pull_request_id){ | |||
r2484 | var postData = $('#reviewers input').serialize(); | ||
r1769 | _updatePullRequest(repo_name, pull_request_id, postData); | ||
}; | |||
r1 | |||
}; | |||
r1769 | |||
r1 | var _updatePullRequest = function(repo_name, pull_request_id, postData) { | ||
var url = pyroutes.url( | |||
'pullrequest_update', | |||
{"repo_name": repo_name, "pull_request_id": pull_request_id}); | |||
r873 | if (typeof postData === 'string' ) { | ||
postData += '&csrf_token=' + CSRF_TOKEN; | |||
} else { | |||
postData.csrf_token = CSRF_TOKEN; | |||
} | |||
r1 | var success = function(o) { | ||
window.location.reload(); | |||
}; | |||
ajaxPOST(url, postData, success); | |||
}; | |||
/** | |||
* PULL REQUEST update commits | |||
*/ | |||
var updateCommits = function(repo_name, pull_request_id) { | |||
var postData = { | |||
'update_commits': true}; | |||
_updatePullRequest(repo_name, pull_request_id, postData); | |||
}; | |||
/** | |||
* PULL REQUEST edit info | |||
*/ | |||
r2903 | var editPullRequest = function(repo_name, pull_request_id, title, description, renderer) { | ||
r1 | var url = pyroutes.url( | ||
'pullrequest_update', | |||
{"repo_name": repo_name, "pull_request_id": pull_request_id}); | |||
var postData = { | |||
'title': title, | |||
'description': description, | |||
r2903 | 'description_renderer': renderer, | ||
r1 | 'edit_pull_request': true, | ||
'csrf_token': CSRF_TOKEN | |||
}; | |||
var success = function(o) { | |||
window.location.reload(); | |||
}; | |||
ajaxPOST(url, postData, success); | |||
}; | |||
/** | |||
* Reviewer autocomplete | |||
*/ | |||
r1678 | var ReviewerAutoComplete = function(inputId) { | ||
$(inputId).autocomplete({ | |||
r1 | serviceUrl: pyroutes.url('user_autocomplete_data'), | ||
minChars:2, | |||
maxHeight:400, | |||
deferRequestBy: 300, //miliseconds | |||
showNoSuggestionNotice: true, | |||
tabDisabled: true, | |||
autoSelectFirst: true, | |||
r1769 | params: { user_id: templateContext.rhodecode_user.user_id, user_groups:true, user_groups_expand:true, skip_default_user:true }, | ||
r1 | formatResult: autocompleteFormatResult, | ||
lookupFilter: autocompleteFilterResult, | |||
r1678 | onSelect: function(element, data) { | ||
r2484 | var mandatory = false; | ||
var reasons = [_gettext('added manually by "{0}"').format(templateContext.rhodecode_user.username)]; | |||
r1678 | |||
r2484 | // add whole user groups | ||
r1678 | if (data.value_type == 'user_group') { | ||
reasons.push(_gettext('member of "{0}"').format(data.value_display)); | |||
$.each(data.members, function(index, member_data) { | |||
r2484 | var reviewer = member_data; | ||
reviewer['user_id'] = member_data['id']; | |||
reviewer['gravatar_link'] = member_data['icon_link']; | |||
reviewer['user_link'] = member_data['profile_link']; | |||
reviewer['rules'] = []; | |||
reviewersController.addReviewMember(reviewer, reasons, mandatory); | |||
r1678 | }) | ||
r2484 | } | ||
// add single user | |||
else { | |||
var reviewer = data; | |||
reviewer['user_id'] = data['id']; | |||
reviewer['gravatar_link'] = data['icon_link']; | |||
reviewer['user_link'] = data['profile_link']; | |||
reviewer['rules'] = []; | |||
reviewersController.addReviewMember(reviewer, reasons, mandatory); | |||
r1678 | } | ||
$(inputId).val(''); | |||
r1 | } | ||
}); | |||
}; | |||
r1371 | |||
VersionController = function () { | |||
var self = this; | |||
this.$verSource = $('input[name=ver_source]'); | |||
this.$verTarget = $('input[name=ver_target]'); | |||
this.$showVersionDiff = $('#show-version-diff'); | |||
this.adjustRadioSelectors = function (curNode) { | |||
var getVal = function (item) { | |||
if (item == 'latest') { | |||
return Number.MAX_SAFE_INTEGER | |||
} | |||
else { | |||
return parseInt(item) | |||
} | |||
}; | |||
var curVal = getVal($(curNode).val()); | |||
var cleared = false; | |||
$.each(self.$verSource, function (index, value) { | |||
var elVal = getVal($(value).val()); | |||
if (elVal > curVal) { | |||
if ($(value).is(':checked')) { | |||
cleared = true; | |||
} | |||
$(value).attr('disabled', 'disabled'); | |||
$(value).removeAttr('checked'); | |||
$(value).css({'opacity': 0.1}); | |||
} | |||
else { | |||
$(value).css({'opacity': 1}); | |||
$(value).removeAttr('disabled'); | |||
} | |||
}); | |||
if (cleared) { | |||
// if we unchecked an active, set the next one to same loc. | |||
$(this.$verSource).filter('[value={0}]'.format( | |||
curVal)).attr('checked', 'checked'); | |||
} | |||
self.setLockAction(false, | |||
$(curNode).data('verPos'), | |||
$(this.$verSource).filter(':checked').data('verPos') | |||
); | |||
}; | |||
this.attachVersionListener = function () { | |||
self.$verTarget.change(function (e) { | |||
self.adjustRadioSelectors(this) | |||
}); | |||
self.$verSource.change(function (e) { | |||
self.adjustRadioSelectors(self.$verTarget.filter(':checked')) | |||
}); | |||
}; | |||
this.init = function () { | |||
var curNode = self.$verTarget.filter(':checked'); | |||
self.adjustRadioSelectors(curNode); | |||
self.setLockAction(true); | |||
self.attachVersionListener(); | |||
}; | |||
this.setLockAction = function (state, selectedVersion, otherVersion) { | |||
var $showVersionDiff = this.$showVersionDiff; | |||
if (state) { | |||
$showVersionDiff.attr('disabled', 'disabled'); | |||
$showVersionDiff.addClass('disabled'); | |||
$showVersionDiff.html($showVersionDiff.data('labelTextLocked')); | |||
} | |||
else { | |||
$showVersionDiff.removeAttr('disabled'); | |||
$showVersionDiff.removeClass('disabled'); | |||
if (selectedVersion == otherVersion) { | |||
$showVersionDiff.html($showVersionDiff.data('labelTextShow')); | |||
} else { | |||
$showVersionDiff.html($showVersionDiff.data('labelTextDiff')); | |||
} | |||
} | |||
}; | |||
this.showVersionDiff = function () { | |||
var target = self.$verTarget.filter(':checked'); | |||
var source = self.$verSource.filter(':checked'); | |||
if (target.val() && source.val()) { | |||
var params = { | |||
'pull_request_id': templateContext.pull_request_data.pull_request_id, | |||
'repo_name': templateContext.repo_name, | |||
'version': target.val(), | |||
'from_version': source.val() | |||
}; | |||
window.location = pyroutes.url('pullrequest_show', params) | |||
} | |||
return false; | |||
}; | |||
this.toggleVersionView = function (elem) { | |||
if (this.$showVersionDiff.is(':visible')) { | |||
$('.version-pr').hide(); | |||
this.$showVersionDiff.hide(); | |||
$(elem).html($(elem).data('toggleOn')) | |||
} else { | |||
$('.version-pr').show(); | |||
this.$showVersionDiff.show(); | |||
$(elem).html($(elem).data('toggleOff')) | |||
} | |||
return false | |||
} | |||
}; |