import got from 'got'; import fs from 'fs'; import { GithubTag, TagCompare } from '../@types/github'; import path from 'path'; import { UpdateFile } from '../@types/updateFile'; import packageJson from '../package.json'; import { CompilerOptions, transpileModule } from 'typescript'; import tsConfig from '../tsconfig.json'; import fsextra from 'fs-extra'; import seiHelper from 'sei-helper'; import { workingDir } from './module.cfg-loader'; import { console } from './log'; const updateFilePlace = path.join(workingDir, 'config', 'updates.json'); const updateIgnore = [ '*.d.ts', '.git', 'lib', 'node_modules', '@types', path.join('bin', 'mkvtoolnix'), path.join('config', 'token.yml'), '.eslint', 'tsconfig.json', 'updates.json', 'tsc.ts' ]; const askBeforeUpdate = [ '*.yml' ]; enum ApplyType { DELETE, ADD, UPDATE } export type ApplyItem = { type: ApplyType, path: string, content: string } export default (async (force = false) => { const isPackaged = (process as NodeJS.Process & { pkg?: unknown }).pkg ? true : !!process.env.contentDirectory; if (isPackaged) { return; } let updateFile: UpdateFile|undefined; if (fs.existsSync(updateFilePlace)) { updateFile = JSON.parse(fs.readFileSync(updateFilePlace).toString()) as UpdateFile; if (new Date() < new Date(updateFile.nextCheck) && !force) { return; } } console.info('Checking for updates...'); const tagRequest = await got('https://api.github.com/repos/anidl/multi-downloader-nx/tags'); const tags = JSON.parse(tagRequest.body) as GithubTag[]; if (tags.length > 0) { const newer = tags.filter(a => { return isNewer(packageJson.version, a.name); }); console.info(`Found ${tags.length} release tags and ${newer.length} that are new.`); if (newer.length < 1) { console.info('No new tags found'); return done(); } const newest = newer.sort((a, b) => a.name < b.name ? 1 : a.name > b.name ? -1 : 0)[0]; const compareRequest = await got(`https://api.github.com/repos/anidl/multi-downloader-nx/compare/${packageJson.version}...${newest.name}`); const compareJSON = JSON.parse(compareRequest.body) as TagCompare; console.info(`You are behind by ${compareJSON.ahead_by} releases!`); const changedFiles = compareJSON.files.map(a => ({ ...a, filename: path.join(...a.filename.split('/')) })).filter(a => { return !updateIgnore.some(_filter => matchString(_filter, a.filename)); }); if (changedFiles.length < 1) { console.info('No file changes found... updating package.json. If you think this is an error please get the newst version yourself.'); return done(newest.name); } console.info(`Found file changes: \n${changedFiles.map(a => ` [${ a.status === 'modified' ? '*' : a.status === 'added' ? '+' : '-' }] ${a.filename}`).join('\n')}`); const remove: string[] = []; for (const a of changedFiles.filter(a => a.status !== 'added')) { if (!askBeforeUpdate.some(pattern => matchString(pattern, a.filename))) continue; const answer = await seiHelper.question(`The developer decided that the file '${a.filename}' may contain information you changed yourself. Should they be overriden to be updated? [y/N]`); if (answer.toLowerCase() === 'y') remove.push(a.sha); } const changesToApply = await Promise.all(changedFiles.filter(a => !remove.includes(a.sha)).map(async (a): Promise => { if (a.filename.endsWith('.ts') || a.filename.endsWith('tsx')) { const isTSX = a.filename.endsWith('tsx'); const ret = { path: a.filename.slice(0, isTSX ? -3 : -2) + `js${isTSX ? 'x' : ''}`, content: transpileModule((await got(a.raw_url)).body, { compilerOptions: tsConfig.compilerOptions as unknown as CompilerOptions }).outputText, type: a.status === 'modified' ? ApplyType.UPDATE : a.status === 'added' ? ApplyType.ADD : ApplyType.DELETE }; console.info('✓ Transpiled %s', ret.path); return ret; } else { const ret = { path: a.filename, content: (await got(a.raw_url)).body, type: a.status === 'modified' ? ApplyType.UPDATE : a.status === 'added' ? ApplyType.ADD : ApplyType.DELETE }; console.info('✓ Got %s', ret.path); return ret; } })); changesToApply.forEach(a => { try { fsextra.ensureDirSync(path.dirname(a.path)); fs.writeFileSync(path.join(__dirname, '..', a.path), a.content); console.info('✓ Written %s', a.path); } catch (er) { console.info('✗ Error while writing %s', a.path); } }); console.info('Done'); return done(); } }); function done(newVersion?: string) { const next = new Date(Date.now() + 1000 * 60 * 60 * 24); fs.writeFileSync(updateFilePlace, JSON.stringify({ lastCheck: Date.now(), nextCheck: next.getTime() } as UpdateFile, null, 2)); if (newVersion) { fs.writeFileSync('../package.json', JSON.stringify({ ...packageJson, version: newVersion }, null, 4)); } console.info('[INFO] Searching for update finished. Next time running on the ' + next.toLocaleDateString() + ' at ' + next.toLocaleTimeString() + '.'); } function isNewer(curr: string, compare: string) : boolean { const currParts = curr.split('.').map(a => parseInt(a)); const compareParts = compare.split('.').map(a => parseInt(a)); for (let i = 0; i < Math.max(currParts.length, compareParts.length); i++) { if (currParts.length <= i) return true; if (compareParts.length <= i) return false; if (currParts[i] !== compareParts[i]) return compareParts[i] > currParts[i]; } return false; } function matchString(pattern: string, toMatch: string) : boolean { const filter = path.join('..', pattern); if (pattern.startsWith('*')) { return toMatch.endsWith(pattern.slice(1)); } else if (filter.split(path.sep).pop()?.indexOf('.') === -1) { return toMatch.startsWith(filter); } else { return toMatch.split(path.sep).pop() === pattern; } }