| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | #!/usr/bin/env node
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | const { GitProcess } = require('dugite') | 
					
						
							| 
									
										
										
										
											2019-01-10 14:01:38 -06:00
										 |  |  | const minimist = require('minimist') | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | const path = require('path') | 
					
						
							|  |  |  | const semver = require('semver') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-24 10:18:04 -07:00
										 |  |  | const { ELECTRON_DIR } = require('../../lib/utils') | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const notesGenerator = require('./notes.js') | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const semverify = version => version.replace(/^origin\//, '').replace('x', '0').replace(/-/g, '.') | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const runGit = async (args) => { | 
					
						
							| 
									
										
										
										
											2019-06-24 10:18:04 -07:00
										 |  |  |   const response = await GitProcess.exec(args, ELECTRON_DIR) | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   if (response.exitCode !== 0) { | 
					
						
							|  |  |  |     throw new Error(response.stderr.trim()) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   return response.stdout.trim() | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const tagIsSupported = tag => tag && !tag.includes('nightly') && !tag.includes('unsupported') | 
					
						
							|  |  |  | const tagIsBeta = tag => tag.includes('beta') | 
					
						
							|  |  |  | const tagIsStable = tag => tagIsSupported(tag) && !tagIsBeta(tag) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const getTagsOf = async (point) => { | 
					
						
							|  |  |  |   return (await runGit(['tag', '--merged', point])) | 
					
						
							|  |  |  |     .split('\n') | 
					
						
							|  |  |  |     .map(tag => tag.trim()) | 
					
						
							|  |  |  |     .filter(tag => semver.valid(tag)) | 
					
						
							|  |  |  |     .sort(semver.compare) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const getTagsOnBranch = async (point) => { | 
					
						
							|  |  |  |   const masterTags = await getTagsOf('master') | 
					
						
							|  |  |  |   if (point === 'master') { | 
					
						
							|  |  |  |     return masterTags | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   const masterTagsSet = new Set(masterTags) | 
					
						
							|  |  |  |   return (await getTagsOf(point)).filter(tag => !masterTagsSet.has(tag)) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const getBranchOf = async (point) => { | 
					
						
							|  |  |  |   const branches = (await runGit(['branch', '-a', '--contains', point])) | 
					
						
							|  |  |  |     .split('\n') | 
					
						
							|  |  |  |     .map(branch => branch.trim()) | 
					
						
							|  |  |  |     .filter(branch => !!branch) | 
					
						
							|  |  |  |   const current = branches.find(branch => branch.startsWith('* ')) | 
					
						
							|  |  |  |   return current ? current.slice(2) : branches.shift() | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const getAllBranches = async () => { | 
					
						
							|  |  |  |   return (await runGit(['branch', '--remote'])) | 
					
						
							|  |  |  |     .split('\n') | 
					
						
							|  |  |  |     .map(branch => branch.trim()) | 
					
						
							|  |  |  |     .filter(branch => !!branch) | 
					
						
							|  |  |  |     .filter(branch => branch !== 'origin/HEAD -> origin/master') | 
					
						
							|  |  |  |     .sort() | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const getStabilizationBranches = async () => { | 
					
						
							|  |  |  |   return (await getAllBranches()) | 
					
						
							|  |  |  |     .filter(branch => /^origin\/\d+-\d+-x$/.test(branch)) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const getPreviousStabilizationBranch = async (current) => { | 
					
						
							|  |  |  |   const stabilizationBranches = (await getStabilizationBranches()) | 
					
						
							|  |  |  |     .filter(branch => branch !== current && branch !== `origin/${current}`) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   if (!semver.valid(current)) { | 
					
						
							|  |  |  |     // since we don't seem to be on a stabilization branch right now,
 | 
					
						
							|  |  |  |     // pick a placeholder name that will yield the newest branch
 | 
					
						
							|  |  |  |     // as a comparison point.
 | 
					
						
							|  |  |  |     current = 'v999.999.999' | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   let newestMatch = null | 
					
						
							|  |  |  |   for (const branch of stabilizationBranches) { | 
					
						
							|  |  |  |     if (semver.gte(semverify(branch), semverify(current))) { | 
					
						
							|  |  |  |       continue | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |     if (newestMatch && semver.lte(semverify(branch), semverify(newestMatch))) { | 
					
						
							|  |  |  |       continue | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |     newestMatch = branch | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   return newestMatch | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const getPreviousPoint = async (point) => { | 
					
						
							|  |  |  |   const currentBranch = await getBranchOf(point) | 
					
						
							|  |  |  |   const currentTag = (await getTagsOf(point)).filter(tag => tagIsSupported(tag)).pop() | 
					
						
							|  |  |  |   const currentIsStable = tagIsStable(currentTag) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   try { | 
					
						
							|  |  |  |     // First see if there's an earlier tag on the same branch
 | 
					
						
							|  |  |  |     // that can serve as a reference point.
 | 
					
						
							|  |  |  |     let tags = (await getTagsOnBranch(`${point}^`)).filter(tag => tagIsSupported(tag)) | 
					
						
							|  |  |  |     if (currentIsStable) { | 
					
						
							|  |  |  |       tags = tags.filter(tag => tagIsStable(tag)) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |     if (tags.length) { | 
					
						
							|  |  |  |       return tags.pop() | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   } catch (error) { | 
					
						
							|  |  |  |     console.log('error', error) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Otherwise, use the newest stable release that preceeds this branch.
 | 
					
						
							|  |  |  |   // To reach that you may have to walk past >1 branch, e.g. to get past
 | 
					
						
							|  |  |  |   // 2-1-x which never had a stable release.
 | 
					
						
							|  |  |  |   let branch = currentBranch | 
					
						
							|  |  |  |   while (branch) { | 
					
						
							|  |  |  |     const prevBranch = await getPreviousStabilizationBranch(branch) | 
					
						
							|  |  |  |     const tags = (await getTagsOnBranch(prevBranch)).filter(tag => tagIsStable(tag)) | 
					
						
							|  |  |  |     if (tags.length) { | 
					
						
							|  |  |  |       return tags.pop() | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |     branch = prevBranch | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-10 14:01:38 -06:00
										 |  |  | async function getReleaseNotes (range, newVersion, explicitLinks) { | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   const rangeList = range.split('..') || ['HEAD'] | 
					
						
							|  |  |  |   const to = rangeList.pop() | 
					
						
							|  |  |  |   const from = rangeList.pop() || (await getPreviousPoint(to)) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-10 14:01:38 -06:00
										 |  |  |   if (!newVersion) { | 
					
						
							|  |  |  |     newVersion = to | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   console.log(`Generating release notes between ${from} and ${to} for version ${newVersion}`) | 
					
						
							|  |  |  |   const notes = await notesGenerator.get(from, to, newVersion) | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   const ret = { | 
					
						
							| 
									
										
										
										
											2018-12-19 09:48:01 -06:00
										 |  |  |     text: notesGenerator.render(notes, explicitLinks) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   if (notes.unknown.length) { | 
					
						
							|  |  |  |     ret.warning = `You have ${notes.unknown.length} unknown release notes. Please fix them before releasing.` | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   return ret | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 18:13:19 +10:00
										 |  |  | async function main () { | 
					
						
							| 
									
										
										
										
											2019-01-10 14:01:38 -06:00
										 |  |  |   const opts = minimist(process.argv.slice(2), { | 
					
						
							|  |  |  |     boolean: [ 'explicit-links', 'help' ], | 
					
						
							|  |  |  |     string: [ 'version' ] | 
					
						
							|  |  |  |   }) | 
					
						
							|  |  |  |   opts.range = opts._.shift() | 
					
						
							|  |  |  |   if (opts.help || !opts.range) { | 
					
						
							|  |  |  |     const name = path.basename(process.argv[1]) | 
					
						
							|  |  |  |     console.log(`
 | 
					
						
							|  |  |  | easy usage: ${name} version | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | full usage: ${name} [begin..]end [--version version] [--explicit-links] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  * 'begin' and 'end' are two git references -- tags, branches, etc -- | 
					
						
							|  |  |  |    from which the release notes are generated. | 
					
						
							|  |  |  |  * if omitted, 'begin' defaults to the previous tag in end's branch. | 
					
						
							|  |  |  |  * if omitted, 'version' defaults to 'end'. Specifying a version is | 
					
						
							|  |  |  |    useful if you're making notes on a new version that isn't tagged yet. | 
					
						
							|  |  |  |  * 'explicit-links' makes every note's issue, commit, or pull an MD link | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | For example, these invocations are equivalent: | 
					
						
							|  |  |  |   ${process.argv[1]} v4.0.1 | 
					
						
							|  |  |  |   ${process.argv[1]} v4.0.0..v4.0.1 --version v4.0.1 | 
					
						
							|  |  |  | `)
 | 
					
						
							|  |  |  |     return 0 | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-10 14:01:38 -06:00
										 |  |  |   const notes = await getReleaseNotes(opts.range, opts.version, opts['explicit-links']) | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   console.log(notes.text) | 
					
						
							|  |  |  |   if (notes.warning) { | 
					
						
							|  |  |  |     throw new Error(notes.warning) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if (process.mainModule === module) { | 
					
						
							|  |  |  |   main().catch((err) => { | 
					
						
							|  |  |  |     console.error('Error Occurred:', err) | 
					
						
							|  |  |  |     process.exit(1) | 
					
						
							|  |  |  |   }) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | module.exports = getReleaseNotes |