| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  | /** | 
					
						
							|  |  |  |  * @license | 
					
						
							|  |  |  |  * Copyright Google LLC All Rights Reserved. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Use of this source code is governed by an MIT-style license that can be | 
					
						
							|  |  |  |  * found in the LICENSE file at https://angular.io/license
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import * as semver from 'semver'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-09 14:42:34 +02:00
										 |  |  | import {ReleaseConfig} from '../../../release/config/index'; | 
					
						
							|  |  |  | import {computeLtsEndDateOfMajor, fetchProjectNpmPackageInfo, getLtsNpmDistTagOfMajor, getVersionOfBranch, GithubRepoWithApi} from '../../../release/versioning'; | 
					
						
							| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  | import {promptConfirm, red, warn, yellow} from '../../../utils/console'; | 
					
						
							|  |  |  | import {InvalidTargetBranchError} from '../target-label'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Asserts that the given branch corresponds to an active LTS version-branch that can receive | 
					
						
							| 
									
										
										
										
											2020-09-01 10:46:41 +02:00
										 |  |  |  * backport fixes. Throws an error if LTS expired or an invalid branch is selected. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2020-09-09 14:42:34 +02:00
										 |  |  |  * @param repo Repository containing the given branch. Used for Github API queries. | 
					
						
							|  |  |  |  * @param releaseConfig Configuration for releases. Used to query NPM about past publishes. | 
					
						
							| 
									
										
										
										
											2020-09-01 10:46:41 +02:00
										 |  |  |  * @param branchName Branch that is checked to be an active LTS version-branch. | 
					
						
							|  |  |  |  * */ | 
					
						
							|  |  |  | export async function assertActiveLtsBranch( | 
					
						
							| 
									
										
										
										
											2020-09-09 14:42:34 +02:00
										 |  |  |     repo: GithubRepoWithApi, releaseConfig: ReleaseConfig, branchName: string) { | 
					
						
							| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  |   const version = await getVersionOfBranch(repo, branchName); | 
					
						
							| 
									
										
										
										
											2020-09-09 14:42:34 +02:00
										 |  |  |   const {'dist-tags': distTags, time} = await fetchProjectNpmPackageInfo(releaseConfig); | 
					
						
							| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // LTS versions should be tagged in NPM in the following format: `v{major}-lts`.
 | 
					
						
							| 
									
										
										
										
											2020-09-01 19:48:03 +02:00
										 |  |  |   const ltsNpmTag = getLtsNpmDistTagOfMajor(version.major); | 
					
						
							|  |  |  |   const ltsVersion = semver.parse(distTags[ltsNpmTag]); | 
					
						
							| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-10 22:02:47 +03:00
										 |  |  |   // Ensure that there is an LTS version tagged for the given version-branch major. e.g.
 | 
					
						
							|  |  |  |   // if the version branch is `9.2.x` then we want to make sure that there is an LTS
 | 
					
						
							| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  |   // version tagged in NPM for `v9`, following the `v{major}-lts` tag convention.
 | 
					
						
							|  |  |  |   if (ltsVersion === null) { | 
					
						
							|  |  |  |     throw new InvalidTargetBranchError(`No LTS version tagged for v${version.major} in NPM.`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Ensure that the correct branch is used for the LTS version. We do not want to merge
 | 
					
						
							|  |  |  |   // changes to older minor version branches that do not reflect the current LTS version.
 | 
					
						
							|  |  |  |   if (branchName !== `${ltsVersion.major}.${ltsVersion.minor}.x`) { | 
					
						
							|  |  |  |     throw new InvalidTargetBranchError( | 
					
						
							|  |  |  |         `Not using last-minor branch for v${version.major} LTS version. PR ` + | 
					
						
							|  |  |  |         `should be updated to target: ${ltsVersion.major}.${ltsVersion.minor}.x`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const today = new Date(); | 
					
						
							| 
									
										
										
										
											2020-09-01 19:48:03 +02:00
										 |  |  |   const majorReleaseDate = new Date(time[`${version.major}.0.0`]); | 
					
						
							|  |  |  |   const ltsEndDate = computeLtsEndDateOfMajor(majorReleaseDate); | 
					
						
							| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Check if LTS has already expired for the targeted major version. If so, we do not
 | 
					
						
							|  |  |  |   // allow the merge as per our LTS guarantees. Can be forcibly overridden if desired.
 | 
					
						
							|  |  |  |   // See: https://angular.io/guide/releases#support-policy-and-schedule.
 | 
					
						
							|  |  |  |   if (today > ltsEndDate) { | 
					
						
							| 
									
										
										
										
											2021-06-03 20:36:38 +02:00
										 |  |  |     const ltsEndDateText = ltsEndDate.toLocaleDateString('en-US'); | 
					
						
							| 
									
										
										
										
											2020-07-24 17:47:30 +02:00
										 |  |  |     warn(red(`Long-term support ended for v${version.major} on ${ltsEndDateText}.`)); | 
					
						
							|  |  |  |     warn(yellow( | 
					
						
							|  |  |  |         `Merging of pull requests for this major is generally not ` + | 
					
						
							|  |  |  |         `desired, but can be forcibly ignored.`)); | 
					
						
							|  |  |  |     if (await promptConfirm('Do you want to forcibly proceed with merging?')) { | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     throw new InvalidTargetBranchError( | 
					
						
							|  |  |  |         `Long-term supported ended for v${version.major} on ${ltsEndDateText}. ` + | 
					
						
							|  |  |  |         `Pull request cannot be merged into the ${branchName} branch.`); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |