sass-site/source/_data/releases.cjs

103 lines
2.9 KiB
JavaScript
Raw Normal View History

2023-02-02 15:44:29 +00:00
const { spawn: nodeSpawn } = require('node:child_process');
const fs = require('node:fs/promises');
2023-02-03 22:58:02 +00:00
const deepEqual = require('deep-equal');
2023-02-02 15:44:29 +00:00
2023-02-02 15:52:26 -05:00
const chalk = require('kleur');
2023-02-02 15:44:29 +00:00
const VERSION_CACHE_PATH = './source/_data/versionCache.json';
// Promise version of `spawn` to avoid blocking the main thread while waiting
// for the child processes
function spawn(cmd, args, options) {
return new Promise((resolve, reject) => {
const child = nodeSpawn(cmd, args, options);
const stderr = [];
const stdout = [];
child.stdout.on('data', (data) => {
stdout.push(data.toString());
});
child.on('error', (e) => {
stderr.push(e.toString());
});
child.on('close', () => {
if (stderr.length) reject(stderr.join(''));
else resolve(stdout.join(''));
});
});
}
async function getCacheFile() {
2023-02-02 16:31:55 -05:00
if (process.env.NETLIFY || process.env.REBUILD_VERSION_CACHE) return {};
2023-02-02 15:44:29 +00:00
let versionCache;
try {
versionCache = JSON.parse(await fs.readFile(VERSION_CACHE_PATH));
} catch (err) {
if (err.code === 'ENOENT') {
versionCache = {}; // Cache is missing and needs to be created
} else {
throw err;
}
}
return versionCache;
}
async function writeCacheFile(cache) {
// eslint-disable-next-line no-console
console.info(chalk.green(`[11ty] Writing version cache file...`));
await fs.writeFile(VERSION_CACHE_PATH, JSON.stringify(cache));
}
2023-02-02 15:44:29 +00:00
// Retrieve the highest stable version of `repo`, based on its git tags
async function getLatestVersion(repo) {
2023-02-02 15:52:26 -05:00
// eslint-disable-next-line no-console
console.info(chalk.cyan(`[11ty] Fetching version information for ${repo}`));
2023-02-02 15:44:29 +00:00
const { parseSemVer, compareSemVer } = await import('semver-parser');
let stdout;
try {
stdout = await spawn(
'git',
['ls-remote', '--tags', '--refs', `https://github.com/${repo}`],
{ env: { ...process.env, GIT_TERMINAL_PROMPT: 0 } },
);
} catch (err) {
2023-02-02 15:52:26 -05:00
// eslint-disable-next-line no-console
console.error(chalk.red(`[11ty] Failed to fetch git tags for ${repo}`));
2023-02-02 15:44:29 +00:00
throw err;
}
const isNotPreRelease = (version) => {
const parsed = parseSemVer(version);
return parsed.matches && !parsed.pre;
};
const version = stdout
2023-02-02 15:44:29 +00:00
.split('\n')
2023-02-02 15:52:26 -05:00
.map((line) => line.split('refs/tags/').at(-1))
2023-02-02 15:44:29 +00:00
.filter(isNotPreRelease)
.sort(compareSemVer)
2023-02-02 15:52:26 -05:00
.at(-1);
2023-02-02 15:44:29 +00:00
return version;
2023-02-02 15:44:29 +00:00
}
module.exports = async () => {
const repos = ['sass/libsass', 'sass/dart-sass', 'sass/migrator'];
const cache = await getCacheFile();
2023-02-03 22:58:02 +00:00
const versions = await Promise.all(
repos.map(async (repo) => [
repo,
2023-02-06 12:52:31 -05:00
cache[repo] ?? (await getLatestVersion(repo)),
2023-02-03 22:58:02 +00:00
]),
);
const data = Object.fromEntries(
versions.map(([repo, version]) => [
repo.replace('sass/', ''),
{ version, url: `https://github.com/${repo}/releases/tag/${version}` },
]),
);
const nextCache = Object.fromEntries(versions);
if (!deepEqual(cache, nextCache)) await writeCacheFile(nextCache);
2023-02-02 15:44:29 +00:00
return data;
};