diff --git a/README.md b/README.md index 6d6f76fb..30942958 100644 --- a/README.md +++ b/README.md @@ -66,6 +66,11 @@ This action deletes versions of a package from [GitHub Packages](https://github. # Cannot be used with `num-old-versions-to-delete`. delete-only-untagged-versions: + # If true it will also ignore tags matching the regex provided in `ignore-versions` + # Defaults to false. + # Can only be used with `ignore-versions`. + ignore-versions-include-tags: + # The token used to authenticate with GitHub Packages. # Defaults to github.token. # Required if the repo running the workflow does not have access to delete the package. @@ -89,6 +94,8 @@ This action deletes versions of a package from [GitHub Packages](https://github. - `min-versions-to-keep` + `delete-only-pre-release-versions` - `delete-only-untagged-versions` - `min-versions-to-keep` + `delete-only-untagged-versions` + - `ignore-versions-include-tags` + `ignore-versions` + - `ignore-versions-include-tags` + `ignore-versions` + `min-versions-to-keep` # Scenarios @@ -106,6 +113,8 @@ This action deletes versions of a package from [GitHub Packages](https://github. - [Delete oldest version of a package](#delete-oldest-version-of-a-package) - [Delete a specific version of a package](#delete-a-specific-version-of-a-package) - [Delete multiple specific versions of a package](#delete-multiple-specific-versions-of-a-package) + - [Delete all while ignoring particular tags](#delete-all-while-ignoring-particular-tags) + - [Delete all except y latest versions while ignoring particular tags](#delete-all-except-y-latest-versions-while-ignoring-particular-tags) - [License](#license) @@ -405,6 +414,44 @@ This action deletes versions of a package from [GitHub Packages](https://github. token: ${{ secrets.PAT }} ``` +
+ + ### Delete all while ignoring particular tags + + To delete all while ignoring particular tags, the __package-name__, __ignore-versions-include-tags__ and __ignore-versions__ inputs are required. + + __Example__ + + Deletes all versions of a package excluding package containing the `important-tag` tag. + + ```yaml + - uses: actions/delete-package-version@v4 + with: + package-name: 'test-package' + package-type: 'container' + ignore-versions: '^important-tag$' + ignore-versions-include-tags: 'true' + token: ${{ secrets.PAT }} + ``` + + ### Delete all except y latest versions while ignoring particular tags + + To delete all except y latest versions while ignoring particular tags, the __package-name__, __min-versions-to-keep__, __ignore-versions-include-tags__ and __ignore-versions__ inputs are required. + + __Example__ + + Delete all except latest 3 package versions excluding package containing the `important-tag` tag. + + ```yaml + - uses: actions/delete-package-versions@v4 + with: + package-name: 'test-package' + package-type: 'container' + min-versions-to-keep: 3 + ignore-versions: '^important-tag$' + ignore-versions-include-tags: 'true' + ``` + # License The scripts and documentation in this project are released under the [MIT License](https://github.com/actions/delete-package-versions/blob/main/LICENSE) diff --git a/__tests__/delete.test.ts b/__tests__/delete.test.ts index 96dda11a..d9ca0133 100644 --- a/__tests__/delete.test.ts +++ b/__tests__/delete.test.ts @@ -341,6 +341,90 @@ describe('index tests -- call rest', () => { ) }) + it('finalIds test - no versions deleted if tags match ignore version and include tags is enabled', done => { + const numVersions = 10 + let apiCalled = 0 + + const versions = getMockedVersionsResponse( + numVersions, + 0, + 'container', + true + ) + + server.use( + rest.get( + 'https://api.github.com/users/test-owner/packages/container/test-package/versions', + (req, res, ctx) => { + apiCalled++ + return res(ctx.status(200), ctx.json(versions)) + } + ) + ) + + finalIds( + getInput({ + minVersionsToKeep: 2, + packageType: 'container', + ignoreVersions: RegExp('^(latest[1-3]{1})$'), + includeTags: 'true' + }) + ).subscribe(ids => { + expect(apiCalled).toBe(1) + expect(ids).toStrictEqual(['4', '5', '6', '7', '8']) + done() + }) + }) + + it('', done => { + const numVersions = 10 + let apiCalled = 0 + + let date = new Date() + date.setUTCFullYear(2000, 1, 1) + + const version = { + id: 1, + name: '1.0.0', + url: '', + created_at: date.toUTCString(), + package_html_url: '', + updated_at: '', + metadata: { + package_type: 'container', + container: { + tags: [ + 'test', + '9970186500fd471320e7340b256229209899bde5', + 'my-first-pr' + ] as string[] + } + } + } + + server.use( + rest.get( + 'https://api.github.com/users/test-owner/packages/container/test-package/versions', + (req, res, ctx) => { + apiCalled++ + return res(ctx.status(200), ctx.json([version])) + } + ) + ) + + finalIds( + getInput({ + packageType: 'container', + ignoreVersions: RegExp('^production|acceptance|test$'), + includeTags: 'true' + }) + ).subscribe(ids => { + expect(apiCalled).toBe(1) + expect(ids.length).toBe(0) + done() + }) + }) + it('deleteVersions test - missing token', done => { deleteVersions(getInput({token: ''})).subscribe({ error: err => { @@ -477,7 +561,8 @@ const defaultInput: InputParams = { numOldVersionsToDelete: RATE_LIMIT, minVersionsToKeep: -1, ignoreVersions: RegExp('^$'), - token: 'test-token' + token: 'test-token', + includeTags: 'false' } function getInput(params?: InputParams): Input { diff --git a/__tests__/version/get-version.test.ts b/__tests__/version/get-version.test.ts index 313a93dc..8df47eec 100644 --- a/__tests__/version/get-version.test.ts +++ b/__tests__/version/get-version.test.ts @@ -114,8 +114,10 @@ describe('get versions tests -- mock rest', () => { expect(result.versions[i].created_at).toBe(resp[i].created_at) if (i < numTaggedVersions) { expect(result.versions[i].tagged).toBe(true) + expect(result.versions[i].tags.length).toBe(1) } else { expect(result.versions[i].tagged).toBe(false) + expect(result.versions[i].tags.length).toBe(0) } } expect(result.paginate).toBe(true) diff --git a/action.yml b/action.yml index 789a96e9..fd6e0834 100644 --- a/action.yml +++ b/action.yml @@ -64,6 +64,13 @@ inputs: required: false default: "false" + ignore-versions-include-tags: + description: > + Next to the version also ignore tags matching the ignore-versions regex. + By default this is set to false + required: false + default: "false" + token: description: > Token with the necessary scopes to delete package versions. diff --git a/dist/index.js b/dist/index.js index a2729f70..5bc91720 100644 --- a/dist/index.js +++ b/dist/index.js @@ -41,6 +41,10 @@ function finalIds(input) { Then compute number of versions to delete (toDelete) based on the inputs. */ value = value.filter(info => !input.ignoreVersions.test(info.version)); + // Filter out tags that are to be ignored only when including tags is enabled + if (input.includeTags === 'true') { + value = value.filter(info => !info.tags.some(tag => input.ignoreVersions.test(tag))); + } if (input.deleteUntaggedVersions === 'true') { value = value.filter(info => !info.tagged); } @@ -95,7 +99,8 @@ const defaultParams = { ignoreVersions: new RegExp(''), deletePreReleaseVersions: '', token: '', - deleteUntaggedVersions: '' + deleteUntaggedVersions: '', + includeTags: '' }; class Input { constructor(params) { @@ -111,6 +116,7 @@ class Input { this.token = validatedParams.token; this.numDeleted = 0; this.deleteUntaggedVersions = validatedParams.deleteUntaggedVersions; + this.includeTags = validatedParams.includeTags; } hasOldestVersionQueryInfo() { return !!(this.owner && @@ -231,17 +237,18 @@ function getOldestVersions(owner, packageName, packageType, numVersions, page, t }), (0, operators_1.map)(response => { const resp = { versions: response.data.map((version) => { - let tagged = false; + let tags = []; if (package_type === 'container' && version.metadata && version.metadata.container) { - tagged = version.metadata.container.tags.length > 0; + tags = version.metadata.container.tags; } return { id: version.id, version: version.name, created_at: version.created_at, - tagged + tagged: tags.length > 0, + tags }; }), page, @@ -43941,7 +43948,8 @@ function getActionInput() { ignoreVersions: RegExp((0, core_1.getInput)('ignore-versions')), deletePreReleaseVersions: (0, core_1.getInput)('delete-only-pre-release-versions').toLowerCase(), token: (0, core_1.getInput)('token'), - deleteUntaggedVersions: (0, core_1.getInput)('delete-only-untagged-versions').toLowerCase() + deleteUntaggedVersions: (0, core_1.getInput)('delete-only-untagged-versions').toLowerCase(), + includeTags: (0, core_1.getInput)('ignore-versions-include-tags').toLowerCase() }); } function run() { diff --git a/src/delete.ts b/src/delete.ts index 4eb953fe..37c9e334 100644 --- a/src/delete.ts +++ b/src/delete.ts @@ -76,6 +76,13 @@ export function finalIds(input: Input): Observable { */ value = value.filter(info => !input.ignoreVersions.test(info.version)) + // Filter out tags that are to be ignored only when including tags is enabled + if (input.includeTags === 'true') { + value = value.filter( + info => !info.tags.some(tag => input.ignoreVersions.test(tag)) + ) + } + if (input.deleteUntaggedVersions === 'true') { value = value.filter(info => !info.tagged) } diff --git a/src/input.ts b/src/input.ts index d05e0690..ef671d48 100644 --- a/src/input.ts +++ b/src/input.ts @@ -9,6 +9,7 @@ export interface InputParams { token?: string deletePreReleaseVersions?: string deleteUntaggedVersions?: string + includeTags?: string } const defaultParams = { @@ -21,7 +22,8 @@ const defaultParams = { ignoreVersions: new RegExp(''), deletePreReleaseVersions: '', token: '', - deleteUntaggedVersions: '' + deleteUntaggedVersions: '', + includeTags: '' } export class Input { @@ -36,6 +38,7 @@ export class Input { token: string numDeleted: number deleteUntaggedVersions: string + includeTags: string constructor(params?: InputParams) { const validatedParams: Required = {...defaultParams, ...params} @@ -51,6 +54,7 @@ export class Input { this.token = validatedParams.token this.numDeleted = 0 this.deleteUntaggedVersions = validatedParams.deleteUntaggedVersions + this.includeTags = validatedParams.includeTags } hasOldestVersionQueryInfo(): boolean { @@ -65,6 +69,7 @@ export class Input { checkInput(): boolean { if (this.packageType.toLowerCase() !== 'container') { this.deleteUntaggedVersions = 'false' + this.includeTags = 'false' } if ( diff --git a/src/main.ts b/src/main.ts index 2e3df1b2..d948dd01 100644 --- a/src/main.ts +++ b/src/main.ts @@ -22,7 +22,8 @@ function getActionInput(): Input { token: getInput('token'), deleteUntaggedVersions: getInput( 'delete-only-untagged-versions' - ).toLowerCase() + ).toLowerCase(), + includeTags: getInput('ignore-versions-include-tags').toLowerCase() }) } diff --git a/src/version/get-versions.ts b/src/version/get-versions.ts index 342cad2b..5ee97f51 100644 --- a/src/version/get-versions.ts +++ b/src/version/get-versions.ts @@ -9,6 +9,7 @@ export interface RestVersionInfo { version: string created_at: string tagged: boolean + tags: string[] } export interface RestQueryInfo { @@ -57,20 +58,22 @@ export function getOldestVersions( map(response => { const resp = { versions: response.data.map((version: GetVersionsResponse[0]) => { - let tagged = false + let tags: string[] = [] + if ( package_type === 'container' && version.metadata && version.metadata.container ) { - tagged = version.metadata.container.tags.length > 0 + tags = version.metadata.container.tags } return { id: version.id, version: version.name, created_at: version.created_at, - tagged + tagged: tags.length > 0, + tags } }), page,