2019-06-20 19:28:39 +02:00
|
|
|
import * as tc from '@actions/tool-cache';
|
|
|
|
import * as path from 'path';
|
2019-08-19 14:28:37 +02:00
|
|
|
import * as semver from 'semver';
|
2020-02-09 06:29:21 +01:00
|
|
|
import * as httpm from '@actions/http-client';
|
|
|
|
import * as sys from './system';
|
2020-02-10 00:09:15 +01:00
|
|
|
import {debug} from '@actions/core';
|
2019-11-20 21:24:28 +01:00
|
|
|
|
2020-02-09 06:29:21 +01:00
|
|
|
export async function downloadGo(
|
|
|
|
versionSpec: string,
|
|
|
|
stable: boolean
|
|
|
|
): Promise<string | undefined> {
|
2020-02-09 06:21:39 +01:00
|
|
|
let toolPath: string | undefined;
|
2019-11-20 21:24:28 +01:00
|
|
|
|
2019-06-20 19:28:39 +02:00
|
|
|
try {
|
2020-02-09 06:21:39 +01:00
|
|
|
let match: IGoVersion | undefined = await findMatch(versionSpec, stable);
|
2019-06-20 19:28:39 +02:00
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
if (match) {
|
|
|
|
// download
|
2020-02-10 00:09:15 +01:00
|
|
|
debug(`match ${match.version}`);
|
2020-02-09 14:44:32 +01:00
|
|
|
let downloadUrl: string = `https://storage.googleapis.com/golang/${match.files[0].filename}`;
|
2020-02-10 04:39:44 +01:00
|
|
|
console.log(`Downloading from ${downloadUrl}`);
|
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
let downloadPath: string = await tc.downloadTool(downloadUrl);
|
2020-02-10 00:09:15 +01:00
|
|
|
debug(`downloaded to ${downloadPath}`);
|
2019-06-20 19:28:39 +02:00
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
// extract
|
2020-02-10 04:39:44 +01:00
|
|
|
console.log('Extracting ...');
|
2020-02-09 06:29:21 +01:00
|
|
|
let extPath: string =
|
|
|
|
sys.getPlatform() == 'windows'
|
|
|
|
? await tc.extractZip(downloadPath)
|
|
|
|
: await tc.extractTar(downloadPath);
|
2020-02-10 00:09:15 +01:00
|
|
|
debug(`extracted to ${extPath}`);
|
2019-06-20 19:28:39 +02:00
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
// extracts with a root folder that matches the fileName downloaded
|
|
|
|
const toolRoot = path.join(extPath, 'go');
|
|
|
|
toolPath = await tc.cacheDir(toolRoot, 'go', versionSpec);
|
|
|
|
}
|
|
|
|
} catch (error) {
|
2020-02-10 00:09:15 +01:00
|
|
|
throw new Error(`Failed to download version ${versionSpec}: ${error}`);
|
2019-06-20 19:28:39 +02:00
|
|
|
}
|
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
return toolPath;
|
2019-06-20 19:28:39 +02:00
|
|
|
}
|
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
export interface IGoVersionFile {
|
2020-02-09 06:29:21 +01:00
|
|
|
filename: string;
|
2020-02-09 06:21:39 +01:00
|
|
|
// darwin, linux, windows
|
2020-02-09 06:29:21 +01:00
|
|
|
os: string;
|
|
|
|
arch: string;
|
2019-06-20 19:28:39 +02:00
|
|
|
}
|
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
export interface IGoVersion {
|
|
|
|
version: string;
|
|
|
|
stable: boolean;
|
|
|
|
files: IGoVersionFile[];
|
2019-06-20 19:28:39 +02:00
|
|
|
}
|
|
|
|
|
2020-02-09 06:29:21 +01:00
|
|
|
export async function findMatch(
|
|
|
|
versionSpec: string,
|
|
|
|
stable: boolean
|
|
|
|
): Promise<IGoVersion | undefined> {
|
2020-02-09 06:21:39 +01:00
|
|
|
let archFilter = sys.getArch();
|
|
|
|
let platFilter = sys.getPlatform();
|
2019-06-20 19:28:39 +02:00
|
|
|
|
2020-02-10 00:09:15 +01:00
|
|
|
let result: IGoVersion | undefined;
|
2020-02-09 06:29:21 +01:00
|
|
|
let match: IGoVersion | undefined;
|
2019-06-20 19:28:39 +02:00
|
|
|
|
2020-02-09 15:25:20 +01:00
|
|
|
const dlUrl: string = 'https://golang.org/dl/?mode=json&include=all';
|
|
|
|
let candidates: IGoVersion[] | null = await module.exports.getVersions(dlUrl);
|
2020-02-09 06:21:39 +01:00
|
|
|
if (!candidates) {
|
2020-02-10 00:48:40 +01:00
|
|
|
throw new Error(`golang download url did not return results`);
|
2019-08-19 14:28:37 +02:00
|
|
|
}
|
2020-02-09 06:29:21 +01:00
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
let goFile: IGoVersionFile | undefined;
|
2020-02-09 06:29:21 +01:00
|
|
|
for (let i = 0; i < candidates.length; i++) {
|
2020-02-09 06:21:39 +01:00
|
|
|
let candidate: IGoVersion = candidates[i];
|
|
|
|
let version = candidate.version.replace('go', '');
|
2020-02-09 06:29:21 +01:00
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
// 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';
|
2019-08-19 14:28:37 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 00:09:15 +01:00
|
|
|
debug(`check ${version} satisfies ${versionSpec}`);
|
2020-02-10 21:21:04 +01:00
|
|
|
if (
|
|
|
|
semver.satisfies(version, versionSpec) &&
|
|
|
|
(!stable || candidate.stable === stable)
|
|
|
|
) {
|
2020-02-09 06:21:39 +01:00
|
|
|
goFile = candidate.files.find(file => {
|
2020-02-10 00:09:15 +01:00
|
|
|
debug(`${file.arch}===${archFilter} && ${file.os}===${platFilter}`);
|
2020-02-09 06:21:39 +01:00
|
|
|
return file.arch === archFilter && file.os === platFilter;
|
|
|
|
});
|
2019-08-19 14:28:37 +02:00
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
if (goFile) {
|
2020-02-10 00:09:15 +01:00
|
|
|
debug(`matched ${candidate.version}`);
|
2020-02-09 06:21:39 +01:00
|
|
|
match = candidate;
|
|
|
|
break;
|
|
|
|
}
|
2019-08-19 14:28:37 +02:00
|
|
|
}
|
2020-02-09 06:29:21 +01:00
|
|
|
}
|
2019-08-19 14:28:37 +02:00
|
|
|
|
2020-02-09 06:21:39 +01:00
|
|
|
if (match && goFile) {
|
2020-02-10 00:09:15 +01:00
|
|
|
// clone since we're mutating the file list to be only the file that matches
|
|
|
|
result = <IGoVersion>Object.assign({}, match);
|
|
|
|
result.files = [goFile];
|
2019-08-19 14:28:37 +02:00
|
|
|
}
|
|
|
|
|
2020-02-10 00:09:15 +01:00
|
|
|
return result;
|
2019-08-19 14:28:37 +02:00
|
|
|
}
|
2020-02-09 15:25:20 +01:00
|
|
|
|
|
|
|
export async function getVersions(dlUrl: string): Promise<IGoVersion[] | null> {
|
|
|
|
// this returns versions descending so latest is first
|
2020-02-09 20:39:34 +01:00
|
|
|
let http: httpm.HttpClient = new httpm.HttpClient('setup-go');
|
2020-02-10 00:48:40 +01:00
|
|
|
return (await http.getJson<IGoVersion[]>(dlUrl)).result;
|
2020-02-09 15:25:20 +01:00
|
|
|
}
|