Skip to content

Commit 68fc468

Browse files
committed
chore: Analyse Next.js releases for app router changes
Having an email on every canary release can be too verbose, I'm only interested in being notified of releases that change the app router core code, which may impact `nuqs`.
1 parent a95d587 commit 68fc468

File tree

5 files changed

+446
-17
lines changed

5 files changed

+446
-17
lines changed

.github/workflows/test-against-nextjs-release.yml

+22
Original file line numberDiff line numberDiff line change
@@ -62,3 +62,25 @@ jobs:
6262
steps:
6363
- name: Invalidate ISR cache for GitHub Actions status on landing page
6464
run: curl -s "https://nuqs.47ng.com/api/isr?tag=github-actions-status&token=${{ secrets.ISR_TOKEN }}"
65+
66+
analyse-release:
67+
runs-on: ubuntu-latest
68+
name: Check for app router changes
69+
steps:
70+
- uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11
71+
- uses: pnpm/action-setup@d882d12c64e032187b2edb46d3a0d003b7a43598
72+
with:
73+
version: 8
74+
- uses: actions/setup-node@b39b52d1213e96004bfcb1c61a8a6fa8ab84f3e8
75+
with:
76+
node-version: lts/*
77+
cache: pnpm
78+
- name: Install dependencies
79+
run: pnpm install
80+
- name: Check for changes in app router
81+
run: ./next-release-analyser.mjs
82+
working-directory: packages/scripts
83+
env:
84+
MAILPACE_API_TOKEN: ${{ secrets.MAILPACE_API_TOKEN }}
85+
EMAIL_ADDRESS_TO: ${{ secrets.EMAIL_ADDRESS_TO }}
86+
EMAIL_ADDRESS_FROM: ${{ secrets.EMAIL_ADDRESS_FROM }}
+97
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
#!/usr/bin/env zx
2+
// @ts-check
3+
4+
import MailPace from '@mailpace/mailpace.js'
5+
import { createEnv } from '@t3-oss/env-core'
6+
import { z } from 'zod'
7+
import 'zx/globals'
8+
9+
const canaryRegexp = /^(\d+)\.(\d+)\.(\d+)-canary\.(\d+)$/
10+
11+
const env = createEnv({
12+
server: {
13+
MAILPACE_API_TOKEN: z.string(),
14+
EMAIL_ADDRESS_FROM: z.string().email(),
15+
EMAIL_ADDRESS_TO: z.string().email()
16+
},
17+
isServer: true,
18+
runtimeEnv: process.env
19+
})
20+
21+
main()
22+
23+
const fileSchema = z.object({
24+
filename: z.string(),
25+
patch: z.string().optional()
26+
})
27+
28+
async function main() {
29+
const thisVersion = argv.version
30+
const previousVersion = getPreviousVersion(argv.version)
31+
32+
if (!previousVersion) {
33+
console.log('No previous version to compare with')
34+
process.exit(0)
35+
}
36+
const compareURL = `https://api.github.com/repos/vercel/next.js/compare/v${previousVersion}...v${thisVersion}`
37+
const compare = await fetch(compareURL).then(res => res.json())
38+
const files = z.array(fileSchema).parse(compare.files)
39+
const appRouterFile = files.find(
40+
file =>
41+
file.filename === 'packages/next/src/client/components/app-router.tsx'
42+
)
43+
if (!appRouterFile) {
44+
console.log('No changes in app-router.tsx')
45+
process.exit(0)
46+
}
47+
sendNotificationEmail(thisVersion, appRouterFile)
48+
}
49+
50+
// --
51+
52+
/**
53+
* @param {string} version
54+
*/
55+
function getPreviousVersion(version) {
56+
const match = canaryRegexp.exec(version)
57+
if (!match || !match[4]) {
58+
return null
59+
}
60+
const canary = parseInt(match[4])
61+
if (canary === 0) {
62+
return null
63+
}
64+
return `${match[1]}.${match[2]}.${match[3]}-canary.${canary - 1}`
65+
}
66+
67+
/**
68+
*
69+
* @param {string} thisVersion
70+
* @param {z.infer<typeof fileSchema>} appRouterFile
71+
* @returns
72+
*/
73+
function sendNotificationEmail(thisVersion, appRouterFile) {
74+
const client = new MailPace.DomainClient(env.MAILPACE_API_TOKEN)
75+
const body = `Changes to the app router have been published in Next.js ${thisVersion}.
76+
77+
Release: https://github.com/vercel/next.js/releases/tag/v${thisVersion}
78+
79+
${
80+
appRouterFile.patch
81+
? `The patch is:
82+
\`\`\`diff
83+
${appRouterFile.patch}
84+
\`\`\``
85+
: 'No patch available'
86+
}
87+
`
88+
console.info('Sending email')
89+
console.info(body)
90+
return client.sendEmail({
91+
from: env.EMAIL_ADDRESS_FROM,
92+
to: env.EMAIL_ADDRESS_TO,
93+
subject: `[nuqs] Next.js ${thisVersion} has app router changes`,
94+
textbody: body,
95+
tags: ['nuqs']
96+
})
97+
}

packages/scripts/package.json

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
{
2+
"name": "scripts",
3+
"private": true,
4+
"version": "0.0.0-internal",
5+
"type": "module",
6+
"dependencies": {
7+
"@mailpace/mailpace.js": "^0.1.1",
8+
"@t3-oss/env-core": "^0.9.2",
9+
"zod": "^3.22.4",
10+
"zx": "^7.2.3"
11+
},
12+
"devDependencies": {
13+
"typescript": "^5.3.3"
14+
}
15+
}

packages/scripts/tsconfig.json

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
{
2+
"$schema": "https://json.schemastore.org/tsconfig",
3+
"compilerOptions": {
4+
// Type checking
5+
"strict": true,
6+
"noUncheckedIndexedAccess": true,
7+
"alwaysStrict": false, // Don't emit "use strict" to avoid conflicts with "use client"
8+
// Modules
9+
"module": "ESNext",
10+
"moduleResolution": "Bundler",
11+
"resolveJsonModule": true,
12+
// Language & Environment
13+
"target": "ESNext",
14+
"lib": ["DOM", "DOM.Iterable", "ESNext"],
15+
// Emit
16+
"noEmit": true,
17+
"declaration": true,
18+
"declarationMap": true,
19+
"verbatimModuleSyntax": true,
20+
"moduleDetection": "force",
21+
22+
"downlevelIteration": true,
23+
// Interop
24+
"allowJs": true,
25+
"isolatedModules": true,
26+
"esModuleInterop": true,
27+
"forceConsistentCasingInFileNames": true,
28+
// Misc
29+
"skipLibCheck": true,
30+
"skipDefaultLibCheck": true,
31+
"incremental": true,
32+
"tsBuildInfoFile": ".tsbuildinfo"
33+
},
34+
"include": ["next-release-analyser.mjs"],
35+
"exclude": ["node_modules"]
36+
}

0 commit comments

Comments
 (0)