mirror of
				https://git.mirrors.martin98.com/https://github.com/actions/upload-artifact
				synced 2025-10-31 08:51:09 +08:00 
			
		
		
		
	Merge branch 'main' into robherley/migration-docs-typo
This commit is contained in:
		
						commit
						9d63e3f2f8
					
				
							
								
								
									
										91
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										91
									
								
								.github/workflows/test.yml
									
									
									
									
										vendored
									
									
								
							| @ -141,12 +141,16 @@ jobs: | |||||||
|         } |         } | ||||||
|       shell: pwsh |       shell: pwsh | ||||||
| 
 | 
 | ||||||
|  |     - name: 'Alter file 1 content' | ||||||
|  |       run: | | ||||||
|  |         echo "This file has changed" > path/to/dir-1/file1.txt | ||||||
|  | 
 | ||||||
|     # Replace the contents of Artifact #1 |     # Replace the contents of Artifact #1 | ||||||
|     - name: 'Overwrite artifact #1 again' |     - name: 'Overwrite artifact #1' | ||||||
|       uses: ./ |       uses: ./ | ||||||
|       with: |       with: | ||||||
|         name: 'Artifact-A-${{ matrix.runs-on }}' |         name: 'Artifact-A-${{ matrix.runs-on }}' | ||||||
|         path: path/to/dir-2/file2.txt |         path: path/to/dir-1/file1.txt | ||||||
|         overwrite: true |         overwrite: true | ||||||
| 
 | 
 | ||||||
|     # Download replaced Artifact #1 and verify the correctness of the content |     # Download replaced Artifact #1 and verify the correctness of the content | ||||||
| @ -158,13 +162,90 @@ jobs: | |||||||
| 
 | 
 | ||||||
|     - name: 'Verify Artifact #1 again' |     - name: 'Verify Artifact #1 again' | ||||||
|       run: | |       run: | | ||||||
|         $file = "overwrite/some/new/path/file2.txt" |         $file = "overwrite/some/new/path/file1.txt" | ||||||
|         if(!(Test-Path -path $file)) |         if(!(Test-Path -path $file)) | ||||||
|         { |         { | ||||||
|             Write-Error "Expected file does not exist" |             Write-Error "Expected file does not exist" | ||||||
|         } |         } | ||||||
|         if(!((Get-Content $file) -ceq "Hello world from file #2")) |         if(!((Get-Content $file) -ceq "This file has changed")) | ||||||
|         { |         { | ||||||
|             Write-Error "File contents of downloaded artifacts are incorrect" |             Write-Error "File contents of downloaded artifact are incorrect" | ||||||
|         } |         } | ||||||
|       shell: pwsh |       shell: pwsh | ||||||
|  |   merge: | ||||||
|  |     name: Merge | ||||||
|  |     needs: build | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  | 
 | ||||||
|  |     steps: | ||||||
|  |     - name: Checkout | ||||||
|  |       uses: actions/checkout@v4 | ||||||
|  | 
 | ||||||
|  |     # Merge all artifacts from previous jobs | ||||||
|  |     - name: Merge all artifacts in run | ||||||
|  |       uses: ./merge/ | ||||||
|  |       with: | ||||||
|  |         # our matrix produces artifacts with the same file, this prevents "stomping" on each other, also makes it | ||||||
|  |         # easier to identify each of the merged artifacts | ||||||
|  |         separate-directories: true | ||||||
|  |     - name: 'Download merged artifacts' | ||||||
|  |       uses: actions/download-artifact@v4 | ||||||
|  |       with: | ||||||
|  |         name: merged-artifacts | ||||||
|  |         path: all-merged-artifacts | ||||||
|  |     - name: 'Check merged artifact has directories for each artifact' | ||||||
|  |       run: | | ||||||
|  |         $artifacts = @( | ||||||
|  |           "Artifact-A-ubuntu-latest", | ||||||
|  |           "Artifact-A-macos-latest", | ||||||
|  |           "Artifact-A-windows-latest", | ||||||
|  |           "Artifact-Wildcard-ubuntu-latest", | ||||||
|  |           "Artifact-Wildcard-macos-latest", | ||||||
|  |           "Artifact-Wildcard-windows-latest", | ||||||
|  |           "Multi-Path-Artifact-ubuntu-latest", | ||||||
|  |           "Multi-Path-Artifact-macos-latest", | ||||||
|  |           "Multi-Path-Artifact-windows-latest" | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         foreach ($artifact in $artifacts) { | ||||||
|  |           $path = "all-merged-artifacts/$artifact" | ||||||
|  |           if (!(Test-Path $path)) { | ||||||
|  |             Write-Error "$path does not exist." | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       shell: pwsh | ||||||
|  | 
 | ||||||
|  |     # Merge Artifact-A-* from previous jobs | ||||||
|  |     - name: Merge all Artifact-A | ||||||
|  |       uses: ./merge/ | ||||||
|  |       with: | ||||||
|  |         name: Merged-Artifact-As | ||||||
|  |         pattern: 'Artifact-A-*' | ||||||
|  |         separate-directories: true | ||||||
|  | 
 | ||||||
|  |     # Download merged artifacts and verify the correctness of the content | ||||||
|  |     - name: 'Download merged artifacts' | ||||||
|  |       uses: actions/download-artifact@v4 | ||||||
|  |       with: | ||||||
|  |         name: Merged-Artifact-As | ||||||
|  |         path: merged-artifact-a | ||||||
|  | 
 | ||||||
|  |     - name: 'Verify merged artifacts' | ||||||
|  |       run: | | ||||||
|  |         $files = @( | ||||||
|  |           "merged-artifact-a/Artifact-A-ubuntu-latest/file1.txt", | ||||||
|  |           "merged-artifact-a/Artifact-A-macos-latest/file1.txt", | ||||||
|  |           "merged-artifact-a/Artifact-A-windows-latest/file1.txt" | ||||||
|  |         ) | ||||||
|  | 
 | ||||||
|  |         foreach ($file in $files) { | ||||||
|  |           if (!(Test-Path $file)) { | ||||||
|  |             Write-Error "$file does not exist." | ||||||
|  |           } | ||||||
|  | 
 | ||||||
|  |           if (!((Get-Content $file) -ceq "This file has changed")) { | ||||||
|  |             Write-Error "$file has incorrect content." | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       shell: pwsh | ||||||
|  | 
 | ||||||
|  | |||||||
							
								
								
									
										26
									
								
								.licenses/npm/minimatch.dep.yml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								.licenses/npm/minimatch.dep.yml
									
									
									
										generated
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | --- | ||||||
|  | name: minimatch | ||||||
|  | version: 9.0.3 | ||||||
|  | type: npm | ||||||
|  | summary:  | ||||||
|  | homepage:  | ||||||
|  | license: isc | ||||||
|  | licenses: | ||||||
|  | - sources: LICENSE | ||||||
|  |   text: | | ||||||
|  |     The ISC License | ||||||
|  | 
 | ||||||
|  |     Copyright (c) 2011-2023 Isaac Z. Schlueter and Contributors | ||||||
|  | 
 | ||||||
|  |     Permission to use, copy, modify, and/or distribute this software for any | ||||||
|  |     purpose with or without fee is hereby granted, provided that the above | ||||||
|  |     copyright notice and this permission notice appear in all copies. | ||||||
|  | 
 | ||||||
|  |     THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES | ||||||
|  |     WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF | ||||||
|  |     MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR | ||||||
|  |     ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES | ||||||
|  |     WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN | ||||||
|  |     ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR | ||||||
|  |     IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. | ||||||
|  | notices: [] | ||||||
| @ -41,6 +41,8 @@ The release of upload-artifact@v4 and download-artifact@v4 are major changes to | |||||||
| 
 | 
 | ||||||
| For more information, see the [`@actions/artifact`](https://github.com/actions/toolkit/tree/main/packages/artifact) documentation. | For more information, see the [`@actions/artifact`](https://github.com/actions/toolkit/tree/main/packages/artifact) documentation. | ||||||
| 
 | 
 | ||||||
|  | There is also a new sub-action, `actions/upload-artifact/merge`. For more info, check out that action's [README](./merge/README.md). | ||||||
|  | 
 | ||||||
| ### Improvements | ### Improvements | ||||||
| 
 | 
 | ||||||
| 1. Uploads are significantly faster, upwards of 90% improvement in worst case scenarios. | 1. Uploads are significantly faster, upwards of 90% improvement in worst case scenarios. | ||||||
| @ -406,7 +408,7 @@ jobs: | |||||||
| 
 | 
 | ||||||
| ### Number of Artifacts | ### Number of Artifacts | ||||||
| 
 | 
 | ||||||
| Within an individual job, there is a limit of 10 artifacts that can be created for that job. | Within an individual job, there is a limit of 500 artifacts that can be created for that job. | ||||||
| 
 | 
 | ||||||
| You may also be limited by Artifacts if you have exceeded your shared storage quota. Storage is calculated every 6-12 hours. See [the documentation](https://docs.github.com/en/billing/managing-billing-for-github-actions/about-billing-for-github-actions#calculating-minute-and-storage-spending) for more info. | You may also be limited by Artifacts if you have exceeded your shared storage quota. Storage is calculated every 6-12 hours. See [the documentation](https://docs.github.com/en/billing/managing-billing-for-github-actions/about-billing-for-github-actions#calculating-minute-and-storage-spending) for more info. | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										175
									
								
								__tests__/merge.test.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								__tests__/merge.test.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,175 @@ | |||||||
|  | import * as core from '@actions/core' | ||||||
|  | import artifact from '@actions/artifact' | ||||||
|  | import {run} from '../src/merge/merge-artifacts' | ||||||
|  | import {Inputs} from '../src/merge/constants' | ||||||
|  | import * as search from '../src/shared/search' | ||||||
|  | 
 | ||||||
|  | const fixtures = { | ||||||
|  |   artifactName: 'my-merged-artifact', | ||||||
|  |   tmpDirectory: '/tmp/merge-artifact', | ||||||
|  |   filesToUpload: [ | ||||||
|  |     '/some/artifact/path/file-a.txt', | ||||||
|  |     '/some/artifact/path/file-b.txt', | ||||||
|  |     '/some/artifact/path/file-c.txt' | ||||||
|  |   ], | ||||||
|  |   artifacts: [ | ||||||
|  |     { | ||||||
|  |       name: 'my-artifact-a', | ||||||
|  |       id: 1, | ||||||
|  |       size: 100, | ||||||
|  |       createdAt: new Date('2024-01-01T00:00:00Z') | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       name: 'my-artifact-b', | ||||||
|  |       id: 2, | ||||||
|  |       size: 100, | ||||||
|  |       createdAt: new Date('2024-01-01T00:00:00Z') | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       name: 'my-artifact-c', | ||||||
|  |       id: 3, | ||||||
|  |       size: 100, | ||||||
|  |       createdAt: new Date('2024-01-01T00:00:00Z') | ||||||
|  |     } | ||||||
|  |   ] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | jest.mock('@actions/github', () => ({ | ||||||
|  |   context: { | ||||||
|  |     repo: { | ||||||
|  |       owner: 'actions', | ||||||
|  |       repo: 'toolkit' | ||||||
|  |     }, | ||||||
|  |     runId: 123, | ||||||
|  |     serverUrl: 'https://github.com' | ||||||
|  |   } | ||||||
|  | })) | ||||||
|  | 
 | ||||||
|  | jest.mock('@actions/core') | ||||||
|  | 
 | ||||||
|  | jest.mock('fs/promises', () => ({ | ||||||
|  |   mkdtemp: jest.fn().mockResolvedValue('/tmp/merge-artifact'), | ||||||
|  |   rm: jest.fn().mockResolvedValue(undefined) | ||||||
|  | })) | ||||||
|  | 
 | ||||||
|  | /* eslint-disable no-unused-vars */ | ||||||
|  | const mockInputs = (overrides?: Partial<{[K in Inputs]?: any}>) => { | ||||||
|  |   const inputs = { | ||||||
|  |     [Inputs.Name]: 'my-merged-artifact', | ||||||
|  |     [Inputs.Pattern]: '*', | ||||||
|  |     [Inputs.SeparateDirectories]: false, | ||||||
|  |     [Inputs.RetentionDays]: 0, | ||||||
|  |     [Inputs.CompressionLevel]: 6, | ||||||
|  |     [Inputs.DeleteMerged]: false, | ||||||
|  |     ...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('merge', () => { | ||||||
|  |   beforeEach(async () => { | ||||||
|  |     mockInputs() | ||||||
|  | 
 | ||||||
|  |     jest | ||||||
|  |       .spyOn(artifact, 'listArtifacts') | ||||||
|  |       .mockResolvedValue({artifacts: fixtures.artifacts}) | ||||||
|  | 
 | ||||||
|  |     jest.spyOn(artifact, 'downloadArtifact').mockResolvedValue({ | ||||||
|  |       downloadPath: fixtures.tmpDirectory | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     jest.spyOn(search, 'findFilesToUpload').mockResolvedValue({ | ||||||
|  |       filesToUpload: fixtures.filesToUpload, | ||||||
|  |       rootDirectory: fixtures.tmpDirectory | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     jest.spyOn(artifact, 'uploadArtifact').mockResolvedValue({ | ||||||
|  |       size: 123, | ||||||
|  |       id: 1337 | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     jest | ||||||
|  |       .spyOn(artifact, 'deleteArtifact') | ||||||
|  |       .mockImplementation(async artifactName => { | ||||||
|  |         const artifact = fixtures.artifacts.find(a => a.name === artifactName) | ||||||
|  |         if (!artifact) throw new Error(`Artifact ${artifactName} not found`) | ||||||
|  |         return {id: artifact.id} | ||||||
|  |       }) | ||||||
|  |   }) | ||||||
|  | 
 | ||||||
|  |   it('merges artifacts', async () => { | ||||||
|  |     await run() | ||||||
|  | 
 | ||||||
|  |     for (const a of fixtures.artifacts) { | ||||||
|  |       expect(artifact.downloadArtifact).toHaveBeenCalledWith(a.id, { | ||||||
|  |         path: fixtures.tmpDirectory | ||||||
|  |       }) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     expect(artifact.uploadArtifact).toHaveBeenCalledWith( | ||||||
|  |       fixtures.artifactName, | ||||||
|  |       fixtures.filesToUpload, | ||||||
|  |       fixtures.tmpDirectory, | ||||||
|  |       {compressionLevel: 6} | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  | 
 | ||||||
|  |   it('fails if no artifacts found', async () => { | ||||||
|  |     mockInputs({[Inputs.Pattern]: 'this-does-not-match'}) | ||||||
|  | 
 | ||||||
|  |     expect(run()).rejects.toThrow() | ||||||
|  | 
 | ||||||
|  |     expect(artifact.uploadArtifact).not.toBeCalled() | ||||||
|  |     expect(artifact.downloadArtifact).not.toBeCalled() | ||||||
|  |   }) | ||||||
|  | 
 | ||||||
|  |   it('supports custom compression level', async () => { | ||||||
|  |     mockInputs({ | ||||||
|  |       [Inputs.CompressionLevel]: 2 | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     await run() | ||||||
|  | 
 | ||||||
|  |     expect(artifact.uploadArtifact).toHaveBeenCalledWith( | ||||||
|  |       fixtures.artifactName, | ||||||
|  |       fixtures.filesToUpload, | ||||||
|  |       fixtures.tmpDirectory, | ||||||
|  |       {compressionLevel: 2} | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  | 
 | ||||||
|  |   it('supports custom retention days', async () => { | ||||||
|  |     mockInputs({ | ||||||
|  |       [Inputs.RetentionDays]: 7 | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     await run() | ||||||
|  | 
 | ||||||
|  |     expect(artifact.uploadArtifact).toHaveBeenCalledWith( | ||||||
|  |       fixtures.artifactName, | ||||||
|  |       fixtures.filesToUpload, | ||||||
|  |       fixtures.tmpDirectory, | ||||||
|  |       {retentionDays: 7, compressionLevel: 6} | ||||||
|  |     ) | ||||||
|  |   }) | ||||||
|  | 
 | ||||||
|  |   it('supports deleting artifacts after merge', async () => { | ||||||
|  |     mockInputs({ | ||||||
|  |       [Inputs.DeleteMerged]: true | ||||||
|  |     }) | ||||||
|  | 
 | ||||||
|  |     await run() | ||||||
|  | 
 | ||||||
|  |     for (const a of fixtures.artifacts) { | ||||||
|  |       expect(artifact.deleteArtifact).toHaveBeenCalledWith(a.name) | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  | }) | ||||||
							
								
								
									
										133673
									
								
								dist/merge/index.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										133673
									
								
								dist/merge/index.js
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										4842
									
								
								dist/upload/index.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4842
									
								
								dist/upload/index.js
									
									
									
									
										vendored
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -3,6 +3,7 @@ | |||||||
| - [Migration](#migration) | - [Migration](#migration) | ||||||
|   - [Multiple uploads to the same named Artifact](#multiple-uploads-to-the-same-named-artifact) |   - [Multiple uploads to the same named Artifact](#multiple-uploads-to-the-same-named-artifact) | ||||||
|   - [Overwriting an Artifact](#overwriting-an-artifact) |   - [Overwriting an Artifact](#overwriting-an-artifact) | ||||||
|  |   - [Merging multiple artifacts](#merging-multiple-artifacts) | ||||||
| 
 | 
 | ||||||
| Several behavioral differences exist between Artifact actions `v3` and below vs `v4`. This document outlines common scenarios in `v3`, and how they would be handled in `v4`. | Several behavioral differences exist between Artifact actions `v3` and below vs `v4`. This document outlines common scenarios in `v3`, and how they would be handled in `v4`. | ||||||
| 
 | 
 | ||||||
| @ -144,3 +145,64 @@ jobs: | |||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| Note that this will create an _entirely_ new Artifact, with a different ID from the previous. | Note that this will create an _entirely_ new Artifact, with a different ID from the previous. | ||||||
|  | 
 | ||||||
|  | ## Merging multiple artifacts | ||||||
|  | 
 | ||||||
|  | In `v3`, multiple uploads from multiple jobs could be done to the same Artifact. This would result in a single archive, which could be useful for sending to upstream systems outside of Actions via API or UI downloads. | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   upload: | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         runs-on: [ubuntu-latest, macos-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.runs-on }} | ||||||
|  |     steps: | ||||||
|  |       - name: Create a File | ||||||
|  |         run: echo "hello from ${{ matrix.runs-on }}" > file-${{ matrix.runs-on }}.txt | ||||||
|  |       - name: Upload Artifact | ||||||
|  |         uses: actions/upload-artifact@v3 | ||||||
|  |         with: | ||||||
|  |           name: all-my-files # NOTE: same artifact name | ||||||
|  |           path: file-${{ matrix.runs-on }}.txt | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | The single `all-my-files` artifact would contain the following: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | . | ||||||
|  |   ∟ file-ubuntu-latest.txt | ||||||
|  |   ∟ file-macos-latest.txt | ||||||
|  |   ∟ file-windows-latest.txt | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | To achieve the same in `v4` you can change it like so: | ||||||
|  | 
 | ||||||
|  | ```diff | ||||||
|  | jobs: | ||||||
|  |   upload: | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         runs-on: [ubuntu-latest, macos-latest, windows-latest] | ||||||
|  |     runs-on: ${{ matrix.runs-on }} | ||||||
|  |     steps: | ||||||
|  |       - name: Create a File | ||||||
|  |         run: echo "hello from ${{ matrix.runs-on }}" > file-${{ matrix.runs-on }}.txt | ||||||
|  |       - name: Upload Artifact | ||||||
|  |         uses: actions/upload-artifact@v3 | ||||||
|  |         with: | ||||||
|  | -         name: all-my-files | ||||||
|  | +         name: my-artifact-${{ matrix.runs-on }} | ||||||
|  |           path: file-${{ matrix.runs-on }}.txt | ||||||
|  | + merge: | ||||||
|  | +   runs-on: ubuntu-latest | ||||||
|  | +   needs: upload | ||||||
|  | +   steps: | ||||||
|  | +     - name: Merge Artifacts | ||||||
|  | +       uses: actions/upload-artifact/merge@v4 | ||||||
|  | +       with: | ||||||
|  | +         name: all-my-files | ||||||
|  | +         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](../merge/README.md). | ||||||
|  | |||||||
							
								
								
									
										200
									
								
								merge/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								merge/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,200 @@ | |||||||
|  | # `@actions/upload-artifact/merge` | ||||||
|  | 
 | ||||||
|  | Merge multiple [Actions Artifacts](https://docs.github.com/en/actions/using-workflows/storing-workflow-data-as-artifacts) in Workflow Runs. Internally powered by [@actions/artifact](https://github.com/actions/toolkit/tree/main/packages/artifact) package. | ||||||
|  | 
 | ||||||
|  | - [`@actions/upload-artifact/merge`](#actionsupload-artifactmerge) | ||||||
|  |   - [Usage](#usage) | ||||||
|  |     - [Inputs](#inputs) | ||||||
|  |     - [Outputs](#outputs) | ||||||
|  |   - [Examples](#examples) | ||||||
|  |     - [Combining all artifacts in a workflow run](#combining-all-artifacts-in-a-workflow-run) | ||||||
|  |     - [Prefix directories in merged artifact](#prefix-directories-in-merged-artifact) | ||||||
|  |     - [Deleting artifacts after merge](#deleting-artifacts-after-merge) | ||||||
|  |     - [Retention and Compression Level](#retention-and-compression-level) | ||||||
|  | 
 | ||||||
|  | ## Usage | ||||||
|  | 
 | ||||||
|  | > [!IMPORTANT] | ||||||
|  | > upload-artifact/merge@v4+ is not currently supported on GHES. | ||||||
|  | 
 | ||||||
|  | Note: this actions can only merge artifacts created with actions/upload-artifact@v4+ | ||||||
|  | 
 | ||||||
|  | This sub-action is a helper to merge multiple artifacts after they are created. To do so, it will download multiple artifacts to a temporary directory and reupload them as a single artifact. | ||||||
|  | 
 | ||||||
|  | For most cases, this may not be the most efficient solution. See [the migration docs](../docs/MIGRATION.md#multiple-uploads-to-the-same-named-artifact) on how to download multiple artifacts to the same directory on a runner. This action should only be necessary for cases where multiple artifacts will need to be downloaded outside the runner environment, like downloads via the UI or REST API. | ||||||
|  | 
 | ||||||
|  | ### Inputs | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | - uses: actions/upload-artifact/merge@v4 | ||||||
|  |   with: | ||||||
|  |     # The name of the artifact that the artifacts will be merged into | ||||||
|  |     # Optional. Default is 'merged-artifacts' | ||||||
|  |     name: | ||||||
|  | 
 | ||||||
|  |     # A glob pattern matching the artifacts that should be merged. | ||||||
|  |     # Optional. Default is '*' | ||||||
|  |     pattern: | ||||||
|  | 
 | ||||||
|  |     # If true, the artifacts will be merged into separate directories. | ||||||
|  |     # If false, the artifacts will be merged into the root of the destination. | ||||||
|  |     # Optional. Default is 'false' | ||||||
|  |     separate-directories: | ||||||
|  | 
 | ||||||
|  |     # If true, the artifacts that were merged will be deleted. | ||||||
|  |     # If false, the artifacts will still exist. | ||||||
|  |     # Optional. Default is 'false' | ||||||
|  |     delete-merged: | ||||||
|  | 
 | ||||||
|  |     # Duration after which artifact will expire in days. 0 means using default retention. | ||||||
|  |     # Minimum 1 day. | ||||||
|  |     # Maximum 90 days unless changed from the repository settings page. | ||||||
|  |     # Optional. Defaults to repository settings. | ||||||
|  |     retention-days: | ||||||
|  | 
 | ||||||
|  |     # The level of compression for Zlib to be applied to the artifact archive. | ||||||
|  |     # The value can range from 0 to 9. | ||||||
|  |     # For large files that are not easily compressed, a value of 0 is recommended for significantly faster uploads. | ||||||
|  |     # Optional. Default is '6' | ||||||
|  |     compression-level: | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Outputs | ||||||
|  | 
 | ||||||
|  | | Name | Description | Example | | ||||||
|  | | - | - | - | | ||||||
|  | | `artifact-id` | GitHub ID of an Artifact, can be used by the REST API | `1234` | | ||||||
|  | | `artifact-url` | URL to download an Artifact. Can be used in many scenarios such as linking to artifacts in issues or pull requests. Users must be logged-in in order for this URL to work. This URL is valid as long as the artifact has not expired or the artifact, run or repository have not been deleted | `https://github.com/example-org/example-repo/actions/runs/1/artifacts/1234` | | ||||||
|  | 
 | ||||||
|  | ## Examples | ||||||
|  | 
 | ||||||
|  | For each of these examples, assume we have a prior job matrix that generates three artifacts: `my-artifact-a`, `my-artifact-b` and `my-artifact-c`. | ||||||
|  | 
 | ||||||
|  | e.g. | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   upload: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  | 
 | ||||||
|  |     strategy: | ||||||
|  |       matrix: | ||||||
|  |         foo: [a, b, c] | ||||||
|  | 
 | ||||||
|  |     steps: | ||||||
|  |       - name: Run a one-line script | ||||||
|  |         run: echo "hello from job ${{ matrix.foo }}" > file-${{ matrix.foo }}.txt | ||||||
|  |       - name: Upload | ||||||
|  |         uses: actions/upload-artifact@v4 | ||||||
|  |         with: | ||||||
|  |           name: my-artifact-${{ matrix.foo }} | ||||||
|  |           path: file-${{ matrix.foo }}.txt | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | Each of the following examples will use the `needs: upload` as a prerequesite before any merging operations. | ||||||
|  | 
 | ||||||
|  | ### Combining all artifacts in a workflow run | ||||||
|  | 
 | ||||||
|  | By default (with no inputs), calling this action will take all the artifacts in the workflow run and combined them into a single artifact called `merged-artifacts`: | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   # ... <upload job> ... | ||||||
|  |   merge: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: upload | ||||||
|  |     steps: | ||||||
|  |       - name: Merge Artifacts | ||||||
|  |         uses: actions/upload-artifact/merge@v4 | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | This will result in an artifact called `merged-artifacts` with the following content: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | . | ||||||
|  |   ∟ file-a.txt | ||||||
|  |   ∟ file-b.txt | ||||||
|  |   ∟ file-c.txt | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | To change the name of the artifact and filter on what artifacts are added, you can use the `name` and `pattern` inputs: | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   # ... <upload job> ... | ||||||
|  |   merge: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: upload | ||||||
|  |     steps: | ||||||
|  |       - name: Merge Artifacts | ||||||
|  |         uses: actions/upload-artifact/merge@v4 | ||||||
|  |         with: | ||||||
|  |           name: my-amazing-merged-artifact | ||||||
|  |           pattern: my-artifact-* | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Prefix directories in merged artifact | ||||||
|  | 
 | ||||||
|  | To prevent overwriting files in artifacts that may have the same name, you can use the `separate-directories` to prefix the extracted files with directories (named after the original artifact): | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   # ... <upload job> ... | ||||||
|  |   merge: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: upload | ||||||
|  |     steps: | ||||||
|  |       - name: Merge Artifacts | ||||||
|  |         uses: actions/upload-artifact/merge@v4 | ||||||
|  |         with: | ||||||
|  |           separate-directories: true | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | This will result in the following artifact structure: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | . | ||||||
|  |   ∟ my-artifact-a | ||||||
|  |     ∟ file-a.txt | ||||||
|  |   ∟ my-artifact-b | ||||||
|  |     ∟ file-b.txt | ||||||
|  |   ∟ my-artifact-c | ||||||
|  |     ∟ file-c.txt | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Deleting artifacts after merge | ||||||
|  | 
 | ||||||
|  | After merge, the old artifacts may no longer be required. To automatically delete them after they are merged into a new artifact, you can use `delete-merged` like so: | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   # ... <upload job> ... | ||||||
|  |   merge: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: upload | ||||||
|  |     steps: | ||||||
|  |       - name: Merge Artifacts | ||||||
|  |         uses: actions/upload-artifact/merge@v4 | ||||||
|  |         with: | ||||||
|  |           delete-merged: true | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | After this runs, the matching artifact (`my-artifact-a`, `my-artifact-b` and `my-artifact-c`) will be merged. | ||||||
|  | 
 | ||||||
|  | ### Retention and Compression Level | ||||||
|  | 
 | ||||||
|  | Similar to actions/upload-artifact, both [`retention-days`](../README.md#retention-period) and [`compression-level`](../README.md#altering-compressions-level-speed-v-size) are supported: | ||||||
|  | 
 | ||||||
|  | ```yaml | ||||||
|  | jobs: | ||||||
|  |   # ... <upload job> ... | ||||||
|  |   merge: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |     needs: upload | ||||||
|  |     steps: | ||||||
|  |       - name: Merge Artifacts | ||||||
|  |         uses: actions/upload-artifact/merge@v4 | ||||||
|  |         with: | ||||||
|  |           retention-days: 1 | ||||||
|  |           compression-level: 9 | ||||||
|  | ``` | ||||||
							
								
								
									
										57
									
								
								merge/action.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								merge/action.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | |||||||
|  | name: 'Merge Build Artifacts' | ||||||
|  | description: 'Merge one or more build Artifacts' | ||||||
|  | author: 'GitHub' | ||||||
|  | inputs: | ||||||
|  |   name: | ||||||
|  |     description: 'The name of the artifact that the artifacts will be merged into.' | ||||||
|  |     required: true | ||||||
|  |     default: 'merged-artifacts' | ||||||
|  |   pattern: | ||||||
|  |     description: 'A glob pattern matching the artifact names that should be merged.' | ||||||
|  |     default: '*' | ||||||
|  |   separate-directories: | ||||||
|  |     description: 'When multiple artifacts are matched, this changes the behavior of how they are merged in the archive. | ||||||
|  |       If true, the matched artifacts will be extracted into individual named directories within the specified path. | ||||||
|  |       If false, the matched artifacts will combined in the same directory.' | ||||||
|  |     default: 'false' | ||||||
|  |   retention-days: | ||||||
|  |     description: > | ||||||
|  |       Duration after which artifact will expire in days. 0 means using default retention. | ||||||
|  | 
 | ||||||
|  |       Minimum 1 day. | ||||||
|  |       Maximum 90 days unless changed from the repository settings page. | ||||||
|  |   compression-level: | ||||||
|  |     description: > | ||||||
|  |       The level of compression for Zlib to be applied to the artifact archive. | ||||||
|  |       The value can range from 0 to 9: | ||||||
|  |       - 0: No compression | ||||||
|  |       - 1: Best speed | ||||||
|  |       - 6: Default compression (same as GNU Gzip) | ||||||
|  |       - 9: Best compression | ||||||
|  |       Higher levels will result in better compression, but will take longer to complete. | ||||||
|  |       For large files that are not easily compressed, a value of 0 is recommended for significantly faster uploads. | ||||||
|  |     default: '6' | ||||||
|  |   delete-merged: | ||||||
|  |     description: > | ||||||
|  |       If true, the artifacts that were merged will be deleted. | ||||||
|  |       If false, the artifacts will still exist. | ||||||
|  |     default: 'false' | ||||||
|  | 
 | ||||||
|  | outputs: | ||||||
|  |   artifact-id: | ||||||
|  |     description: > | ||||||
|  |       A unique identifier for the artifact that was just uploaded. Empty if the artifact upload failed. | ||||||
|  | 
 | ||||||
|  |       This ID can be used as input to other APIs to download, delete or get more information about an artifact: https://docs.github.com/en/rest/actions/artifacts | ||||||
|  |   artifact-url: | ||||||
|  |     description: > | ||||||
|  |       A download URL for the artifact that was just uploaded. Empty if the artifact upload failed. | ||||||
|  | 
 | ||||||
|  |       This download URL only works for requests Authenticated with GitHub. Anonymous downloads will be prompted to first login.  | ||||||
|  |       If an anonymous download URL is needed than a short time restricted URL can be generated using the download artifact API: https://docs.github.com/en/rest/actions/artifacts#download-an-artifact     | ||||||
|  | 
 | ||||||
|  |       This URL will be valid for as long as the artifact exists and the workflow run and repository exists. Once an artifact has expired this URL will no longer work. | ||||||
|  |       Common uses cases for such a download URL can be adding download links to artifacts in descriptions or comments on pull requests or issues. | ||||||
|  | runs: | ||||||
|  |   using: 'node20' | ||||||
|  |   main: '../dist/merge/index.js' | ||||||
							
								
								
									
										648
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										648
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -1,11 +1,11 @@ | |||||||
| { | { | ||||||
|   "name": "upload-artifact", |   "name": "upload-artifact", | ||||||
|   "version": "4.2.0", |   "version": "4.3.0", | ||||||
|   "description": "Upload an Actions Artifact in a workflow run", |   "description": "Upload an Actions Artifact in a workflow run", | ||||||
|   "main": "dist/upload/index.js", |   "main": "dist/upload/index.js", | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "build": "tsc", |     "build": "tsc", | ||||||
|     "release": "ncc build src/upload/index.ts -o dist/upload", |     "release": "ncc build src/upload/index.ts -o dist/upload && ncc build src/merge/index.ts -o dist/merge", | ||||||
|     "check-all": "concurrently \"npm:format-check\" \"npm:lint\" \"npm:test\" \"npm:build\"", |     "check-all": "concurrently \"npm:format-check\" \"npm:lint\" \"npm:test\" \"npm:build\"", | ||||||
|     "format": "prettier --write **/*.ts", |     "format": "prettier --write **/*.ts", | ||||||
|     "format-check": "prettier --check **/*.ts", |     "format-check": "prettier --check **/*.ts", | ||||||
| @ -33,7 +33,8 @@ | |||||||
|     "@actions/core": "^1.10.0", |     "@actions/core": "^1.10.0", | ||||||
|     "@actions/github": "^6.0.0", |     "@actions/github": "^6.0.0", | ||||||
|     "@actions/glob": "^0.3.0", |     "@actions/glob": "^0.3.0", | ||||||
|     "@actions/io": "^1.1.2" |     "@actions/io": "^1.1.2", | ||||||
|  |     "minimatch": "^9.0.3" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@types/jest": "^29.2.5", |     "@types/jest": "^29.2.5", | ||||||
|  | |||||||
							
								
								
									
										9
									
								
								src/merge/constants.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/merge/constants.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | /* eslint-disable no-unused-vars */ | ||||||
|  | export enum Inputs { | ||||||
|  |   Name = 'name', | ||||||
|  |   Pattern = 'pattern', | ||||||
|  |   SeparateDirectories = 'separate-directories', | ||||||
|  |   RetentionDays = 'retention-days', | ||||||
|  |   CompressionLevel = 'compression-level', | ||||||
|  |   DeleteMerged = 'delete-merged' | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								src/merge/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								src/merge/index.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,6 @@ | |||||||
|  | import * as core from '@actions/core' | ||||||
|  | import {run} from './merge-artifacts' | ||||||
|  | 
 | ||||||
|  | run().catch(error => { | ||||||
|  |   core.setFailed((error as Error).message) | ||||||
|  | }) | ||||||
							
								
								
									
										44
									
								
								src/merge/input-helper.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								src/merge/input-helper.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,44 @@ | |||||||
|  | import * as core from '@actions/core' | ||||||
|  | import {Inputs} from './constants' | ||||||
|  | import {MergeInputs} from './merge-inputs' | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Helper to get all the inputs for the action | ||||||
|  |  */ | ||||||
|  | export function getInputs(): MergeInputs { | ||||||
|  |   const name = core.getInput(Inputs.Name, {required: true}) | ||||||
|  |   const pattern = core.getInput(Inputs.Pattern, {required: true}) | ||||||
|  |   const separateDirectories = core.getBooleanInput(Inputs.SeparateDirectories) | ||||||
|  |   const deleteMerged = core.getBooleanInput(Inputs.DeleteMerged) | ||||||
|  | 
 | ||||||
|  |   const inputs = { | ||||||
|  |     name, | ||||||
|  |     pattern, | ||||||
|  |     separateDirectories, | ||||||
|  |     deleteMerged, | ||||||
|  |     retentionDays: 0, | ||||||
|  |     compressionLevel: 6 | ||||||
|  |   } as MergeInputs | ||||||
|  | 
 | ||||||
|  |   const retentionDaysStr = core.getInput(Inputs.RetentionDays) | ||||||
|  |   if (retentionDaysStr) { | ||||||
|  |     inputs.retentionDays = parseInt(retentionDaysStr) | ||||||
|  |     if (isNaN(inputs.retentionDays)) { | ||||||
|  |       core.setFailed('Invalid retention-days') | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const compressionLevelStr = core.getInput(Inputs.CompressionLevel) | ||||||
|  |   if (compressionLevelStr) { | ||||||
|  |     inputs.compressionLevel = parseInt(compressionLevelStr) | ||||||
|  |     if (isNaN(inputs.compressionLevel)) { | ||||||
|  |       core.setFailed('Invalid compression-level') | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (inputs.compressionLevel < 0 || inputs.compressionLevel > 9) { | ||||||
|  |       core.setFailed('Invalid compression-level. Valid values are 0-9') | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return inputs | ||||||
|  | } | ||||||
							
								
								
									
										93
									
								
								src/merge/merge-artifacts.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								src/merge/merge-artifacts.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,93 @@ | |||||||
|  | import * as path from 'path' | ||||||
|  | import {mkdtemp, rm} from 'fs/promises' | ||||||
|  | import * as core from '@actions/core' | ||||||
|  | import {Minimatch} from 'minimatch' | ||||||
|  | import artifactClient, {UploadArtifactOptions} from '@actions/artifact' | ||||||
|  | import {getInputs} from './input-helper' | ||||||
|  | import {uploadArtifact} from '../shared/upload-artifact' | ||||||
|  | import {findFilesToUpload} from '../shared/search' | ||||||
|  | 
 | ||||||
|  | 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[][]) | ||||||
|  | 
 | ||||||
|  | export async function run(): Promise<void> { | ||||||
|  |   const inputs = getInputs() | ||||||
|  |   const tmpDir = await mkdtemp('merge-artifact') | ||||||
|  | 
 | ||||||
|  |   const listArtifactResponse = await artifactClient.listArtifacts({ | ||||||
|  |     latest: true | ||||||
|  |   }) | ||||||
|  |   const matcher = new Minimatch(inputs.pattern) | ||||||
|  |   const artifacts = listArtifactResponse.artifacts.filter(artifact => | ||||||
|  |     matcher.match(artifact.name) | ||||||
|  |   ) | ||||||
|  |   core.debug( | ||||||
|  |     `Filtered from ${listArtifactResponse.artifacts.length} to ${artifacts.length} artifacts` | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   if (artifacts.length === 0) { | ||||||
|  |     throw new Error(`No artifacts found matching pattern '${inputs.pattern}'`) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   core.info(`Preparing to download the following artifacts:`) | ||||||
|  |   artifacts.forEach(artifact => { | ||||||
|  |     core.info(`- ${artifact.name} (ID: ${artifact.id}, Size: ${artifact.size})`) | ||||||
|  |   }) | ||||||
|  | 
 | ||||||
|  |   const downloadPromises = artifacts.map(artifact => | ||||||
|  |     artifactClient.downloadArtifact(artifact.id, { | ||||||
|  |       path: inputs.separateDirectories | ||||||
|  |         ? path.join(tmpDir, artifact.name) | ||||||
|  |         : tmpDir | ||||||
|  |     }) | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const chunkedPromises = chunk(downloadPromises, PARALLEL_DOWNLOADS) | ||||||
|  |   for (const chunk of chunkedPromises) { | ||||||
|  |     await Promise.all(chunk) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const options: UploadArtifactOptions = {} | ||||||
|  |   if (inputs.retentionDays) { | ||||||
|  |     options.retentionDays = inputs.retentionDays | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   if (typeof inputs.compressionLevel !== 'undefined') { | ||||||
|  |     options.compressionLevel = inputs.compressionLevel | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const searchResult = await findFilesToUpload(tmpDir) | ||||||
|  | 
 | ||||||
|  |   await uploadArtifact( | ||||||
|  |     inputs.name, | ||||||
|  |     searchResult.filesToUpload, | ||||||
|  |     searchResult.rootDirectory, | ||||||
|  |     options | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   core.info( | ||||||
|  |     `The ${artifacts.length} artifact(s) have been successfully merged!` | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   if (inputs.deleteMerged) { | ||||||
|  |     const deletePromises = artifacts.map(artifact => | ||||||
|  |       artifactClient.deleteArtifact(artifact.name) | ||||||
|  |     ) | ||||||
|  |     await Promise.all(deletePromises) | ||||||
|  |     core.info(`The ${artifacts.length} artifact(s) have been deleted`) | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   try { | ||||||
|  |     await rm(tmpDir, {recursive: true}) | ||||||
|  |   } catch (error) { | ||||||
|  |     core.warning( | ||||||
|  |       `Unable to remove temporary directory: ${(error as Error).message}` | ||||||
|  |     ) | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										33
									
								
								src/merge/merge-inputs.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								src/merge/merge-inputs.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | |||||||
|  | export interface MergeInputs { | ||||||
|  |   /** | ||||||
|  |    * The name of the artifact that the artifacts will be merged into | ||||||
|  |    */ | ||||||
|  |   name: string | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * A glob pattern matching the artifacts that should be merged. | ||||||
|  |    */ | ||||||
|  |   pattern: string | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Duration after which artifact will expire in days | ||||||
|  |    */ | ||||||
|  |   retentionDays: number | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * The level of compression for Zlib to be applied to the artifact archive. | ||||||
|  |    */ | ||||||
|  |   compressionLevel?: number | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * If true, the artifacts that were merged will be deleted. | ||||||
|  |    * If false, the artifacts will still exist. | ||||||
|  |    */ | ||||||
|  |   deleteMerged: boolean | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * If true, the artifacts will be merged into separate directories. | ||||||
|  |    * If false, the artifacts will be merged into the root of the destination. | ||||||
|  |    */ | ||||||
|  |   separateDirectories: boolean | ||||||
|  | } | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Rob Herley
						Rob Herley