| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | #!/usr/bin/env node
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | const { GitProcess } = require('dugite') | 
					
						
							|  |  |  | const path = require('path') | 
					
						
							|  |  |  | const semver = require('semver') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | const notesGenerator = require('./notes.js') | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | const gitDir = path.resolve(__dirname, '..', '..') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) => { | 
					
						
							|  |  |  |   const response = await GitProcess.exec(args, gitDir) | 
					
						
							|  |  |  |   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
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  | async function getReleaseNotes (range) { | 
					
						
							|  |  |  |   const rangeList = range.split('..') || ['HEAD'] | 
					
						
							|  |  |  |   const to = rangeList.pop() | 
					
						
							|  |  |  |   const from = rangeList.pop() || (await getPreviousPoint(to)) | 
					
						
							|  |  |  |   console.log(`Generating release notes between ${from} and ${to}`) | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   const notes = await notesGenerator.get(from, to) | 
					
						
							|  |  |  |   const ret = { | 
					
						
							|  |  |  |     text: notesGenerator.render(notes) | 
					
						
							| 
									
										
										
										
											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 () { | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   if (process.argv.length > 3) { | 
					
						
							|  |  |  |     console.log('Use: script/release-notes/index.js [tag | tag1..tag2]') | 
					
						
							|  |  |  |     return 1 | 
					
						
							| 
									
										
										
										
											2018-06-21 18:06:23 +10:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-06 14:06:11 -06:00
										 |  |  |   const range = process.argv[2] || 'HEAD' | 
					
						
							|  |  |  |   const notes = await getReleaseNotes(range) | 
					
						
							|  |  |  |   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 |