Files
next.js/scripts/generate-release-log.mjs
Arian Tron 61f56f997c
Some checks failed
Test examples / Test Examples (20) (push) Has been cancelled
Test examples / Test Examples (22) (push) Has been cancelled
Lock Threads / action (push) Has been cancelled
Trigger Release / start (push) Has been cancelled
Stale issue handler / stale (push) Has been cancelled
Update Font Data / create-pull-request (push) Has been cancelled
build-and-deploy / deploy-target (push) Has been cancelled
build-and-deploy / build (push) Has been cancelled
build-and-deploy / stable - aarch64-unknown-linux-musl - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-unknown-linux-musl - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-unknown-linux-gnu - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-unknown-linux-gnu - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-pc-windows-msvc - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-pc-windows-msvc - node@16 (push) Has been cancelled
build-and-deploy / stable - aarch64-apple-darwin - node@16 (push) Has been cancelled
build-and-deploy / stable - x86_64-apple-darwin - node@16 (push) Has been cancelled
build-and-deploy / build-wasm (nodejs) (push) Has been cancelled
build-and-deploy / build-wasm (web) (push) Has been cancelled
build-and-deploy / Deploy preview tarball (push) Has been cancelled
build-and-deploy / Potentially publish release (push) Has been cancelled
build-and-deploy / publish-turbopack-npm-packages (push) Has been cancelled
build-and-deploy / Deploy examples (push) Has been cancelled
build-and-deploy / thank you, build (push) Has been cancelled
build-and-deploy / Upload Turbopack Bytesize metrics to Datadog (push) Has been cancelled
Rspack Next.js development integration tests / Rspack integration tests (push) Has been cancelled
Rspack Next.js production integration tests / Rspack integration tests (push) Has been cancelled
Turbopack Next.js development integration tests / Next.js integration tests (push) Has been cancelled
Turbopack Next.js production integration tests / Next.js integration tests (push) Has been cancelled
Update Rspack test manifest / Update and upload Rspack development test manifest (push) Has been cancelled
Update Rspack test manifest / Update and upload Rspack production test manifest (push) Has been cancelled
Upload bundler test manifests to areweturboyet.com / Upload test results (push) Has been cancelled
Update React / create-pull-request (push) Has been cancelled
test-e2e-project-reset-cron / reset-test-project (push) Has been cancelled
Notify about the top 15 issues/PRs/feature requests (most reacted) in the last 90 days / run (push) Has been cancelled
first commit
2026-03-10 19:37:31 +03:30

119 lines
3.0 KiB
JavaScript

// @ts-check
import fetch from 'node-fetch'
export async function main() {
const releasesArray = await fetch(
'https://api.github.com/repos/vercel/next.js/releases?per_page=100'
).then((r) => r.json())
const allReleases = releasesArray
.map(({ id, tag_name, created_at, body }) => ({
id,
tag_name,
created_at,
body: body
.replace(/\r\n/g, '\n')
.split('\n')
.map((e) => e.trim()),
}))
.sort((a, b) => a.created_at.localeCompare(b.created_at))
// targetVersion format is `13.4.15-`, generating changes for all 13.4.15-* canary releases
const targetVersion = /v(.*?-)/
.exec(allReleases.filter((e) => /v.*?-/.exec(e.tag_name)).pop().tag_name)
.pop()
const releases = allReleases.filter((v) => v.tag_name.includes(targetVersion))
const lineItems = {
'### Core Changes': [],
'### Minor Changes': [],
'### Documentation Changes': [],
'### Example Changes': [],
'### Misc Changes': [],
'### Patches': [],
'### Credits': [],
}
Object.keys(lineItems).forEach((header) => {
releases.forEach((release) => {
const headerIndex = release.body.indexOf(header)
if (!~headerIndex) return
let headerLastIndex = release.body
.slice(headerIndex + 1)
.findIndex((v) => v.startsWith('###'))
if (~headerLastIndex) {
headerLastIndex = headerLastIndex + headerIndex
} else {
headerLastIndex = release.body.length - 1
}
if (header === '### Credits') {
release.body.slice(headerIndex, headerLastIndex + 1).forEach((e) => {
const re = /@[a-z\d](?:[a-z\d]|-(?=[a-z\d])){0,38}/gi
let m
do {
m = re.exec(e)
if (m) {
lineItems[header].push(m.pop())
}
} while (m)
})
} else {
release.body.slice(headerIndex, headerLastIndex + 1).forEach((e) => {
if (!e.startsWith('-')) {
return
}
lineItems[header].push(e)
})
}
})
})
let finalMessage = []
Object.keys(lineItems).forEach((header) => {
let items = lineItems[header]
if (!items.length) {
return
}
finalMessage.push(header)
finalMessage.push('')
if (header === '### Credits') {
items = [...new Set(items)]
let creditsMessage = `Huge thanks to `
if (items.length > 1) {
creditsMessage += items.slice(0, items.length - 1).join(`, `)
creditsMessage += `, and `
}
creditsMessage += items[items.length - 1]
creditsMessage += ` for helping!`
finalMessage.push(creditsMessage)
} else {
items.forEach((e) => finalMessage.push(e))
}
finalMessage.push('')
})
return {
version: targetVersion.slice(0, -1),
firstVersion: releases[0].tag_name,
lastVersion: releases[releases.length - 1].tag_name,
content: finalMessage.join('\n'),
}
}
main().then((result) => {
console.log(result.content)
})