setup-nextflow/index.ts

176 lines
4.6 KiB
TypeScript
Raw Normal View History

2022-06-15 16:28:15 +00:00
import * as core from "@actions/core";
import * as exec from "@actions/exec";
import * as fs from "fs";
import * as github from "@actions/github";
import * as tc from "@actions/tool-cache";
import retry = require("async-retry");
import semver = require("semver");
2022-06-13 18:29:56 +00:00
const NEXTFLOW_REPO = { owner: "nextflow-io", repo: "nextflow" };
async function all_nf_releases(ok) {
const { data: releases } = await ok.rest.repos.listReleases(NEXTFLOW_REPO);
return releases;
}
async function latest_stable_release_data(ok) {
const { data: stable_release } = await ok.rest.repos.getLatestRelease(
NEXTFLOW_REPO
);
return stable_release;
}
async function release_data(version, ok) {
// Setup tag-based filtering
filter = (r) => {
return semver.satisfies(r.tag_name, version, true);
};
// Check if the user passed a 'latest*' tag, and override filtering
// accordingly
if (version.includes("latest")) {
if (version.includes("-everything")) {
// No filtering
filter = (r) => {
return true;
};
} else if (version.includes("-edge")) {
filter = (r) => {
return r.tag_name.endsWith("-edge");
};
} else {
// This is special: passing 'latest' or 'latest-stable' allows us to use
// the latest stable GitHub release direct from the API
stable_release = await latest_stable_release_data(ok);
return stable_release;
}
}
// Get all the releases
const all_releases = await all_nf_releases(ok);
matching_releases = all_releases.filter(filter);
matching_releases.sort(function (x, y) {
semver.compare(x.tag_name, y.tag_name, true);
});
return matching_releases[0];
}
function nextflow_bin_url(release, get_all) {
const release_assets = release.assets;
const all_asset = release_assets.filter((a) => {
return a.browser_download_url.endsWith("-all");
})[0];
const regular_asset = release_assets.filter((a) => {
return a.name == "nextflow";
})[0];
const dl_asset = get_all ? all_asset : regular_asset;
return dl_asset.browser_download_url;
}
2022-06-13 20:07:53 +00:00
async function install_nextflow(url, version) {
core.debug(`Downloading Nextflow from ${url}`);
const nf_dl_path = await retry(
async (bail) => {
return await tc.downloadTool(url);
},
{
onRetry: (err) => {
core.debug(`Download of ${url} failed, trying again. Error ${err}`);
},
}
);
const temp_install_dir = fs.mkdtempSync(`nxf-${version}`);
2022-06-13 20:18:22 +00:00
const nf_path = `${temp_install_dir}/nextflow`;
2022-06-13 20:07:53 +00:00
2022-06-13 20:43:31 +00:00
fs.renameSync(nf_dl_path, nf_path);
2022-06-13 20:34:37 +00:00
fs.chmodSync(nf_path, "0711");
2022-06-13 20:07:53 +00:00
return temp_install_dir;
}
2022-06-13 18:29:56 +00:00
async function run() {
2022-06-13 20:07:53 +00:00
// Read in the arguments
const token = core.getInput("token");
const version = core.getInput("version");
const get_all = core.getBooleanInput("all");
2022-06-13 20:34:37 +00:00
let resolved_version = "";
2022-06-13 18:29:56 +00:00
// Setup the API
let octokit = {};
try {
octokit = github.getOctokit(token);
} catch (e) {
core.setFailed(
`Could not authenticate to GitHub Releases API with provided token\n${e.message}`
);
}
// Get the release info for the desired release
let release = {};
try {
2022-06-13 19:00:17 +00:00
release = await release_data(version, octokit);
2022-06-13 20:34:37 +00:00
resolved_version = release.tag_name;
2022-06-13 18:40:20 +00:00
core.info(
`Input version '${version}' resolved to Nextflow ${release.name}`
);
2022-06-13 18:29:56 +00:00
} catch (e) {
core.setFailed(
`Could not retrieve Nextflow release matching ${version}.\n${e.message}`
);
}
// Get the download url for the desired release
let url = "";
try {
url = nextflow_bin_url(release, get_all);
2022-06-13 18:40:20 +00:00
core.info(`Preparing to download from ${url}`);
2022-06-13 18:29:56 +00:00
} catch (e) {
core.setFailed(`Could not parse the download URL\n${e.message}`);
}
try {
2022-06-13 20:07:53 +00:00
// Download Nextflow and add it to path
let nf_path = "";
2022-06-13 20:34:37 +00:00
nf_path = tc.find("nextflow", resolved_version);
2022-06-13 20:07:53 +00:00
if (!nf_path) {
2022-06-13 20:34:37 +00:00
core.debug(`Could not find Nextflow ${resolved_version} in cache`);
const nf_install_path = await install_nextflow(url, resolved_version);
nf_path = await tc.cacheDir(
nf_install_path,
"nextflow",
resolved_version
);
2022-06-13 20:07:53 +00:00
core.debug(`Added Nextflow to cache: ${nf_path}`);
2022-06-13 20:43:31 +00:00
fs.rmdirSync(nf_install_path, { recursive: true });
2022-06-13 20:07:53 +00:00
} else {
core.debug(`Using cached version of Nextflow: ${nf_path}`);
}
core.addPath(nf_path);
2022-06-13 18:29:56 +00:00
2022-06-13 18:40:20 +00:00
core.info(`Downloaded \`nextflow\` to ${nf_path} and added to PATH`);
2022-06-13 18:29:56 +00:00
} catch (e) {
core.setFailed(e.message);
}
2022-06-13 21:38:31 +00:00
// Run Nextflow so it downloads its dependencies
try {
const nf_exit_code = await exec.exec("nextflow", ["help"])
} catch (e) {
core.warning("Nextflow appears to have installed correctly, but an error was thrown while running it.")
}
2022-06-13 18:29:56 +00:00
}
run();