2023-08-24 15:57:52 +00:00
|
|
|
import * as os from 'os'
|
|
|
|
import * as path from 'path'
|
2020-04-28 13:45:21 +00:00
|
|
|
import * as core from '@actions/core'
|
|
|
|
import * as artifact from '@actions/artifact'
|
2020-07-15 10:09:31 +00:00
|
|
|
import {Inputs, Outputs} from './constants'
|
2020-04-28 13:45:21 +00:00
|
|
|
|
2023-08-24 15:57:52 +00:00
|
|
|
const PARALLEL_DOWNLOADS = 5
|
|
|
|
|
|
|
|
export const chunk = <T>(arr: T[], n: number): T[][] =>
|
|
|
|
arr.reduce((acc, cur, i) => {
|
|
|
|
const index = Math.floor(i / n)
|
|
|
|
acc[index] = [...(acc[index] || []), cur]
|
|
|
|
return acc
|
|
|
|
}, [] as T[][])
|
|
|
|
|
2020-04-28 13:45:21 +00:00
|
|
|
async function run(): Promise<void> {
|
2023-08-24 15:57:52 +00:00
|
|
|
const inputs = {
|
|
|
|
name: core.getInput(Inputs.Name, {required: false}),
|
|
|
|
path: core.getInput(Inputs.Path, {required: false}),
|
|
|
|
token: core.getInput(Inputs.GitHubToken, {required: true}),
|
|
|
|
repository: core.getInput(Inputs.Repository, {required: true}),
|
|
|
|
runID: parseInt(core.getInput(Inputs.RunID, {required: true}))
|
|
|
|
}
|
|
|
|
|
2023-08-24 16:05:05 +00:00
|
|
|
if (!inputs.path) {
|
|
|
|
inputs.path = process.env['GITHUB_WORKSPACE'] || process.cwd()
|
|
|
|
}
|
|
|
|
|
2023-08-24 15:57:52 +00:00
|
|
|
if (inputs.path.startsWith(`~`)) {
|
|
|
|
inputs.path = inputs.path.replace('~', os.homedir())
|
|
|
|
}
|
|
|
|
|
2023-10-27 14:11:07 +00:00
|
|
|
const isSingleArtifactDownload: boolean = !!inputs.name
|
2023-08-24 15:57:52 +00:00
|
|
|
const resolvedPath = path.resolve(inputs.path)
|
|
|
|
core.debug(`Resolved path is ${resolvedPath}`)
|
|
|
|
|
|
|
|
const [owner, repo] = inputs.repository.split('/')
|
|
|
|
if (!owner || !repo) {
|
|
|
|
throw new Error(
|
|
|
|
`Invalid repository: '${inputs.repository}'. Must be in format owner/repo`
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
const artifactClient = artifact.create()
|
|
|
|
let artifacts: artifact.Artifact[] = []
|
|
|
|
|
2023-10-27 14:11:07 +00:00
|
|
|
if (isSingleArtifactDownload) {
|
|
|
|
core.info(`Downloading single artifact`)
|
|
|
|
|
2023-08-24 15:57:52 +00:00
|
|
|
const {artifact: targetArtifact} = await artifactClient.getArtifact(
|
|
|
|
inputs.name,
|
|
|
|
inputs.runID,
|
|
|
|
owner,
|
|
|
|
repo,
|
|
|
|
inputs.token
|
|
|
|
)
|
|
|
|
|
|
|
|
if (!targetArtifact) {
|
|
|
|
throw new Error(`Artifact '${inputs.name}' not found`)
|
2020-07-30 10:01:38 +00:00
|
|
|
}
|
2023-08-24 15:57:52 +00:00
|
|
|
|
|
|
|
core.debug(
|
|
|
|
`Found named artifact '${inputs.name}' (ID: ${targetArtifact.id}, Size: ${targetArtifact.size})`
|
|
|
|
)
|
|
|
|
|
|
|
|
artifacts = [targetArtifact]
|
|
|
|
} else {
|
2023-10-27 14:11:07 +00:00
|
|
|
core.info(`No input name specified, downloading all artifacts. Extra directory with the artifact name will be created for each download`)
|
|
|
|
|
2023-08-24 15:57:52 +00:00
|
|
|
const listArtifactResponse = await artifactClient.listArtifacts(
|
|
|
|
inputs.runID,
|
|
|
|
owner,
|
|
|
|
repo,
|
|
|
|
inputs.token
|
|
|
|
)
|
|
|
|
|
|
|
|
if (listArtifactResponse.artifacts.length === 0) {
|
|
|
|
throw new Error(
|
|
|
|
`No artifacts found for run '${inputs.runID}' in '${inputs.repository}'`
|
2020-04-28 13:45:21 +00:00
|
|
|
)
|
|
|
|
}
|
2023-08-24 15:57:52 +00:00
|
|
|
|
|
|
|
core.debug(`Found ${listArtifactResponse.artifacts.length} artifacts`)
|
|
|
|
artifacts = listArtifactResponse.artifacts
|
|
|
|
}
|
|
|
|
|
|
|
|
const downloadPromises = artifacts.map(artifact =>
|
|
|
|
artifactClient.downloadArtifact(artifact.id, owner, repo, inputs.token, {
|
2023-10-27 14:11:07 +00:00
|
|
|
path: isSingleArtifactDownload ? resolvedPath : path.join(resolvedPath, artifact.name)
|
2023-08-24 15:57:52 +00:00
|
|
|
})
|
|
|
|
)
|
|
|
|
|
|
|
|
const chunkedPromises = chunk(downloadPromises, PARALLEL_DOWNLOADS)
|
|
|
|
for (const chunk of chunkedPromises) {
|
|
|
|
await Promise.all(chunk)
|
2020-04-28 13:45:21 +00:00
|
|
|
}
|
2023-08-24 15:57:52 +00:00
|
|
|
|
|
|
|
core.info(`Total of ${artifacts.length} artifact(s) downloaded`)
|
|
|
|
core.setOutput(Outputs.DownloadPath, resolvedPath)
|
|
|
|
core.info('Download artifact has finished successfully')
|
2020-04-28 13:45:21 +00:00
|
|
|
}
|
|
|
|
|
2023-08-24 15:57:52 +00:00
|
|
|
run().catch(err =>
|
|
|
|
core.setFailed(`Unable to download artifact(s): ${err.message}`)
|
|
|
|
)
|