| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  | import * as core from '@actions/core'; | 
					
						
							|  |  |  | import NotImplementedException from './error/not-implemented-exception'; | 
					
						
							|  |  |  | import ValidationError from './error/validation-error'; | 
					
						
							| 
									
										
										
										
											2020-05-21 15:44:56 +00:00
										 |  |  | import Input from './input'; | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  | import System from './system'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default class Versioning { | 
					
						
							| 
									
										
										
										
											2020-05-20 22:06:08 +00:00
										 |  |  |   static get projectPath() { | 
					
						
							| 
									
										
										
										
											2020-05-21 15:44:56 +00:00
										 |  |  |     return Input.projectPath; | 
					
						
							| 
									
										
										
										
											2020-05-20 22:06:08 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-21 12:45:52 +00:00
										 |  |  |   static get isDirtyAllowed() { | 
					
						
							| 
									
										
										
										
											2020-05-21 15:44:56 +00:00
										 |  |  |     return Input.allowDirtyBuild === 'true'; | 
					
						
							| 
									
										
										
										
											2020-05-21 12:45:52 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   static get strategies() { | 
					
						
							|  |  |  |     return { None: 'None', Semantic: 'Semantic', Tag: 'Tag', Custom: 'Custom' }; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Get the branch name of the (related) branch | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static get branch() { | 
					
						
							| 
									
										
										
										
											2020-04-27 23:43:15 +00:00
										 |  |  |     // Todo - use optional chaining (https://github.com/zeit/ncc/issues/534)
 | 
					
						
							|  |  |  |     return this.headRef || (this.ref && this.ref.slice(11)); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * For pull requests we can reliably use GITHUB_HEAD_REF | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static get headRef() { | 
					
						
							|  |  |  |     return process.env.GITHUB_HEAD_REF; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * For branches GITHUB_REF will have format `refs/heads/feature-branch-1` | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static get ref() { | 
					
						
							|  |  |  |     return process.env.GITHUB_REF; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-15 20:35:53 +00:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * The commit SHA that triggered the workflow run. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static get sha() { | 
					
						
							|  |  |  |     return process.env.GITHUB_SHA; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 23:48:46 +00:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * Maximum number of lines to print when logging the git diff | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static get maxDiffLines() { | 
					
						
							|  |  |  |     return 60; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Log up to maxDiffLines of the git diff. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async logDiff() { | 
					
						
							|  |  |  |     this.git(['--no-pager', 'diff', '|', 'head', '-n', this.maxDiffLines.toString()]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   /** | 
					
						
							|  |  |  |    * Regex to parse version description into separate fields | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static get descriptionRegex() { | 
					
						
							|  |  |  |     return /^v([\d.]+)-(\d+)-g(\w+)-?(\w+)*/g; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   static async determineVersion(strategy, inputVersion) { | 
					
						
							|  |  |  |     // Validate input
 | 
					
						
							|  |  |  |     if (!Object.hasOwnProperty.call(this.strategies, strategy)) { | 
					
						
							|  |  |  |       throw new ValidationError( | 
					
						
							|  |  |  |         `Versioning strategy should be one of ${Object.values(this.strategies).join(', ')}.`, | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let version; | 
					
						
							|  |  |  |     switch (strategy) { | 
					
						
							|  |  |  |       case this.strategies.None: | 
					
						
							|  |  |  |         version = 'none'; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case this.strategies.Custom: | 
					
						
							|  |  |  |         version = inputVersion; | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case this.strategies.Semantic: | 
					
						
							|  |  |  |         version = await this.generateSemanticVersion(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case this.strategies.Tag: | 
					
						
							|  |  |  |         version = await this.generateTagVersion(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         throw new NotImplementedException(`Strategy ${strategy} is not implemented.`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return version; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Automatically generates a version based on SemVer out of the box. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The version works as follows: `<major>.<minor>.<patch>` for example `0.1.2`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * The latest tag dictates `<major>.<minor>` | 
					
						
							|  |  |  |    * The number of commits since that tag dictates`<patch>`. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * @See: https://semver.org/
 | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async generateSemanticVersion() { | 
					
						
							| 
									
										
										
										
											2020-05-01 11:57:42 +00:00
										 |  |  |     await this.fetch(); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 23:48:46 +00:00
										 |  |  |     await this.logDiff(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-21 12:45:52 +00:00
										 |  |  |     if ((await this.isDirty()) && !this.isDirtyAllowed) { | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |       throw new Error('Branch is dirty. Refusing to base semantic version on uncommitted changes'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!(await this.hasAnyVersionTags())) { | 
					
						
							|  |  |  |       const version = `0.0.${await this.getTotalNumberOfCommits()}`; | 
					
						
							|  |  |  |       core.info(`Generated version ${version} (no version tags found).`); | 
					
						
							|  |  |  |       return version; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const { tag, commits, hash } = await this.parseSemanticVersion(); | 
					
						
							|  |  |  |     core.info(`Found semantic version ${tag}.${commits} for ${this.branch}@${hash}`); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return `${tag}.${commits}`; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Generate the proper version for unity based on an existing tag. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async generateTagVersion() { | 
					
						
							|  |  |  |     let tag = await this.getTag(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (tag.charAt(0) === 'v') { | 
					
						
							|  |  |  |       tag = tag.slice(1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return tag; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Parses the versionDescription into their named parts. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async parseSemanticVersion() { | 
					
						
							|  |  |  |     const description = await this.getVersionDescription(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 23:43:15 +00:00
										 |  |  |     try { | 
					
						
							|  |  |  |       const [match, tag, commits, hash] = this.descriptionRegex.exec(description); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return { | 
					
						
							|  |  |  |         match, | 
					
						
							|  |  |  |         tag, | 
					
						
							|  |  |  |         commits, | 
					
						
							|  |  |  |         hash, | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  |     } catch (error) { | 
					
						
							|  |  |  |       throw new Error(`Failed to parse git describe output: "${description}".`); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-01 11:57:42 +00:00
										 |  |  |   /** | 
					
						
							| 
									
										
										
										
											2020-05-01 12:11:37 +00:00
										 |  |  |    * Retrieves refs from the configured remote. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2020-05-01 12:17:30 +00:00
										 |  |  |    * Fetch unshallow for incomplete repository, but fall back to normal fetch. | 
					
						
							|  |  |  |    * | 
					
						
							| 
									
										
										
										
											2020-05-01 12:11:37 +00:00
										 |  |  |    * Note: `--all` should not be used, and would break fetching for push event. | 
					
						
							| 
									
										
										
										
											2020-05-01 11:57:42 +00:00
										 |  |  |    */ | 
					
						
							|  |  |  |   static async fetch() { | 
					
						
							| 
									
										
										
										
											2020-05-01 12:17:30 +00:00
										 |  |  |     try { | 
					
						
							| 
									
										
										
										
											2020-05-20 22:06:08 +00:00
										 |  |  |       await this.git(['fetch', '--unshallow']); | 
					
						
							| 
									
										
										
										
											2020-05-01 12:17:30 +00:00
										 |  |  |     } catch (error) { | 
					
						
							| 
									
										
										
										
											2020-05-21 18:51:17 +00:00
										 |  |  |       core.warning(`Fetch --unshallow caught: ${error}`); | 
					
						
							| 
									
										
										
										
											2020-05-20 22:06:08 +00:00
										 |  |  |       await this.git(['fetch']); | 
					
						
							| 
									
										
										
										
											2020-05-01 12:17:30 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Retrieves information about the branch. | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * Format: `v0.12-24-gd2198ab` | 
					
						
							|  |  |  |    * | 
					
						
							|  |  |  |    * In this format v0.12 is the latest tag, 24 are the number of commits since, and gd2198ab | 
					
						
							|  |  |  |    * identifies the current commit. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async getVersionDescription() { | 
					
						
							| 
									
										
										
										
											2020-06-15 20:35:53 +00:00
										 |  |  |     return this.git(['describe', '--long', '--tags', '--always', '--debug', this.sha]); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Returns whether there are uncommitted changes that are not ignored. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async isDirty() { | 
					
						
							| 
									
										
										
										
											2020-05-20 22:06:08 +00:00
										 |  |  |     const output = await this.git(['status', '--porcelain']); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 23:48:46 +00:00
										 |  |  |     return output !== ''; | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Get the tag if there is one pointing at HEAD | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async getTag() { | 
					
						
							| 
									
										
										
										
											2020-05-20 22:06:08 +00:00
										 |  |  |     return this.git(['tag', '--points-at', 'HEAD']); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Whether or not the repository has any version tags yet. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async hasAnyVersionTags() { | 
					
						
							| 
									
										
										
										
											2020-05-01 18:19:25 +00:00
										 |  |  |     const numberOfCommitsAsString = await System.run('sh', undefined, { | 
					
						
							|  |  |  |       input: Buffer.from('git tag --list --merged HEAD | grep v[0-9]* | wc -l'), | 
					
						
							|  |  |  |       silent: false, | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const numberOfCommits = Number.parseInt(numberOfCommitsAsString, 10); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-01 18:19:25 +00:00
										 |  |  |     return numberOfCommits !== 0; | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Get the total number of commits on head. | 
					
						
							| 
									
										
										
										
											2020-05-02 13:43:21 +00:00
										 |  |  |    * | 
					
						
							|  |  |  |    * Note: HEAD should not be used, as it may be detached, resulting in an additional count. | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |    */ | 
					
						
							|  |  |  |   static async getTotalNumberOfCommits() { | 
					
						
							| 
									
										
										
										
											2020-06-15 20:35:53 +00:00
										 |  |  |     const numberOfCommitsAsString = await this.git(['rev-list', '--count', this.sha]); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-05-01 14:52:08 +00:00
										 |  |  |     return Number.parseInt(numberOfCommitsAsString, 10); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-05-20 22:06:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** | 
					
						
							|  |  |  |    * Run git in the specified project path | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  |   static async git(arguments_, options = {}) { | 
					
						
							|  |  |  |     return System.run('git', arguments_, { cwd: this.projectPath, ...options }); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-04-26 18:22:09 +00:00
										 |  |  | } |