import * as tc from '@actions/tool-cache'; import * as path from 'path'; import * as semver from 'semver'; import * as httpm from '@actions/http-client'; import * as sys from './system'; import {debug} from '@actions/core'; export async function downloadGo( versionSpec: string, stable: boolean ): Promise { let toolPath: string | undefined; try { let match: IGoVersion | undefined = await findMatch(versionSpec, stable); if (match) { // download debug(`match ${match.version}`); let downloadUrl: string = `https://storage.googleapis.com/golang/${match.files[0].filename}`; console.log(`Downloading from ${downloadUrl}`); let downloadPath: string = await tc.downloadTool(downloadUrl); debug(`downloaded to ${downloadPath}`); // extract console.log('Extracting ...'); let extPath: string = sys.getPlatform() == 'windows' ? await tc.extractZip(downloadPath) : await tc.extractTar(downloadPath); debug(`extracted to ${extPath}`); // extracts with a root folder that matches the fileName downloaded const toolRoot = path.join(extPath, 'go'); toolPath = await tc.cacheDir(toolRoot, 'go', makeSemver(match.version)); } } catch (error) { throw new Error(`Failed to download version ${versionSpec}: ${error}`); } return toolPath; } export interface IGoVersionFile { filename: string; // darwin, linux, windows os: string; arch: string; } export interface IGoVersion { version: string; stable: boolean; files: IGoVersionFile[]; } export async function findMatch( versionSpec: string, stable: boolean ): Promise { let archFilter = sys.getArch(); let platFilter = sys.getPlatform(); let result: IGoVersion | undefined; let match: IGoVersion | undefined; const dlUrl: string = 'https://golang.org/dl/?mode=json&include=all'; let candidates: IGoVersion[] | null = await module.exports.getVersions(dlUrl); if (!candidates) { throw new Error(`golang download url did not return results`); } let goFile: IGoVersionFile | undefined; for (let i = 0; i < candidates.length; i++) { let candidate: IGoVersion = candidates[i]; let version = makeSemver(candidate.version); // 1.13.0 is advertised as 1.13 preventing being able to match exactly 1.13.0 // since a semver of 1.13 would match latest 1.13 let parts: string[] = version.split('.'); if (parts.length == 2) { version = version + '.0'; } debug(`check ${version} satisfies ${versionSpec}`); if ( semver.satisfies(version, versionSpec) && (!stable || candidate.stable === stable) ) { goFile = candidate.files.find(file => { debug(`${file.arch}===${archFilter} && ${file.os}===${platFilter}`); return file.arch === archFilter && file.os === platFilter; }); if (goFile) { debug(`matched ${candidate.version}`); match = candidate; break; } } } if (match && goFile) { // clone since we're mutating the file list to be only the file that matches result = Object.assign({}, match); result.files = [goFile]; } return result; } export async function getVersions(dlUrl: string): Promise { // this returns versions descending so latest is first let http: httpm.HttpClient = new httpm.HttpClient('setup-go'); return (await http.getJson(dlUrl)).result; } // // Convert the go version syntax into semver for semver matching // 1.13.1 => 1.13.1 // 1.13 => 1.13.0 // 1.10beta1 => 1.10.0-beta1, 1.10rc1 => 1.10.0-rc1 // 1.8.5beta1 => 1.8.5-beta1, 1.8.5rc1 => 1.8.5-rc1 export function makeSemver(version: string): string { version = version.replace('go', ''); version = version.replace('beta', '-beta').replace('rc', '-rc'); let parts = version.split('-'); let verPart: string = parts[0]; let prereleasePart = parts.length > 1 ? `-${parts[1]}` : ''; let verParts: string[] = verPart.split('.'); if (verParts.length == 2) { verPart += '.0'; } return `${verPart}${prereleasePart}`; }