2022-11-15 15:57:34 +00:00
|
|
|
import * as core from "@actions/core"
|
|
|
|
import * as tc from "@actions/tool-cache"
|
|
|
|
import retry from "async-retry"
|
|
|
|
import * as fs from "fs"
|
|
|
|
import semver from "semver"
|
2022-11-13 05:00:23 +00:00
|
|
|
|
2024-01-09 12:28:07 +00:00
|
|
|
import { NextflowRelease } from "./nextflow-release"
|
2022-11-13 05:00:23 +00:00
|
|
|
|
2024-01-26 21:52:06 +00:00
|
|
|
async function get_latest_everything_nextflow_release(
|
|
|
|
releases: AsyncGenerator<NextflowRelease>
|
|
|
|
): Promise<NextflowRelease> {
|
|
|
|
// Need to make sure we aren't in the edge case where a patch release is
|
|
|
|
// more recent chronologically than the edge release
|
|
|
|
let latest_release = {} as NextflowRelease
|
|
|
|
|
|
|
|
for await (const release of releases) {
|
|
|
|
// First iteration:
|
|
|
|
if (Object.keys(latest_release).length === 0) {
|
|
|
|
// If the most recent release is an edge release, then we have nothing to
|
|
|
|
// worry about, return it.
|
|
|
|
if (release.isEdge) {
|
|
|
|
return release
|
|
|
|
}
|
|
|
|
// Ok, so the most recent release is a stable release. We need to keep
|
|
|
|
// tabs on it
|
|
|
|
latest_release = release
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// A larger version number that is older than the "latest" release
|
|
|
|
// indicates that we've hit an edge release that is more up-to-date
|
|
|
|
// than a patch release. Return it.
|
|
|
|
if (semver.gt(release.versionNumber, latest_release.versionNumber, true)) {
|
|
|
|
return release
|
|
|
|
}
|
|
|
|
|
|
|
|
// A smaller version number that is also an edge release indicates that the
|
|
|
|
// chronologically most recent release is also the most up-to-date
|
|
|
|
if (
|
|
|
|
release.isEdge &&
|
|
|
|
semver.lt(release.versionNumber, latest_release.versionNumber, true)
|
|
|
|
) {
|
|
|
|
return latest_release
|
|
|
|
}
|
|
|
|
|
|
|
|
// Once we've hit the major.minor.0 of the version that is the most recent
|
|
|
|
// patch, we know that we would have traversed any edge releases along the
|
|
|
|
// way, so check to see if we've hit that point yet.
|
|
|
|
const latest_release_major = semver.major(
|
|
|
|
latest_release.versionNumber,
|
|
|
|
true
|
|
|
|
)
|
|
|
|
const latest_release_minor = semver.minor(
|
|
|
|
latest_release.versionNumber,
|
|
|
|
true
|
|
|
|
)
|
|
|
|
const latest_release_minver = `${latest_release_major}.${latest_release_minor}.0`
|
|
|
|
if (semver.eq(release.versionNumber, latest_release_minver, true)) {
|
|
|
|
return latest_release
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should never get here, but just in case
|
|
|
|
return {} as NextflowRelease
|
|
|
|
}
|
|
|
|
|
|
|
|
async function get_latest_edge_nextflow_release(
|
|
|
|
releases: AsyncGenerator<NextflowRelease>
|
|
|
|
): Promise<NextflowRelease> {
|
|
|
|
// Because we don't have to worry about crossing between edge and stable
|
|
|
|
// releases, we can just return the first edge release we come across
|
|
|
|
for await (const release of releases) {
|
|
|
|
if (release.isEdge) {
|
|
|
|
return release
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should never get here, but just in case
|
|
|
|
return {} as NextflowRelease
|
|
|
|
}
|
|
|
|
|
|
|
|
async function get_latest_stable_nextflow_release(
|
|
|
|
releases: AsyncGenerator<NextflowRelease>
|
|
|
|
): Promise<NextflowRelease> {
|
|
|
|
// Because we don't have to worry about crossing between edge and stable
|
|
|
|
// releases, we can just return the first stable release we come across
|
|
|
|
for await (const release of releases) {
|
|
|
|
if (!release.isEdge) {
|
|
|
|
return release
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should never get here, but just in case
|
|
|
|
return {} as NextflowRelease
|
|
|
|
}
|
|
|
|
|
2023-12-23 17:42:54 +00:00
|
|
|
export async function get_nextflow_release(
|
|
|
|
version: string,
|
2024-01-26 21:52:06 +00:00
|
|
|
releases: AsyncGenerator<NextflowRelease>
|
2023-12-23 17:42:54 +00:00
|
|
|
): Promise<NextflowRelease> {
|
2024-01-26 21:52:06 +00:00
|
|
|
// First, check to see if we are using a "latest-*" version system, and return
|
|
|
|
// early
|
|
|
|
if (version === "latest-everything") {
|
|
|
|
return await get_latest_everything_nextflow_release(releases)
|
|
|
|
}
|
|
|
|
if (version === "latest-edge") {
|
|
|
|
return await get_latest_edge_nextflow_release(releases)
|
|
|
|
}
|
|
|
|
if (version === "latest" || version === "latest-stable") {
|
|
|
|
return await get_latest_stable_nextflow_release(releases)
|
|
|
|
}
|
2022-11-13 05:00:23 +00:00
|
|
|
|
2024-01-26 21:52:06 +00:00
|
|
|
// The releases are sent in reverse chronological order
|
|
|
|
// If we are sent a numbered tag, then back through the list until we find
|
|
|
|
// a release that fulfils the requested version number
|
|
|
|
for await (const release of releases) {
|
|
|
|
if (semver.satisfies(release.versionNumber, version, true)) {
|
|
|
|
return release
|
|
|
|
}
|
|
|
|
}
|
2022-11-13 05:00:23 +00:00
|
|
|
|
2024-01-26 21:52:06 +00:00
|
|
|
// We should never get here, but just in case
|
|
|
|
return {} as NextflowRelease
|
2022-11-13 05:00:23 +00:00
|
|
|
}
|
|
|
|
|
2022-11-13 21:56:49 +00:00
|
|
|
export async function install_nextflow(
|
2023-12-23 17:43:41 +00:00
|
|
|
release: NextflowRelease,
|
|
|
|
get_all: boolean
|
2022-11-13 21:56:49 +00:00
|
|
|
): Promise<string> {
|
2023-12-23 17:43:41 +00:00
|
|
|
const url = get_all ? release.allBinaryURL : release.binaryURL
|
|
|
|
const version = release.versionNumber
|
|
|
|
|
2022-11-13 05:00:23 +00:00
|
|
|
core.debug(`Downloading Nextflow from ${url}`)
|
|
|
|
const nf_dl_path = await retry(
|
2022-11-13 21:56:49 +00:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
2022-11-13 05:00:23 +00:00
|
|
|
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}`)
|
|
|
|
const nf_path = `${temp_install_dir}/nextflow`
|
|
|
|
|
2023-06-22 07:18:37 +00:00
|
|
|
try {
|
|
|
|
fs.renameSync(nf_dl_path, nf_path)
|
|
|
|
} catch (err: unknown) {
|
|
|
|
core.debug(`Failed to rename file: ${err}`)
|
|
|
|
fs.copyFileSync(nf_dl_path, nf_path)
|
|
|
|
fs.unlinkSync(nf_dl_path)
|
|
|
|
}
|
2022-11-15 15:57:34 +00:00
|
|
|
fs.chmodSync(nf_path, "0711")
|
2022-11-13 05:00:23 +00:00
|
|
|
|
|
|
|
return temp_install_dir
|
|
|
|
}
|
2023-05-16 23:13:04 +00:00
|
|
|
|
|
|
|
export function check_cache(version: string): boolean {
|
2024-01-06 16:08:13 +00:00
|
|
|
// A 'latest*' version indicates that a cached version would be invalid until
|
|
|
|
// the version is resolved: abort
|
|
|
|
if (version.includes("latest")) {
|
|
|
|
return false
|
|
|
|
}
|
2023-05-16 23:13:04 +00:00
|
|
|
const cleaned_version = semver.clean(version, true)
|
|
|
|
if (cleaned_version === null) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
const resolved_version = String(cleaned_version)
|
|
|
|
|
|
|
|
const nf_path = tc.find("nextflow", resolved_version)
|
|
|
|
if (!nf_path) {
|
|
|
|
core.debug(`Could not find Nextflow ${resolved_version} in the tool cache`)
|
|
|
|
return false
|
|
|
|
} else {
|
|
|
|
core.debug(`Found Nextflow ${resolved_version} at path '${nf_path}'`)
|
|
|
|
core.debug(`Adding '${nf_path}' to PATH`)
|
|
|
|
core.addPath(nf_path)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|