281 lines
10 KiB
JavaScript
281 lines
10 KiB
JavaScript
// (C) Copyright 2015 Moodle Pty Ltd.
|
|
//
|
|
// 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.
|
|
|
|
const gulp = require('gulp');
|
|
const inquirer = require('inquirer');
|
|
const DevConfig = require('./dev-config');
|
|
const Git = require('./git');
|
|
const Jira = require('./jira');
|
|
const Utils = require('./utils');
|
|
|
|
/**
|
|
* Task to push a git branch and update tracker issue.
|
|
*/
|
|
class PushTask {
|
|
|
|
/**
|
|
* Ask the user whether he wants to continue.
|
|
*
|
|
* @return Promise resolved with boolean: true if he wants to continue.
|
|
*/
|
|
async askConfirmContinue() {
|
|
const answer = await inquirer.prompt([
|
|
{
|
|
type: 'input',
|
|
name: 'confirm',
|
|
message: 'Are you sure you want to continue?',
|
|
default: 'n',
|
|
},
|
|
]);
|
|
|
|
return answer.confirm == 'y';
|
|
}
|
|
|
|
/**
|
|
* Push a patch to the tracker and remove the previous one.
|
|
*
|
|
* @param branch Branch name.
|
|
* @param branchData Parsed branch data.
|
|
* @param remote Remote used.
|
|
* @return Promise resolved when done.
|
|
*/
|
|
async pushPatch(branch, branchData, remote) {
|
|
const headCommit = await Git.getHeadCommit(branch, branchData);
|
|
|
|
if (!headCommit) {
|
|
throw new Error('Head commit not resolved, abort pushing patch.');
|
|
}
|
|
|
|
// Create the patch file.
|
|
const fileName = branch + '.patch';
|
|
const tmpPatchPath = `./tmp/${fileName}`;
|
|
|
|
await Git.createPatch(`${headCommit}...${branch}`, tmpPatchPath);
|
|
console.log('Git patch created');
|
|
|
|
// Check if there is an attachment with same name in the issue.
|
|
const issue = await Jira.getIssue(branchData.issue, 'attachment');
|
|
|
|
let existingAttachmentId;
|
|
const attachments = (issue.fields && issue.fields.attachment) || [];
|
|
for (const i in attachments) {
|
|
if (attachments[i].filename == fileName) {
|
|
// Found an existing attachment with the same name, we keep track of it.
|
|
existingAttachmentId = attachments[i].id;
|
|
break
|
|
}
|
|
}
|
|
|
|
// Push the patch to the tracker.
|
|
console.log(`Uploading patch ${fileName} to the tracker...`);
|
|
await Jira.upload(branchData.issue, tmpPatchPath);
|
|
|
|
if (existingAttachmentId) {
|
|
// On success, deleting file that was there before.
|
|
try {
|
|
console.log('Deleting older patch...')
|
|
await Jira.deleteAttachment(existingAttachmentId);
|
|
} catch (error) {
|
|
console.log('Could not delete older attachment.');
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Run the task.
|
|
*
|
|
* @param args Command line arguments.
|
|
* @param done Function to call when done.
|
|
*/
|
|
async run(args, done) {
|
|
try {
|
|
const remote = args.remote || DevConfig.get('upstreamRemote', 'origin');
|
|
let branch = args.branch;
|
|
const force = !!args.force;
|
|
|
|
if (!branch) {
|
|
branch = await Git.getCurrentBranch();
|
|
}
|
|
|
|
if (!branch) {
|
|
throw new Error('Cannot determine the current branch. Please make sure youu aren\'t in detached HEAD state');
|
|
} else if (branch == 'HEAD') {
|
|
throw new Error('Cannot push HEAD branch');
|
|
}
|
|
|
|
// Parse the branch to get the project and issue number.
|
|
const branchData = Utils.parseBranch(branch);
|
|
const keepRunning = await this.validateCommitMessages(branchData);
|
|
|
|
if (!keepRunning) {
|
|
// Last commit not valid, stop.
|
|
console.log('Exiting...');
|
|
done();
|
|
return;
|
|
}
|
|
|
|
if (!args.patch) {
|
|
// Check if it's a security issue to force patch mode.
|
|
try {
|
|
args.patch = await Jira.isSecurityIssue(branchData.issue);
|
|
|
|
if (args.patch) {
|
|
console.log(`${branchData.issue} appears to be a security issue, switching to patch mode...`);
|
|
}
|
|
} catch (error) {
|
|
console.log(`Could not check if ${branchData.issue} is a security issue.`);
|
|
}
|
|
}
|
|
|
|
if (args.patch) {
|
|
// Create and upload a patch file.
|
|
await this.pushPatch(branch, branchData, remote);
|
|
} else {
|
|
// Push the branch.
|
|
console.log(`Pushing branch ${branch} to remote ${remote}...`);
|
|
await Git.push(remote, branch, force);
|
|
|
|
// Update tracker info.
|
|
console.log(`Branch pushed, update tracker info...`);
|
|
await this.updateTrackerGitInfo(branch, branchData, remote);
|
|
}
|
|
} catch (error) {
|
|
console.error(error);
|
|
}
|
|
|
|
done();
|
|
}
|
|
|
|
/**
|
|
* Update git info in the tracker issue.
|
|
*
|
|
* @param branch Branch name.
|
|
* @param branchData Parsed branch data.
|
|
* @param remote Remote used.
|
|
* @return Promise resolved when done.
|
|
*/
|
|
async updateTrackerGitInfo(branch, branchData, remote) {
|
|
// Get the repository data for the project.
|
|
let repositoryUrl = DevConfig.get(branchData.project + '.repositoryUrl');
|
|
let diffUrlTemplate = DevConfig.get(branchData.project + '.diffUrlTemplate', '');
|
|
|
|
if (!repositoryUrl) {
|
|
// Calculate the repositoryUrl based on the remote URL.
|
|
repositoryUrl = await Git.getRemoteUrl(remote);
|
|
}
|
|
|
|
// Make sure the repository URL uses the regular format.
|
|
repositoryUrl = repositoryUrl.replace(/^(git@|git:\/\/)/, 'https://')
|
|
.replace(/\.git$/, '')
|
|
.replace('github.com:', 'github.com/');
|
|
|
|
if (!diffUrlTemplate) {
|
|
diffUrlTemplate = Utils.concatenatePaths([repositoryUrl, 'compare/%headcommit%...%branch%']);
|
|
}
|
|
|
|
// Now create the git URL for the repository.
|
|
const repositoryGitUrl = repositoryUrl.replace(/^https?:\/\//, 'git://') + '.git';
|
|
|
|
// Search HEAD commit to put in the diff URL.
|
|
console.log ('Searching for head commit...');
|
|
let headCommit = await Git.getHeadCommit(branch, branchData);
|
|
|
|
if (!headCommit) {
|
|
throw new Error('Head commit not resolved, aborting update of tracker fields');
|
|
}
|
|
|
|
headCommit = headCommit.substr(0, 10);
|
|
console.log(`Head commit resolved to ${headCommit}`);
|
|
|
|
// Calculate last properties needed.
|
|
const diffUrl = diffUrlTemplate.replace('%branch%', branch).replace('%headcommit%', headCommit);
|
|
const fieldRepositoryUrl = DevConfig.get('tracker.fieldnames.repositoryurl', 'Pull from Repository');
|
|
const fieldBranch = DevConfig.get('tracker.fieldnames.branch', 'Pull Master Branch');
|
|
const fieldDiffUrl = DevConfig.get('tracker.fieldnames.diffurl', 'Pull Master Diff URL');
|
|
|
|
// Update tracker fields.
|
|
const updates = {};
|
|
updates[fieldRepositoryUrl] = repositoryGitUrl;
|
|
updates[fieldBranch] = branch;
|
|
updates[fieldDiffUrl] = diffUrl;
|
|
|
|
console.log('Setting tracker fields...');
|
|
await Jira.setCustomFields(branchData.issue, updates);
|
|
}
|
|
|
|
/**
|
|
* Validate commit messages comparing them with the branch name.
|
|
*
|
|
* @param branchData Parsed branch data.
|
|
* @return True if value is ok or the user wants to continue anyway, false to stop.
|
|
*/
|
|
async validateCommitMessages(branchData) {
|
|
const messages = await Git.messages(30);
|
|
|
|
let numConsecutive = 0;
|
|
let wrongCommitCandidate = null;
|
|
|
|
for (let i = 0; i < messages.length; i++) {
|
|
const message = messages[i];
|
|
const issue = Utils.getIssueFromCommitMessage(message);
|
|
|
|
if (!issue || issue != branchData.issue) {
|
|
if (i === 0) {
|
|
// Last commit is wrong, it shouldn't happen. Ask the user if he wants to continue.
|
|
if (!issue) {
|
|
console.log('The issue number could not be found in the last commit message.');
|
|
console.log(`Commit: ${message}`);
|
|
} else if (issue != branchData.issue) {
|
|
console.log('The issue number in the last commit does not match the branch being pushed to.');
|
|
console.log(`Branch: ${branchData.issue} vs. commit: ${issue}`);
|
|
}
|
|
|
|
return this.askConfirmContinue();
|
|
}
|
|
|
|
numConsecutive++;
|
|
if (numConsecutive > 2) {
|
|
// 3 consecutive commits with different branch, probably the branch commits are over. Everything OK.
|
|
return true;
|
|
|
|
// Don't treat a merge pull request commit as a wrong commit between right commits.
|
|
// The current push could be a quick fix after a merge.
|
|
} else if (!wrongCommitCandidate && message.indexOf('Merge pull request') == -1) {
|
|
wrongCommitCandidate = {
|
|
message: message,
|
|
issue: issue,
|
|
index: i,
|
|
};
|
|
}
|
|
} else if (wrongCommitCandidate) {
|
|
// We've found a commit with the branch name after a commit with a different branch. Probably wrong commit.
|
|
if (!wrongCommitCandidate.issue) {
|
|
console.log('The issue number could not be found in one of the commit messages.');
|
|
console.log(`Commit: ${wrongCommitCandidate.message}`);
|
|
} else {
|
|
console.log('The issue number in a certain commit does not match the branch being pushed to.');
|
|
console.log(`Branch: ${branchData.issue} vs. commit: ${wrongCommitCandidate.issue}`);
|
|
console.log(`Commit message: ${wrongCommitCandidate.message}`);
|
|
}
|
|
|
|
return this.askConfirmContinue();
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
module.exports = PushTask;
|