mirror of
https://github.com/actions/download-artifact.git
synced 2025-04-05 02:50:12 +08:00
Compare commits
No commits in common. "main" and "v4.1.5" have entirely different histories.
20
.github/workflows/publish-immutable-actions.yml
vendored
20
.github/workflows/publish-immutable-actions.yml
vendored
@ -1,20 +0,0 @@
|
|||||||
name: 'Publish Immutable Action Version'
|
|
||||||
|
|
||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
publish:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
id-token: write
|
|
||||||
packages: write
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checking out
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
- name: Publish
|
|
||||||
id: publish
|
|
||||||
uses: actions/publish-immutable-action@0.0.3
|
|
3
.github/workflows/test.yml
vendored
3
.github/workflows/test.yml
vendored
@ -40,9 +40,6 @@ jobs:
|
|||||||
- name: Format
|
- name: Format
|
||||||
run: npm run format-check
|
run: npm run format-check
|
||||||
|
|
||||||
- name: Run Unit Tests
|
|
||||||
run: npm test
|
|
||||||
|
|
||||||
- name: Create artifacts
|
- name: Create artifacts
|
||||||
run: |
|
run: |
|
||||||
mkdir -p path/to/artifact-A
|
mkdir -p path/to/artifact-A
|
||||||
|
2
.licenses/npm/@actions/artifact.dep.yml
generated
2
.licenses/npm/@actions/artifact.dep.yml
generated
@ -1,6 +1,6 @@
|
|||||||
---
|
---
|
||||||
name: "@actions/artifact"
|
name: "@actions/artifact"
|
||||||
version: 2.3.2
|
version: 2.1.5
|
||||||
type: npm
|
type: npm
|
||||||
summary: Actions artifact lib
|
summary: Actions artifact lib
|
||||||
homepage: https://github.com/actions/toolkit/tree/main/packages/artifact
|
homepage: https://github.com/actions/toolkit/tree/main/packages/artifact
|
||||||
|
@ -1,5 +1,11 @@
|
|||||||
# `@actions/download-artifact`
|
# `@actions/download-artifact`
|
||||||
|
|
||||||
|
> [!WARNING]
|
||||||
|
> actions/download-artifact@v3 is scheduled for deprecation on **November 30, 2024**. [Learn more.](https://github.blog/changelog/2024-04-16-deprecation-notice-v3-of-the-artifact-actions/)
|
||||||
|
> Similarly, v1/v2 are scheduled for deprecation on **June 30, 2024**.
|
||||||
|
> Please update your workflow to use v4 of the artifact actions.
|
||||||
|
> This deprecation will not impact any existing versions of GitHub Enterprise Server being used by customers.
|
||||||
|
|
||||||
Download [Actions Artifacts](https://docs.github.com/en/actions/using-workflows/storing-workflow-data-as-artifacts) from your Workflow Runs. Internally powered by the [@actions/artifact](https://github.com/actions/toolkit/tree/main/packages/artifact) package.
|
Download [Actions Artifacts](https://docs.github.com/en/actions/using-workflows/storing-workflow-data-as-artifacts) from your Workflow Runs. Internally powered by the [@actions/artifact](https://github.com/actions/toolkit/tree/main/packages/artifact) package.
|
||||||
|
|
||||||
See also [upload-artifact](https://github.com/actions/upload-artifact).
|
See also [upload-artifact](https://github.com/actions/upload-artifact).
|
||||||
|
@ -1,224 +0,0 @@
|
|||||||
import * as core from '@actions/core'
|
|
||||||
import artifact, {ArtifactNotFoundError} from '@actions/artifact'
|
|
||||||
import {run} from '../src/download-artifact'
|
|
||||||
import {Inputs} from '../src/constants'
|
|
||||||
|
|
||||||
jest.mock('@actions/github', () => ({
|
|
||||||
context: {
|
|
||||||
repo: {
|
|
||||||
owner: 'actions',
|
|
||||||
repo: 'toolkit'
|
|
||||||
},
|
|
||||||
runId: 123,
|
|
||||||
serverUrl: 'https://github.com'
|
|
||||||
}
|
|
||||||
}))
|
|
||||||
|
|
||||||
jest.mock('@actions/core')
|
|
||||||
|
|
||||||
/* eslint-disable no-unused-vars */ /* eslint-disable @typescript-eslint/no-explicit-any */
|
|
||||||
const mockInputs = (overrides?: Partial<{[K in Inputs]?: any}>) => {
|
|
||||||
const inputs = {
|
|
||||||
[Inputs.Name]: 'artifact-name',
|
|
||||||
[Inputs.Path]: '/some/artifact/path',
|
|
||||||
[Inputs.GitHubToken]: 'warn',
|
|
||||||
[Inputs.Repository]: 'owner/some-repository',
|
|
||||||
[Inputs.RunID]: 'some-run-id',
|
|
||||||
[Inputs.Pattern]: 'some-pattern',
|
|
||||||
...overrides
|
|
||||||
}
|
|
||||||
|
|
||||||
;(core.getInput as jest.Mock).mockImplementation((name: string) => {
|
|
||||||
return inputs[name]
|
|
||||||
})
|
|
||||||
;(core.getBooleanInput as jest.Mock).mockImplementation((name: string) => {
|
|
||||||
return inputs[name]
|
|
||||||
})
|
|
||||||
|
|
||||||
return inputs
|
|
||||||
}
|
|
||||||
|
|
||||||
describe('download', () => {
|
|
||||||
beforeEach(async () => {
|
|
||||||
mockInputs()
|
|
||||||
jest.clearAllMocks()
|
|
||||||
|
|
||||||
// Mock artifact client methods
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'listArtifacts')
|
|
||||||
.mockImplementation(() => Promise.resolve({artifacts: []}))
|
|
||||||
jest.spyOn(artifact, 'getArtifact').mockImplementation(name => {
|
|
||||||
throw new ArtifactNotFoundError(`Artifact '${name}' not found`)
|
|
||||||
})
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'downloadArtifact')
|
|
||||||
.mockImplementation(() => Promise.resolve({digestMismatch: false}))
|
|
||||||
})
|
|
||||||
|
|
||||||
test('downloads a single artifact by name', async () => {
|
|
||||||
const mockArtifact = {
|
|
||||||
id: 123,
|
|
||||||
name: 'artifact-name',
|
|
||||||
size: 1024,
|
|
||||||
digest: 'abc123'
|
|
||||||
}
|
|
||||||
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'getArtifact')
|
|
||||||
.mockImplementation(() => Promise.resolve({artifact: mockArtifact}))
|
|
||||||
|
|
||||||
await run()
|
|
||||||
|
|
||||||
expect(artifact.downloadArtifact).toHaveBeenCalledWith(
|
|
||||||
mockArtifact.id,
|
|
||||||
expect.objectContaining({
|
|
||||||
expectedHash: mockArtifact.digest
|
|
||||||
})
|
|
||||||
)
|
|
||||||
expect(core.info).toHaveBeenCalledWith('Total of 1 artifact(s) downloaded')
|
|
||||||
|
|
||||||
expect(core.setOutput).toHaveBeenCalledWith(
|
|
||||||
'download-path',
|
|
||||||
expect.any(String)
|
|
||||||
)
|
|
||||||
|
|
||||||
expect(core.info).toHaveBeenCalledWith(
|
|
||||||
'Download artifact has finished successfully'
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('downloads multiple artifacts when no name or pattern provided', async () => {
|
|
||||||
jest.clearAllMocks()
|
|
||||||
mockInputs({
|
|
||||||
[Inputs.Name]: '',
|
|
||||||
[Inputs.Pattern]: ''
|
|
||||||
})
|
|
||||||
|
|
||||||
const mockArtifacts = [
|
|
||||||
{id: 123, name: 'artifact1', size: 1024, digest: 'abc123'},
|
|
||||||
{id: 456, name: 'artifact2', size: 2048, digest: 'def456'}
|
|
||||||
]
|
|
||||||
|
|
||||||
// Set up artifact mock after clearing mocks
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'listArtifacts')
|
|
||||||
.mockImplementation(() => Promise.resolve({artifacts: mockArtifacts}))
|
|
||||||
|
|
||||||
// Reset downloadArtifact mock as well
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'downloadArtifact')
|
|
||||||
.mockImplementation(() => Promise.resolve({digestMismatch: false}))
|
|
||||||
|
|
||||||
await run()
|
|
||||||
|
|
||||||
expect(core.info).toHaveBeenCalledWith(
|
|
||||||
'No input name or pattern filtered specified, downloading all artifacts'
|
|
||||||
)
|
|
||||||
|
|
||||||
expect(core.info).toHaveBeenCalledWith('Total of 2 artifact(s) downloaded')
|
|
||||||
expect(artifact.downloadArtifact).toHaveBeenCalledTimes(2)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('sets download path output even when no artifacts are found', async () => {
|
|
||||||
mockInputs({[Inputs.Name]: ''})
|
|
||||||
|
|
||||||
await run()
|
|
||||||
|
|
||||||
expect(core.setOutput).toHaveBeenCalledWith(
|
|
||||||
'download-path',
|
|
||||||
expect.any(String)
|
|
||||||
)
|
|
||||||
|
|
||||||
expect(core.info).toHaveBeenCalledWith(
|
|
||||||
'Download artifact has finished successfully'
|
|
||||||
)
|
|
||||||
|
|
||||||
expect(core.info).toHaveBeenCalledWith('Total of 0 artifact(s) downloaded')
|
|
||||||
})
|
|
||||||
|
|
||||||
test('filters artifacts by pattern', async () => {
|
|
||||||
const mockArtifacts = [
|
|
||||||
{id: 123, name: 'test-artifact', size: 1024, digest: 'abc123'},
|
|
||||||
{id: 456, name: 'prod-artifact', size: 2048, digest: 'def456'}
|
|
||||||
]
|
|
||||||
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'listArtifacts')
|
|
||||||
.mockImplementation(() => Promise.resolve({artifacts: mockArtifacts}))
|
|
||||||
|
|
||||||
mockInputs({
|
|
||||||
[Inputs.Name]: '',
|
|
||||||
[Inputs.Pattern]: 'test-*'
|
|
||||||
})
|
|
||||||
|
|
||||||
await run()
|
|
||||||
|
|
||||||
expect(artifact.downloadArtifact).toHaveBeenCalledTimes(1)
|
|
||||||
expect(artifact.downloadArtifact).toHaveBeenCalledWith(
|
|
||||||
123,
|
|
||||||
expect.anything()
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('uses token and repository information when provided', async () => {
|
|
||||||
const token = 'ghp_testtoken123'
|
|
||||||
|
|
||||||
mockInputs({
|
|
||||||
[Inputs.Name]: '',
|
|
||||||
[Inputs.GitHubToken]: token,
|
|
||||||
[Inputs.Repository]: 'myorg/myrepo',
|
|
||||||
[Inputs.RunID]: '789'
|
|
||||||
})
|
|
||||||
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'listArtifacts')
|
|
||||||
.mockImplementation(() => Promise.resolve({artifacts: []}))
|
|
||||||
|
|
||||||
await run()
|
|
||||||
|
|
||||||
expect(artifact.listArtifacts).toHaveBeenCalledWith(
|
|
||||||
expect.objectContaining({
|
|
||||||
findBy: {
|
|
||||||
token,
|
|
||||||
workflowRunId: 789,
|
|
||||||
repositoryName: 'myrepo',
|
|
||||||
repositoryOwner: 'myorg'
|
|
||||||
}
|
|
||||||
})
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('throws error when repository format is invalid', async () => {
|
|
||||||
mockInputs({
|
|
||||||
[Inputs.GitHubToken]: 'some-token',
|
|
||||||
[Inputs.Repository]: 'invalid-format' // Missing the owner/repo format
|
|
||||||
})
|
|
||||||
|
|
||||||
await expect(run()).rejects.toThrow(
|
|
||||||
"Invalid repository: 'invalid-format'. Must be in format owner/repo"
|
|
||||||
)
|
|
||||||
})
|
|
||||||
|
|
||||||
test('warns when digest validation fails', async () => {
|
|
||||||
const mockArtifact = {
|
|
||||||
id: 123,
|
|
||||||
name: 'corrupted-artifact',
|
|
||||||
size: 1024,
|
|
||||||
digest: 'abc123'
|
|
||||||
}
|
|
||||||
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'getArtifact')
|
|
||||||
.mockImplementation(() => Promise.resolve({artifact: mockArtifact}))
|
|
||||||
|
|
||||||
jest
|
|
||||||
.spyOn(artifact, 'downloadArtifact')
|
|
||||||
.mockImplementation(() => Promise.resolve({digestMismatch: true}))
|
|
||||||
|
|
||||||
await run()
|
|
||||||
|
|
||||||
expect(core.warning).toHaveBeenCalledWith(
|
|
||||||
expect.stringContaining('digest validation failed')
|
|
||||||
)
|
|
||||||
})
|
|
||||||
})
|
|
3061
dist/index.js
vendored
3061
dist/index.js
vendored
File diff suppressed because it is too large
Load Diff
@ -189,8 +189,7 @@ jobs:
|
|||||||
- name: Create a File
|
- name: Create a File
|
||||||
run: echo "hello from ${{ matrix.runs-on }}" > file-${{ matrix.runs-on }}.txt
|
run: echo "hello from ${{ matrix.runs-on }}" > file-${{ matrix.runs-on }}.txt
|
||||||
- name: Upload Artifact
|
- name: Upload Artifact
|
||||||
- uses: actions/upload-artifact@v3
|
uses: actions/upload-artifact@v3
|
||||||
+ uses: actions/upload-artifact@v4
|
|
||||||
with:
|
with:
|
||||||
- name: all-my-files
|
- name: all-my-files
|
||||||
+ name: my-artifact-${{ matrix.runs-on }}
|
+ name: my-artifact-${{ matrix.runs-on }}
|
||||||
@ -206,4 +205,4 @@ jobs:
|
|||||||
+ pattern: my-artifact-*
|
+ pattern: my-artifact-*
|
||||||
```
|
```
|
||||||
|
|
||||||
Note that this will download all artifacts to a temporary directory and reupload them as a single artifact. For more information on inputs and other use cases for `actions/upload-artifact/merge@v4`, see [the action documentation](https://github.com/actions/upload-artifact/blob/main/merge/README.md).
|
Note that this will download all artifacts to a temporary directory and reupload them as a single artifact. For more information on inputs and other use cases for `actions/upload-artifact/merge@v4`, see [the action documentation](../merge/README.md).
|
||||||
|
@ -1,12 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
clearMocks: true,
|
|
||||||
moduleFileExtensions: ['js', 'ts'],
|
|
||||||
roots: ['<rootDir>'],
|
|
||||||
testEnvironment: 'node',
|
|
||||||
testMatch: ['**/*.test.ts'],
|
|
||||||
testRunner: 'jest-circus/runner',
|
|
||||||
transform: {
|
|
||||||
'^.+\\.ts$': 'ts-jest'
|
|
||||||
},
|
|
||||||
verbose: true
|
|
||||||
}
|
|
8705
package-lock.json
generated
8705
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
11
package.json
11
package.json
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "download-artifact",
|
"name": "download-artifact",
|
||||||
"version": "4.2.0",
|
"version": "4.1.5",
|
||||||
"description": "Download an Actions Artifact from a workflow run",
|
"description": "Download an Actions Artifact from a workflow run",
|
||||||
"main": "dist/index.js",
|
"main": "dist/index.js",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@ -9,8 +9,7 @@
|
|||||||
"check-all": "concurrently \"npm:format-check\" \"npm:lint\" \"npm:build\"",
|
"check-all": "concurrently \"npm:format-check\" \"npm:lint\" \"npm:build\"",
|
||||||
"format": "prettier --write **/*.ts",
|
"format": "prettier --write **/*.ts",
|
||||||
"format-check": "prettier --check **/*.ts",
|
"format-check": "prettier --check **/*.ts",
|
||||||
"lint": "eslint **/*.ts",
|
"lint": "eslint **/*.ts"
|
||||||
"test": "jest"
|
|
||||||
},
|
},
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
@ -29,13 +28,12 @@
|
|||||||
},
|
},
|
||||||
"homepage": "https://github.com/actions/download-artifact#readme",
|
"homepage": "https://github.com/actions/download-artifact#readme",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@actions/artifact": "^2.3.2",
|
"@actions/artifact": "^2.1.5",
|
||||||
"@actions/core": "^1.10.1",
|
"@actions/core": "^1.10.1",
|
||||||
"@actions/github": "^5.1.1",
|
"@actions/github": "^5.1.1",
|
||||||
"minimatch": "^9.0.3"
|
"minimatch": "^9.0.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/jest": "^29.5.14",
|
|
||||||
"@types/node": "^12.12.6",
|
"@types/node": "^12.12.6",
|
||||||
"@typescript-eslint/eslint-plugin": "^6.14.0",
|
"@typescript-eslint/eslint-plugin": "^6.14.0",
|
||||||
"@vercel/ncc": "^0.33.4",
|
"@vercel/ncc": "^0.33.4",
|
||||||
@ -43,10 +41,7 @@
|
|||||||
"eslint": "^8.55.0",
|
"eslint": "^8.55.0",
|
||||||
"eslint-plugin-github": "^4.10.1",
|
"eslint-plugin-github": "^4.10.1",
|
||||||
"eslint-plugin-prettier": "^5.0.1",
|
"eslint-plugin-prettier": "^5.0.1",
|
||||||
"jest": "^29.7.0",
|
|
||||||
"prettier": "^3.1.1",
|
"prettier": "^3.1.1",
|
||||||
"ts-jest": "^29.2.6",
|
|
||||||
"ts-node": "^10.9.2",
|
|
||||||
"typescript": "^5.3.3"
|
"typescript": "^5.3.3"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,7 @@ export const chunk = <T>(arr: T[], n: number): T[][] =>
|
|||||||
return acc
|
return acc
|
||||||
}, [] as T[][])
|
}, [] as T[][])
|
||||||
|
|
||||||
export async function run(): Promise<void> {
|
async function run(): Promise<void> {
|
||||||
const inputs = {
|
const inputs = {
|
||||||
name: core.getInput(Inputs.Name, {required: false}),
|
name: core.getInput(Inputs.Name, {required: false}),
|
||||||
path: core.getInput(Inputs.Path, {required: false}),
|
path: core.getInput(Inputs.Path, {required: false}),
|
||||||
@ -106,39 +106,26 @@ export async function run(): Promise<void> {
|
|||||||
core.info(`Preparing to download the following artifacts:`)
|
core.info(`Preparing to download the following artifacts:`)
|
||||||
artifacts.forEach(artifact => {
|
artifacts.forEach(artifact => {
|
||||||
core.info(
|
core.info(
|
||||||
`- ${artifact.name} (ID: ${artifact.id}, Size: ${artifact.size}, Expected Digest: ${artifact.digest})`
|
`- ${artifact.name} (ID: ${artifact.id}, Size: ${artifact.size})`
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
const downloadPromises = artifacts.map(artifact => ({
|
const downloadPromises = artifacts.map(artifact =>
|
||||||
name: artifact.name,
|
artifactClient.downloadArtifact(artifact.id, {
|
||||||
promise: artifactClient.downloadArtifact(artifact.id, {
|
|
||||||
...options,
|
...options,
|
||||||
path:
|
path:
|
||||||
isSingleArtifactDownload || inputs.mergeMultiple
|
isSingleArtifactDownload || inputs.mergeMultiple
|
||||||
? resolvedPath
|
? resolvedPath
|
||||||
: path.join(resolvedPath, artifact.name),
|
: path.join(resolvedPath, artifact.name)
|
||||||
expectedHash: artifact.digest
|
|
||||||
})
|
})
|
||||||
}))
|
)
|
||||||
|
|
||||||
const chunkedPromises = chunk(downloadPromises, PARALLEL_DOWNLOADS)
|
const chunkedPromises = chunk(downloadPromises, PARALLEL_DOWNLOADS)
|
||||||
for (const chunk of chunkedPromises) {
|
for (const chunk of chunkedPromises) {
|
||||||
const chunkPromises = chunk.map(item => item.promise)
|
await Promise.all(chunk)
|
||||||
const results = await Promise.all(chunkPromises)
|
|
||||||
|
|
||||||
for (let i = 0; i < results.length; i++) {
|
|
||||||
const outcome = results[i]
|
|
||||||
const artifactName = chunk[i].name
|
|
||||||
|
|
||||||
if (outcome.digestMismatch) {
|
|
||||||
core.warning(
|
|
||||||
`Artifact '${artifactName}' digest validation failed. Please verify the integrity of the artifact.`
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
core.info(`Total of ${artifacts.length} artifact(s) downloaded`)
|
core.info(`Total of ${artifacts.length} artifact(s) downloaded`)
|
||||||
core.setOutput(Outputs.DownloadPath, resolvedPath)
|
core.setOutput(Outputs.DownloadPath, resolvedPath)
|
||||||
core.info('Download artifact has finished successfully')
|
core.info('Download artifact has finished successfully')
|
||||||
|
@ -9,5 +9,5 @@
|
|||||||
"moduleResolution": "node",
|
"moduleResolution": "node",
|
||||||
"esModuleInterop": true
|
"esModuleInterop": true
|
||||||
},
|
},
|
||||||
"exclude": ["node_modules", "**/*.test.ts", "jest.config.ts", "__tests__"]
|
"exclude": ["node_modules", "**/*.test.ts"]
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user