joplin/packages/tools/postPreReleasesToForum.ts

186 lines
5.4 KiB
TypeScript
Raw Normal View History

import { pathExists } from 'fs-extra';
import { readFile, writeFile } from 'fs/promises';
import { GitHubRelease, gitHubLatestReleases, gitHubLinkify } from './tool-utils';
import { config, createPost, createTopic, getForumTopPostByExternalId, getTopicByExternalId, updatePost } from './utils/discourse';
import { compareVersions } from 'compare-versions';
import dayjs = require('dayjs');
import { getRootDir } from '@joplin/utils';
interface State {
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
processedReleases: Record<string, any>;
}
enum Platform {
Desktop = 'Desktop',
Android = 'Android',
iOS = 'iOS',
}
const stateFilePath = `${__dirname}/postPreReleasesToForum.json`;
const betaCategoryId = 10;
const loadState = async (): Promise<State> => {
if (await pathExists(stateFilePath)) {
const content = await readFile(stateFilePath, 'utf-8');
return JSON.parse(content) as State;
}
return {
processedReleases: {},
};
};
const saveState = async (state: State) => {
await writeFile(stateFilePath, JSON.stringify(state, null, '\t'), 'utf-8');
};
const getPatchVersion = (tagName: string) => {
if (tagName.includes('-')) tagName = tagName.split('-')[1];
return tagName.substring(1);
};
const getMinorVersion = (tagName: string) => {
const s = getPatchVersion(tagName).split('.');
return `${s[0]}.${s[1]}`;
};
const getExternalId = (platform: string, minorVersion: string) => {
let prefix = '';
if (platform !== Platform.Desktop) {
prefix = `${platform.toLowerCase()}-`;
}
return `prerelease-${prefix}${minorVersion.replace(/\./g, '-')}`;
};
const getDownloadInfo = (platform: Platform) => {
const infos: Record<Platform, string> = {
[Platform.Desktop]: 'Download the latest pre-releases from here: <https://github.com/laurent22/joplin/releases>',
[Platform.Android]: 'Download the latest pre-releases from here: <https://github.com/laurent22/joplin-android/tags>',
[Platform.iOS]: 'In order to try the iOS pre-release, you will need to join the Joplin iOS beta program from here: <https://testflight.apple.com/join/p5iLVzrG>',
};
return infos[platform];
};
const getReleasesFromMarkdown = async (filePath: string) => {
const content = await readFile(filePath, 'utf-8');
const lines = content.split('\n');
const releases: GitHubRelease[] = [];
let currentRelease: GitHubRelease = null;
for (let line of lines) {
line = line.trim();
if (!line) continue;
if (line.startsWith('##')) {
const matches = line.match(/## \[(.*?)\]/);
if (!matches) throw new Error(`Could not parse version: ${line}`);
const tag = matches[1];
currentRelease = {
tag_name: tag,
body: '',
assets: [],
draft: false,
html_url: `https://github.com/laurent22/joplin-android/releases/tag/${tag}`,
prerelease: false,
upload_url: '',
};
releases.push(currentRelease);
continue;
}
if (currentRelease) {
if (currentRelease.body) currentRelease.body += '\n';
currentRelease.body += line;
}
}
releases.sort((a, b) => compareVersions(getPatchVersion(a.tag_name), getPatchVersion(b.tag_name)));
return releases;
};
const processReleases = async (releases: GitHubRelease[], platform: Platform, startFromVersion: string, state: State) => {
for (const release of releases) {
const minorVersion = getMinorVersion(release.tag_name);
const patchVersion = getPatchVersion(release.tag_name);
if (compareVersions(startFromVersion, minorVersion) > 0) continue;
if (!state.processedReleases[release.tag_name]) {
console.info(`Processing release ${release.tag_name}`);
const externalId = getExternalId(platform, minorVersion);
const postBody = `## [v${patchVersion}](${release.html_url})\n\n${gitHubLinkify(release.body)}`;
let topic = await getTopicByExternalId(externalId);
const topicTitle = `${platform} pre-release v${minorVersion} is now available (Updated ${dayjs(new Date()).format('DD/MM/YYYY')})`;
if (!topic) {
console.info('No topic exists - creating one...');
topic = await createTopic({
title: topicTitle,
raw: `${getDownloadInfo(platform)}\n\n* * *\n\n${postBody}`,
category: betaCategoryId,
external_id: externalId,
});
} else {
console.info('A topic exists - appending the new pre-release to it...');
const topPost = await getForumTopPostByExternalId(externalId);
await updatePost(topPost.id, {
title: topicTitle,
raw: `${topPost.raw}\n\n${postBody}`,
edit_reason: 'Auto-updated by script',
});
await createPost(topic.id, {
raw: postBody,
});
}
state.processedReleases[release.tag_name] = true;
await saveState(state);
}
}
return state;
};
const main = async () => {
const rootDir = await getRootDir();
const argv = require('yargs').argv;
config.key = argv._[0];
config.username = argv._[1];
let state = await loadState();
{
const releases = await gitHubLatestReleases(1, 50);
releases.sort((a, b) => compareVersions(a.tag_name, b.tag_name));
state = await processReleases(releases, Platform.Desktop, '2.13', state);
}
{
const releases = await getReleasesFromMarkdown(`${rootDir}/readme/about/changelog/android.md`);
state = await processReleases(releases, Platform.Android, '2.14', state);
}
{
const releases = await getReleasesFromMarkdown(`${rootDir}/readme/about/changelog/ios.md`);
await processReleases(releases, Platform.iOS, '12.14', state);
}
};
main().catch((error) => {
console.error('Fatal error', error);
process.exit(1);
});