2019-08-13 12:11:30 +02:00
|
|
|
import * as core from '@actions/core';
|
|
|
|
import * as io from '@actions/io';
|
|
|
|
import * as tc from '@actions/tool-cache';
|
|
|
|
import * as fs from 'fs';
|
|
|
|
import * as path from 'path';
|
2019-08-13 12:46:10 +02:00
|
|
|
import uuidV4 from 'uuid/v4';
|
|
|
|
import {exec} from '@actions/exec/lib/exec';
|
2019-08-13 12:11:30 +02:00
|
|
|
|
|
|
|
const IS_WINDOWS = process.platform === 'win32';
|
|
|
|
const IS_DARWIN = process.platform === 'darwin';
|
|
|
|
const IS_LINUX = process.platform === 'linux';
|
|
|
|
|
|
|
|
let tempDirectory = process.env['RUNNER_TEMP'] || '';
|
|
|
|
|
|
|
|
if (!tempDirectory) {
|
|
|
|
let baseLocation;
|
|
|
|
|
|
|
|
if (IS_WINDOWS) {
|
|
|
|
baseLocation = process.env['USERPROFILE'] || 'C:\\';
|
|
|
|
} else {
|
|
|
|
if (process.platform === 'darwin') {
|
|
|
|
baseLocation = '/Users';
|
|
|
|
} else {
|
|
|
|
baseLocation = '/home';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tempDirectory = path.join(baseLocation, 'actions', 'temp');
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function getFlutter(
|
|
|
|
version: string,
|
|
|
|
channel: string
|
|
|
|
): Promise<void> {
|
2019-08-14 03:04:15 +02:00
|
|
|
// make semver compatible, eg: 1.7.8+hotfix.4 -> 1.7.8-hotfix.4
|
|
|
|
const semver = version.replace('+', '-');
|
|
|
|
const cleanver = `${semver}-${channel}`;
|
|
|
|
let toolPath = tc.find('Flutter', cleanver);
|
2019-08-13 12:11:30 +02:00
|
|
|
|
|
|
|
if (toolPath) {
|
|
|
|
core.debug(`Tool found in cache ${toolPath}`);
|
|
|
|
} else {
|
|
|
|
core.debug('Downloading Flutter from Google storage');
|
|
|
|
|
|
|
|
const downloadInfo = getDownloadInfo(version, channel);
|
|
|
|
const sdkFile = await tc.downloadTool(downloadInfo.url);
|
|
|
|
|
|
|
|
let tempDir: string = generateTempDir();
|
|
|
|
const sdkDir = await extractDownload(sdkFile, tempDir);
|
|
|
|
core.debug(`Flutter sdk extracted to ${sdkDir}`);
|
|
|
|
|
2019-08-14 03:04:15 +02:00
|
|
|
toolPath = await tc.cacheDir(sdkDir, 'Flutter', cleanver);
|
2019-08-13 12:11:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
core.exportVariable('FLUTTER_HOME', toolPath);
|
|
|
|
core.addPath(path.join(toolPath, 'bin'));
|
|
|
|
}
|
|
|
|
|
|
|
|
function osName(): string {
|
|
|
|
if (IS_DARWIN) return 'macos';
|
|
|
|
if (IS_WINDOWS) return 'windows';
|
|
|
|
|
|
|
|
return process.platform;
|
|
|
|
}
|
|
|
|
|
|
|
|
function extName(): string {
|
|
|
|
if (IS_LINUX) return 'tar.xz';
|
|
|
|
|
|
|
|
return 'zip';
|
|
|
|
}
|
|
|
|
|
|
|
|
function getDownloadInfo(
|
|
|
|
version: string,
|
|
|
|
channel: string
|
|
|
|
): {version: string; url: string} {
|
|
|
|
const os = osName();
|
|
|
|
const ext = extName();
|
2019-08-13 14:01:25 +02:00
|
|
|
const url = `https://storage.googleapis.com/flutter_infra/releases/${channel}/${os}/flutter_${os}_v${version}-${channel}.${ext}`;
|
2019-08-13 12:11:30 +02:00
|
|
|
|
|
|
|
return {
|
|
|
|
version,
|
|
|
|
url
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function generateTempDir(): string {
|
|
|
|
return path.join(
|
|
|
|
tempDirectory,
|
|
|
|
'temp_' + Math.floor(Math.random() * 2000000000)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
async function extractDownload(
|
|
|
|
sdkFile: string,
|
|
|
|
destDir: string
|
|
|
|
): Promise<string> {
|
|
|
|
await io.mkdirP(destDir);
|
|
|
|
|
|
|
|
const sdkPath = path.normalize(sdkFile);
|
|
|
|
const stats = fs.statSync(sdkPath);
|
|
|
|
|
|
|
|
if (stats.isFile()) {
|
|
|
|
await extractFile(sdkFile, destDir);
|
|
|
|
|
|
|
|
const sdkDir = path.join(destDir, fs.readdirSync(destDir)[0]);
|
|
|
|
|
|
|
|
return sdkDir;
|
|
|
|
} else {
|
|
|
|
throw new Error(`Flutter sdk argument ${sdkFile} is not a file`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function extractFile(file: string, destDir: string): Promise<void> {
|
|
|
|
const stats = fs.statSync(file);
|
|
|
|
|
|
|
|
if (!stats) {
|
|
|
|
throw new Error(`Failed to extract ${file} - it doesn't exist`);
|
|
|
|
} else if (stats.isDirectory()) {
|
|
|
|
throw new Error(`Failed to extract ${file} - it is a directory`);
|
|
|
|
}
|
|
|
|
|
2019-08-13 12:29:45 +02:00
|
|
|
if ('tar.xz' === extName()) {
|
2019-08-13 12:46:10 +02:00
|
|
|
await extractTarXz(file, destDir);
|
2019-08-13 12:29:45 +02:00
|
|
|
} else {
|
2019-08-13 13:52:19 +02:00
|
|
|
if (IS_DARWIN) {
|
|
|
|
await extractZipDarwin(file, destDir);
|
|
|
|
} else {
|
|
|
|
await tc.extractZip(file, destDir);
|
|
|
|
}
|
2019-08-13 12:11:30 +02:00
|
|
|
}
|
|
|
|
}
|
2019-08-13 12:46:10 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Extract a tar.xz
|
|
|
|
*
|
|
|
|
* @param file path to the tar.xz
|
|
|
|
* @param dest destination directory. Optional.
|
|
|
|
* @returns path to the destination directory
|
|
|
|
*/
|
|
|
|
export async function extractTarXz(
|
|
|
|
file: string,
|
|
|
|
dest?: string
|
|
|
|
): Promise<string> {
|
|
|
|
if (!file) {
|
|
|
|
throw new Error("parameter 'file' is required");
|
|
|
|
}
|
|
|
|
|
|
|
|
dest = dest || (await _createExtractFolder(dest));
|
|
|
|
const tarPath: string = await io.which('tar', true);
|
|
|
|
await exec(`"${tarPath}"`, ['xC', dest, '-f', file]);
|
|
|
|
|
|
|
|
return dest;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function _createExtractFolder(dest?: string): Promise<string> {
|
|
|
|
if (!dest) {
|
|
|
|
dest = path.join(tempDirectory, uuidV4());
|
|
|
|
}
|
|
|
|
|
|
|
|
await io.mkdirP(dest);
|
|
|
|
return dest;
|
|
|
|
}
|
2019-08-13 13:52:19 +02:00
|
|
|
|
|
|
|
async function extractZipDarwin(file: string, dest: string): Promise<void> {
|
|
|
|
const unzipPath = path.join(
|
|
|
|
__dirname,
|
|
|
|
'..',
|
|
|
|
'scripts',
|
|
|
|
'externals',
|
|
|
|
'unzip-darwin'
|
|
|
|
);
|
|
|
|
await exec(`"${unzipPath}"`, [file], {cwd: dest});
|
|
|
|
}
|