import { execSync } from 'node:child_process' import { promises as fsp } from 'node:fs' import { $fetch } from 'ofetch' import { resolve } from 'pathe' import { globby } from 'globby' import { execaSync } from 'execa' import { determineSemverChange, getGitDiff, loadChangelogConfig, parseCommits } from 'changelogen' export interface Dep { name: string range: string type: string } type ThenArg = T extends PromiseLike ? U : T export type Package = ThenArg> export async function loadPackage(dir: string) { const pkgPath = resolve(dir, 'package.json') const data = JSON.parse(await fsp.readFile(pkgPath, 'utf-8').catch(() => '{}')) const save = () => fsp.writeFile(pkgPath, `${JSON.stringify(data, null, 2)}\n`) const updateDeps = (reviver: (dep: Dep) => Dep | void) => { for (const type of ['dependencies', 'devDependencies', 'optionalDependencies', 'peerDependencies']) { if (!data[type]) continue for (const e of Object.entries(data[type])) { const dep: Dep = { name: e[0], range: e[1] as string, type } delete data[type][dep.name] const updated = reviver(dep) || dep data[updated.type] = data[updated.type] || {} data[updated.type][updated.name] = updated.range } } } return { dir, data, save, updateDeps, } } export async function loadWorkspace(dir: string) { const workspacePkg = await loadPackage(dir) const pkgDirs = (await globby(['packages/*', 'apps/*'], { onlyDirectories: true })).sort() const packages: Package[] = [] for (const pkgDir of pkgDirs) { const pkg = await loadPackage(pkgDir) if (!pkg.data.name) continue packages.push(pkg) } const find = (name: string) => { const pkg = packages.find(pkg => pkg.data.name === name) if (!pkg) throw new Error(`Workspace package not found: ${name}`) return pkg } const rename = (from: string, to: string) => { find(from).data._name = find(from).data.name find(from).data.name = to for (const pkg of packages) { pkg.updateDeps((dep) => { if (dep.name === from && !dep.range.startsWith('npm:')) dep.range = `npm:${to}@${dep.range}` }) } } const setVersion = (name: string, newVersion: string, opts: { updateDeps?: boolean } = {}) => { find(name).data.version = newVersion if (!opts.updateDeps) return for (const pkg of packages) { pkg.updateDeps((dep) => { if (dep.name === name) dep.range = newVersion }) } } const save = () => Promise.all(packages.map(pkg => pkg.save())) return { dir, workspacePkg, packages, save, find, rename, setVersion, } } export async function determineBumpType() { const config = await loadChangelogConfig(process.cwd()) const commits = await getLatestCommits() const bumpType = determineSemverChange(commits, config) return bumpType === 'major' ? 'minor' : bumpType } export async function getLatestCommits() { const config = await loadChangelogConfig(process.cwd()) const latestTag = execaSync('git', ['describe', '--tags', '--abbrev=0']).stdout return parseCommits(await getGitDiff(latestTag), config) } export async function getContributors(repo: string, token: string) { const contributors = [] as Array<{ name: string, username: string }> const emails = new Set() const latestTag = execSync('git describe --tags --abbrev=0').toString().trim() const rawCommits = await getGitDiff(latestTag) for (const commit of rawCommits) { if (emails.has(commit.author.email) || commit.author.name === 'renovate[bot]') continue const { author } = await $fetch<{ author: { login: string, email: string } }>(`https://api.github.com/repos/${repo}/commits/${commit.shortHash}`, { headers: { Accept: 'application/vnd.github.v3+json', Authorization: `token ${token}`, }, }) if (!contributors.some(c => c.username === author.login)) contributors.push({ name: commit.author.name, username: author.login }) emails.add(author.email) } return contributors } export function isGitClean() { const result = execSync('git status --porcelain').toString().trim() return result.length === 0 } export function updateChangelogFile(fromCommit: string, toCommit: string) { execSync(`npx changelogen@latest --output --from ${fromCommit} --to ${toCommit}`) }