Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add release hook to format the changelog by labels #14592

Merged
merged 7 commits into from
Jul 28, 2020
Merged
Show file tree
Hide file tree
Changes from 4 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@
"types": "lerna run types --stream",
"typescript": "lerna run typescript",
"prepublish": "lerna run prepublish",
"publish-canary": "lerna version prerelease --preid canary --force-publish && release --pre",
"publish-canary": "lerna version prerelease --preid canary --force-publish && release --pre --skip-questions",
Copy link
Contributor Author

@rafaelalmeidatk rafaelalmeidatk Jun 27, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this has to be changed somewhere else too for stable releases?

"publish-stable": "lerna version --force-publish",
"lint-staged": "lint-staged",
"next": "node packages/next/dist/bin/next",
Expand Down
162 changes: 162 additions & 0 deletions release.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,162 @@
const fetch = require('node-fetch')

// --------------------
// Repo

const owner = 'vercel'
const repo = 'next.js'

// --------------------
// Config

// section -> label
const sectionLabelMap = {
'Core Changes': 'type: next',
'Documentation Changes': 'type: documentation',
'Example Changes': 'type: example',
}

const fallbackSection = 'Misc Changes'

// --------------------------------------------------

const prNumberRegex = /\(#([-0-9]+)\)/

const getCommitPullRequest = async (commit) => {
const match = prNumberRegex.exec(commit.title)

if (!match) {
return null
}

const number = parseInt(match[1], 10)
const url = `https://api.github.com/repos/${owner}/${repo}/pulls/${number}`
const res = await fetch(url)

if (!res.ok) {
console.warn(`\n\n⚠️ Failed to fetch PR #${number}\n`)
return null
}

return await res.json()
}

const getSectionForPullRequest = (pullRequest) => {
const { labels } = pullRequest

// sections defined first will take priority
for (const [section, label] of Object.entries(sectionLabelMap)) {
if (labels.some((prLabel) => prLabel.name === label)) {
return section
}
}

return null
}

const groupByLabels = async (commits, github) => {
// Initialize the sections object with empty arrays
const sections = Object.keys(sectionLabelMap).reduce((sections, section) => {
sections[section] = []
return sections
}, {})
sections.__fallback = []

for (const commit of commits) {
const pullRequest = await getCommitPullRequest(commit, github)

if (pullRequest) {
const section = getSectionForPullRequest(pullRequest)

if (section) {
// Add the change to the respective section
sections[section].push({
title: pullRequest.title,
number: pullRequest.number,
})

continue
}

// No section found, add it to the fallback section
sections.__fallback.push({
title: pullRequest.title,
number: pullRequest.number,
})

continue
}

// No Pull Request found, add it to the fallback section but without the number
sections.__fallback.push({
title: commit.title,
})
}

return sections
}

const buildChangelog = (sections, authors) => {
let text = ''

for (const section in sections) {
const changes = sections[section]

// No changes in this section? Don't render it
if (changes.length === 0) {
continue
}

const title = section === '__fallback' ? fallbackSection : section
text += `### ${title}\n\n`

for (const change of changes) {
const numberText = change.number != null ? `: #${change.number}` : ''

text += `- ${change.title}${numberText}\n`
}

text += '\n'
}

if (authors.size > 0) {
text += '### Credits \n\n'
text += 'Huge thanks to '

let index = 1
authors.forEach((author) => {
// GitHub links usernames if prefixed with @
text += `@${author}`

const penultimate = index === authors.size - 1
const notLast = index !== authors.size

if (penultimate) {
// Oxford comma is applied when list is bigger than 2 names
if (authors.size > 2) {
text += ','
}

text += ' and '
} else if (notLast) {
text += ', '
}

index += 1
})

text += ' for helping!'
text += '\n'
}

return text
}

module.exports = async (markdown, metadata) => {
const { commits, authors } = metadata

const sections = await groupByLabels(commits.all)
const changelog = buildChangelog(sections, authors)

return changelog
}