152 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
		
		
			
		
	
	
			152 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
|  | import * as tc from '@actions/tool-cache'; | ||
|  | import * as core from '@actions/core'; | ||
|  | import * as fs from 'fs'; | ||
|  | import semver from 'semver'; | ||
|  | import path from 'path'; | ||
|  | import * as httpm from '@actions/http-client'; | ||
|  | import { getToolcachePath, getVersionFromToolcachePath, isVersionSatisfies } from '../util'; | ||
|  | import { JavaDownloadRelease, JavaInstallerOptions, JavaInstallerResults } from './base-models'; | ||
|  | import { MACOS_JAVA_CONTENT_POSTFIX } from '../constants'; | ||
|  | 
 | ||
|  | export abstract class JavaBase { | ||
|  |   protected http: httpm.HttpClient; | ||
|  |   protected version: string; | ||
|  |   protected architecture: string; | ||
|  |   protected packageType: string; | ||
|  |   protected stable: boolean; | ||
|  |   protected checkLatest: boolean; | ||
|  | 
 | ||
|  |   constructor(protected distribution: string, installerOptions: JavaInstallerOptions) { | ||
|  |     this.http = new httpm.HttpClient('actions/setup-java', undefined, { | ||
|  |       allowRetries: true, | ||
|  |       maxRetries: 3 | ||
|  |     }); | ||
|  | 
 | ||
|  |     ({ version: this.version, stable: this.stable } = this.normalizeVersion( | ||
|  |       installerOptions.version | ||
|  |     )); | ||
|  |     this.architecture = installerOptions.architecture; | ||
|  |     this.packageType = installerOptions.packageType; | ||
|  |     this.checkLatest = installerOptions.checkLatest; | ||
|  |   } | ||
|  | 
 | ||
|  |   protected abstract downloadTool(javaRelease: JavaDownloadRelease): Promise<JavaInstallerResults>; | ||
|  |   protected abstract findPackageForDownload(range: string): Promise<JavaDownloadRelease>; | ||
|  | 
 | ||
|  |   public async setupJava(): Promise<JavaInstallerResults> { | ||
|  |     let foundJava = this.findInToolcache(); | ||
|  |     if (foundJava && !this.checkLatest) { | ||
|  |       core.info(`Resolved Java ${foundJava.version} from tool-cache`); | ||
|  |     } else { | ||
|  |       core.info('Trying to resolve the latest version from remote'); | ||
|  |       const javaRelease = await this.findPackageForDownload(this.version); | ||
|  |       core.info(`Resolved latest version as ${javaRelease.version}`); | ||
|  |       if (foundJava?.version === javaRelease.version) { | ||
|  |         core.info(`Resolved Java ${foundJava.version} from tool-cache`); | ||
|  |       } else { | ||
|  |         core.info('Trying to download...'); | ||
|  |         foundJava = await this.downloadTool(javaRelease); | ||
|  |         core.info(`Java ${foundJava.version} was downloaded`); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     // JDK folder may contain postfix "Contents/Home" on macOS
 | ||
|  |     const macOSPostfixPath = path.join(foundJava.path, MACOS_JAVA_CONTENT_POSTFIX); | ||
|  |     if (process.platform === 'darwin' && fs.existsSync(macOSPostfixPath)) { | ||
|  |       foundJava.path = macOSPostfixPath; | ||
|  |     } | ||
|  | 
 | ||
|  |     core.info(`Setting Java ${foundJava.version} as the default`); | ||
|  |     this.setJavaDefault(foundJava.version, foundJava.path); | ||
|  | 
 | ||
|  |     return foundJava; | ||
|  |   } | ||
|  | 
 | ||
|  |   protected get toolcacheFolderName(): string { | ||
|  |     return `Java_${this.distribution}_${this.packageType}`; | ||
|  |   } | ||
|  | 
 | ||
|  |   protected getToolcacheVersionName(version: string): string { | ||
|  |     if (!this.stable) { | ||
|  |       if (version.includes('+')) { | ||
|  |         return version.replace('+', '-ea.'); | ||
|  |       } else { | ||
|  |         return `${version}-ea`; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     // Kotlin and some Java dependencies don't work properly when Java path contains "+" sign
 | ||
|  |     // so replace "/hostedtoolcache/Java/11.0.3+4/x64" to "/hostedtoolcache/Java/11.0.3-4/x64" when saves to cache
 | ||
|  |     // related issue: https://github.com/actions/virtual-environments/issues/3014
 | ||
|  |     return version.replace('+', '-'); | ||
|  |   } | ||
|  | 
 | ||
|  |   protected findInToolcache(): JavaInstallerResults | null { | ||
|  |     // we can't use tc.find directly because firstly, we need to filter versions by stability flag
 | ||
|  |     // if *-ea is provided, take only ea versions from toolcache, otherwise - only stable versions
 | ||
|  |     const availableVersions = tc | ||
|  |       .findAllVersions(this.toolcacheFolderName, this.architecture) | ||
|  |       .map(item => { | ||
|  |         return { | ||
|  |           version: item | ||
|  |             .replace('-ea.', '+') | ||
|  |             .replace(/-ea$/, '') | ||
|  |             // Kotlin and some Java dependencies don't work properly when Java path contains "+" sign
 | ||
|  |             // so replace "/hostedtoolcache/Java/11.0.3-4/x64" to "/hostedtoolcache/Java/11.0.3+4/x64" when retrieves  to cache
 | ||
|  |             // related issue: https://github.com/actions/virtual-environments/issues/3014
 | ||
|  |             .replace('-', '+'), | ||
|  |           path: getToolcachePath(this.toolcacheFolderName, item, this.architecture) || '', | ||
|  |           stable: !item.includes('-ea') | ||
|  |         }; | ||
|  |       }) | ||
|  |       .filter(item => item.stable === this.stable); | ||
|  | 
 | ||
|  |     const satisfiedVersions = availableVersions | ||
|  |       .filter(item => isVersionSatisfies(this.version, item.version)) | ||
|  |       .filter(item => item.path) | ||
|  |       .sort((a, b) => { | ||
|  |         return -semver.compareBuild(a.version, b.version); | ||
|  |       }); | ||
|  |     if (!satisfiedVersions || satisfiedVersions.length === 0) { | ||
|  |       return null; | ||
|  |     } | ||
|  | 
 | ||
|  |     return { | ||
|  |       version: satisfiedVersions[0].version, | ||
|  |       path: satisfiedVersions[0].path | ||
|  |     }; | ||
|  |   } | ||
|  | 
 | ||
|  |   protected normalizeVersion(version: string) { | ||
|  |     let stable = true; | ||
|  | 
 | ||
|  |     if (version.endsWith('-ea')) { | ||
|  |       version = version.replace(/-ea$/, ''); | ||
|  |       stable = false; | ||
|  |     } else if (version.includes('-ea.')) { | ||
|  |       // transform '11.0.3-ea.2' -> '11.0.3+2'
 | ||
|  |       version = version.replace('-ea.', '+'); | ||
|  |       stable = false; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!semver.validRange(version)) { | ||
|  |       throw new Error( | ||
|  |         `The string '${version}' is not valid SemVer notation for a Java version. Please check README file for code snippets and more detailed information` | ||
|  |       ); | ||
|  |     } | ||
|  | 
 | ||
|  |     return { | ||
|  |       version, | ||
|  |       stable | ||
|  |     }; | ||
|  |   } | ||
|  | 
 | ||
|  |   protected setJavaDefault(version: string, toolPath: string) { | ||
|  |     core.exportVariable('JAVA_HOME', toolPath); | ||
|  |     core.addPath(path.join(toolPath, 'bin')); | ||
|  |     core.setOutput('distribution', this.distribution); | ||
|  |     core.setOutput('path', toolPath); | ||
|  |     core.setOutput('version', version); | ||
|  |   } | ||
|  | } |