grafana/scripts/cli/tasks/closeMilestone.ts
gotjosh bc94f85dee
Improvement: Grafana release process minor improvements (#17661)
* Don't display changelog category title when no items

The output of the changelog is meant to be copy/pasted with ease. When a
changelog category does not contain items is better to not display title
at all thus avoiding having the manually modify the output as we include
it in the steps of the process.

* Introduce a CLI task to close milestones whilst doing a Grafana release

As part of a Grafana release, we need to eventually close the GitHub
milestone to indicate is done and remove all the cherry-pick labels from
issues/prs within the milestone to avoid our cherry-pick CLI command to
pick them up on the next release.

* Abstract the GitHub client into a module

* Introduce `GitHubClient` to all CLI tasks
2019-06-24 16:00:01 +01:00

76 lines
2.2 KiB
TypeScript

import { Task, TaskRunner } from './task';
import GithubClient from '../utils/githubClient';
interface CloseMilestoneOptions {
milestone: string;
}
const closeMilestoneTaskRunner: TaskRunner<CloseMilestoneOptions> = async ({ milestone }) => {
const githubClient = new GithubClient(true);
const cherryPickLabel = 'cherry-pick needed';
const client = githubClient.client;
if (!/^\d+$/.test(milestone)) {
console.log('Use milestone number not title, find number in milestone url');
return;
}
const milestoneRes = await client.get(`/milestones/${milestone}`, {});
const milestoneState = milestoneRes.data.state;
if (milestoneState === 'closed') {
console.log('milestone already closed. ✅');
return;
}
console.log('fetching issues/PRs of the milestone ⏬');
// Get all the issues/PRs with the label cherry-pick
// Every pull request is actually an issue
const issuesRes = await client.get('/issues', {
params: {
state: 'closed',
labels: cherryPickLabel,
per_page: 100,
milestone: milestone,
},
});
if (issuesRes.data.length < 1) {
console.log('no issues to remove label from');
} else {
console.log(`found ${issuesRes.data.length} issues to remove the cherry-pick label from 🔎`);
}
for (const issue of issuesRes.data) {
// the reason for using stdout.write is for achieving 'action -> result' on
// the same line
process.stdout.write(`🔧removing label from issue #${issue.number} 🗑...`);
const resDelete = await client.delete(`/issues/${issue.number}/labels/${cherryPickLabel}`, {});
if (resDelete.status === 200) {
process.stdout.write('done ✅\n');
} else {
console.log('failed ❌');
}
}
console.log(`cleaned up ${issuesRes.data.length} issues/prs ⚡️`);
const resClose = await client.patch(`/milestones/${milestone}`, {
state: 'closed',
});
if (resClose.status === 200) {
console.log('milestone closed 🙌');
} else {
console.log('failed to close the milestone, response:');
console.log(resClose);
}
};
export const closeMilestoneTask = new Task<CloseMilestoneOptions>();
closeMilestoneTask.setName('Close Milestone generator task');
closeMilestoneTask.setRunner(closeMilestoneTaskRunner);