const path = require('path'); const fs = require('fs'); const semver = require('semver'); const { GitProcess } = require('dugite'); const { promisify } = require('util'); const { ELECTRON_DIR } = require('../lib/utils'); const readFile = promisify(fs.readFile); const preType = { NONE: 'none', PARTIAL: 'partial', FULL: 'full' }; const getCurrentDate = () => { const d = new Date(); const dd = `${d.getDate()}`.padStart(2, '0'); const mm = `${d.getMonth() + 1}`.padStart(2, '0'); const yyyy = d.getFullYear(); return `${yyyy}${mm}${dd}`; }; const isNightly = v => v.includes('nightly'); const isAlpha = v => v.includes('alpha'); const isBeta = v => v.includes('beta'); const isStable = v => { const parsed = semver.parse(v); return !!(parsed && parsed.prerelease.length === 0); }; const makeVersion = (components, delim, pre = preType.NONE) => { let version = [components.major, components.minor, components.patch].join(delim); if (pre === preType.PARTIAL) { version += `${delim}${components.pre[1] || 0}`; } else if (pre === preType.FULL) { version += `-${components.pre[0]}${delim}${components.pre[1]}`; } return version; }; async function nextAlpha (v) { const next = semver.coerce(semver.clean(v)); const tagBlob = await GitProcess.exec(['tag', '--list', '-l', `v${next}-alpha.*`], ELECTRON_DIR); const tags = tagBlob.stdout.split('\n').filter(e => e !== ''); tags.sort((t1, t2) => { const a = parseInt(t1.split('.').pop(), 10); const b = parseInt(t2.split('.').pop(), 10); return a - b; }); // increment the latest existing alpha tag or start at alpha.1 if it's a new alpha line return tags.length === 0 ? `${next}-alpha.1` : semver.inc(tags.pop(), 'prerelease'); } async function nextBeta (v) { const next = semver.coerce(semver.clean(v)); const tagBlob = await GitProcess.exec(['tag', '--list', '-l', `v${next}-beta.*`], ELECTRON_DIR); const tags = tagBlob.stdout.split('\n').filter(e => e !== ''); tags.sort((t1, t2) => { const a = parseInt(t1.split('.').pop(), 10); const b = parseInt(t2.split('.').pop(), 10); return a - b; }); // increment the latest existing beta tag or start at beta.1 if it's a new beta line return tags.length === 0 ? `${next}-beta.1` : semver.inc(tags.pop(), 'prerelease'); } async function nextNightly (v) { let next = semver.valid(semver.coerce(v)); const pre = `nightly.${getCurrentDate()}`; const branch = (await GitProcess.exec(['rev-parse', '--abbrev-ref', 'HEAD'], ELECTRON_DIR)).stdout.trim(); // TODO(main-migration): Simplify once main branch is renamed if (branch === 'master' || branch === 'main') { next = semver.inc(await getLastMajorForMain(), 'major'); } else if (isStable(v)) { next = semver.inc(next, 'patch'); } return `${next}-${pre}`; } async function getLastMajorForMain () { let branchNames; const result = await GitProcess.exec(['branch', '-a', '--remote', '--list', 'origin/[0-9]*-x-y'], ELECTRON_DIR); if (result.exitCode === 0) { branchNames = result.stdout.trim().split('\n'); const filtered = branchNames.map(b => b.replace('origin/', '')); return getNextReleaseBranch(filtered); } else { throw new Error('Release branches could not be fetched.'); } } function getNextReleaseBranch (branches) { const converted = branches.map(b => b.replace(/-/g, '.').replace('x', '0').replace('y', '0')); return converted.reduce((v1, v2) => semver.gt(v1, v2) ? v1 : v2); } module.exports = { isStable, isAlpha, isBeta, isNightly, nextAlpha, nextBeta, makeVersion, nextNightly, preType };