Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Delete versions for multiple packages #96

Open
wants to merge 47 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
47 commits
Select commit Hold shift + click to select a range
f5facbe
feat: create package name filter
OmarCastro Apr 1, 2022
8a81d7a
feat: add query for repo packages
OmarCastro Apr 1, 2022
31e12b6
test: add packages query tests
OmarCastro Apr 1, 2022
4513f31
chore: lint code
OmarCastro Apr 1, 2022
f0a7caf
src: handle packageNames on delete versions
OmarCastro Apr 1, 2022
3f26736
src: add * wildcard to match all packages
OmarCastro Apr 4, 2022
6531663
test: add tests for all wildcard cases
OmarCastro Apr 4, 2022
34301c8
doc: update README.md
OmarCastro Apr 4, 2022
7e81fc4
chore: lint code
OmarCastro Apr 4, 2022
36d9774
doc: add example of comma separated list in README
OmarCastro Apr 4, 2022
f98a76d
review: fix typo
OmarCastro May 31, 2022
3986572
review: remove no longer used package totalCount
OmarCastro May 31, 2022
c307ec0
review: add package-names input to actions.yml
OmarCastro May 31, 2022
03234f8
dist: execute 'npm run pack'
OmarCastro May 31, 2022
fa40cba
fix: fix package graphql query variable name
OmarCastro May 31, 2022
d253172
refactor: move graphql.ts to a common folder
OmarCastro Jun 1, 2022
43b503f
Merge branch 'pr/72' into merge_multiple_packages
casvanluijtelaar Mar 1, 2023
7aa5101
fix merge conflicts
casvanluijtelaar Mar 1, 2023
968ce24
cleanup
casvanluijtelaar Mar 1, 2023
e9cb747
wevrbf
casvanluijtelaar Mar 1, 2023
23d1b6f
Revert "cleanup"
casvanluijtelaar Mar 2, 2023
2a31fbf
include package-names in input check
casvanluijtelaar Mar 8, 2023
e5dd8e6
Merge remote-tracking branch 'upstream/main'
casvanluijtelaar Mar 8, 2023
61ae987
fix input
casvanluijtelaar Mar 8, 2023
7de4077
test
casvanluijtelaar Mar 8, 2023
7510471
test
casvanluijtelaar Mar 8, 2023
d9ea0e3
test
casvanluijtelaar Mar 8, 2023
afa09f0
test
casvanluijtelaar Mar 8, 2023
5108feb
test
casvanluijtelaar Mar 8, 2023
dafc901
test
casvanluijtelaar Mar 8, 2023
9628c4e
test
casvanluijtelaar Mar 8, 2023
5a5dbef
test
casvanluijtelaar Mar 8, 2023
205d786
test
casvanluijtelaar Mar 8, 2023
b7a46fc
test
casvanluijtelaar Mar 8, 2023
5372edd
test
casvanluijtelaar Mar 8, 2023
256fb2f
test
casvanluijtelaar Mar 8, 2023
90050d8
test
casvanluijtelaar Mar 8, 2023
2890194
test
casvanluijtelaar Mar 8, 2023
bde8d5e
test
casvanluijtelaar Mar 8, 2023
f63cefd
test
casvanluijtelaar Mar 8, 2023
df798ef
test
casvanluijtelaar Mar 8, 2023
3befe74
test
casvanluijtelaar Mar 8, 2023
a65d13c
test
casvanluijtelaar Mar 8, 2023
6b2ed21
test
casvanluijtelaar Mar 8, 2023
6e59739
remove unused "after" param in query
casvanluijtelaar Sep 26, 2023
4503ade
build new version
casvanluijtelaar Sep 26, 2023
a8e5458
try default value
casvanluijtelaar Sep 27, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
61 changes: 54 additions & 7 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,8 @@ This action deletes versions of a package from [GitHub Packages](https://github.
* Delete all package versions except n most recent versions
* Delete oldest version(s)
* Ignore version(s) from deletion through regex
* Delete version(s) of a package that is hosted from a repo having access to package
* Delete version(s) of a package that is hosted from a repo not having access to package
* Delete version(s) of packages that are hosted in the same repo that is executing the workflow
* Delete version(s) of packages that are hosted in a different repo than the one executing the workflow
* Delete a single version
* Delete multiple versions
* Delete specific version(s)
Expand All @@ -28,13 +28,24 @@ This action deletes versions of a package from [GitHub Packages](https://github.
# Required if deleting a version from a package hosted in a different org than the one executing the workflow.
owner:

# Name of the package.
# Required
package-name:

# Type of the package. Can be one of container, maven, npm, nuget, or rubygems.
# Required
package-type:

# Defaults to an empty string.
# Required if `package-version-ids` or `package-names` input is not given.
package-name:

# Names of the package.
# Can be one of the following:
# - a single package name
# - a group of packages that matches a wildcard at start, end, both sides, or all packages (e.g. "package*")
# - a group of packages that matches a regex, must start with slash at the beginning and end (e.g. "/package.*/")
# - a comma separated list of the previous cases (e.g. "package-lorem, *-ipsum, /.*dolor/")
# Defaults to an empty string.
# Required if `package-version-ids` or `package-name` input is not given.
package-names:


# The number of old versions to delete starting from the oldest version.
# Defaults to 1.
Expand Down Expand Up @@ -75,7 +86,7 @@ This action deletes versions of a package from [GitHub Packages](https://github.

# Valid Input Combinations

`owner`, `package-name`, `package-type` and `token` can be used with the following combinations in a workflow -
`owner`, `repo`, `package-name` (or `package-names`) and `token` can be used with the following combinations in a workflow -

- `num-old-versions-to-delete`
- `min-versions-to-keep`
Expand Down Expand Up @@ -104,6 +115,7 @@ This action deletes versions of a package from [GitHub Packages](https://github.
- [License](#license)



### Delete all pre-release versions except y latest pre-release package versions

To delete all pre release versions except y latest pre-release package versions, the __package-name__, __min-versions-to-keep__ and __delete-only-pre-release-versions__ inputs are required.
Expand Down Expand Up @@ -272,6 +284,41 @@ This action deletes versions of a package from [GitHub Packages](https://github.

<br>

### Delete all except y latest versions of a package

To delete all except y latest versions of all packages hosted in the same repo as the workflow the __package-names__ and __min-versions-to-keep__ inputs are required.

__Example__

Delete all except latest 2 versions of a package hosted in the same repo as the workflow

```yaml
- uses: actions/delete-package-versions@v3
with:
package-names: '*'
min-versions-to-keep: 2
```

To delete all except y latest versions of all packages hosted in a repo other than the workflow the __owner__, __repo__, __package-names__, __token__ and __min-versions-to-keep__ inputs are required.

The [token][token] needs the delete packages and read packages scope. It is recommended [to store the token as a secret][secret]. In this example the [token][token] was stored as a secret named __GITHUB_PAT__.

__Example__

Delete all except latest 2 versions of a package hosted in a repo other than the workflow

```yaml
- uses: actions/delete-package-versions@v3
with:
owner: 'github'
repo: 'packages'
package-names: '*'
token: ${{ secrets.PAT }}
min-versions-to-keep: 2
```

<br>

### Delete oldest x number of versions of a package

To delete the oldest x number of versions of a package hosted, the __package-name__, and __num-old-versions-to-delete__ inputs are required.
Expand Down
66 changes: 66 additions & 0 deletions __tests__/packages/get-packages.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
import {mockPackagesQueryResponse} from './graphql.mock'
import {
getRepoPackages as _getRepoPackages,
QueryInfo
} from '../../src/packages'
import {Observable} from 'rxjs'

describe.skip('get versions tests -- call graphql', () => {
it('getRepoPackages -- succeeds', done => {
const numPackages = 1
getRepoPackages({numPackages}).subscribe(result => {
expect(result.packages.length).toBe(numPackages)
done()
})
})

it('getRepoPackages -- fails for invalid repo', done => {
getRepoPackages({repo: 'actions-testin'}).subscribe({
error: err => {
expect(err).toBeTruthy()
done()
},
complete: async () => done.fail('no error thrown')
})
})
})

describe('get versions tests -- mock graphql', () => {
it('getRepoPackages -- success', done => {
const numPackages = 5
mockPackagesQueryResponse(numPackages)

getRepoPackages({numPackages}).subscribe(result => {
expect(result.packages.length).toBe(numPackages)
done()
})
})
})

interface Params {
owner?: string
repo?: string
numPackages?: number
startCursor?: string
token?: string
}

const defaultParams = {
owner: 'namratajha',
repo: 'test-repo',
packageName: 'test-repo',
numPackages: 1,
startCursor: '',
token: process.env.GITHUB_TOKEN as string
}

function getRepoPackages(params?: Params): Observable<QueryInfo> {
const p: Required<Params> = {...defaultParams, ...params}
return _getRepoPackages(
p.owner,
p.repo,
p.numPackages,
p.startCursor,
p.token
)
}
44 changes: 44 additions & 0 deletions __tests__/packages/graphql.mock.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import {
GraphQlQueryResponseData,
RequestParameters
} from '@octokit/graphql/dist-types/types'

import * as Graphql from '../../src/common/graphql'
import {GetPackagesQueryResponse} from '../../src/packages'

export function getMockedPackagesQueryResponse(
numPackages: number
): GetPackagesQueryResponse {
const packages: any[] = []
for (let i = 1; i <= numPackages; ++i) {
packages.push({
node: {
id: i.toString(),
name: `package${i}`
}
})
}

return {
repository: {
packages: {
pageInfo: {
endCursor: 'AAA',
hasNextPage: false
},
edges: packages
}
}
}
}

export function mockPackagesQueryResponse(numVersions: number): void {
const response = new Promise<GetPackagesQueryResponse>(resolve => {
resolve(getMockedPackagesQueryResponse(numVersions))
}) as Promise<GraphQlQueryResponseData>
jest
.spyOn(Graphql, 'graphql')
.mockImplementation(
(token: string, query: string, parameters: RequestParameters) => response
)
}
114 changes: 114 additions & 0 deletions __tests__/packages/package-name-filter.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,114 @@
import { getPackageNameFilter } from '../../src/packages'

describe('package name filter -- create filter', () => {

const packageNameList = [
'com.company.project.module1.package1',
'com.company.project.module1.package2',
'com.company.project.module2.package1',
'com.company.project.module2.package2',
'com.company.project.module3.package-name-lorem',
'com.company.project.module3.package-name-ipsum',
'com.company.project.module3.package-name-dolor',
]

it('getPackageNameFilter -- wildcard end filter', done => {
const filter = getPackageNameFilter('com.company.project.module1.*')

const result = packageNameList.filter(filter.apply);

expect(filter.subfilters[0].type).toBe('wildcard')
expect(result).toEqual([
'com.company.project.module1.package1',
'com.company.project.module1.package2',
])
done()
})

it('getPackageNameFilter -- wildcard start filter', done => {
const filter = getPackageNameFilter('*.package1')

const result = packageNameList.filter(filter.apply);

expect(filter.subfilters[0].type).toBe('wildcard')
expect(result).toEqual([
'com.company.project.module1.package1',
'com.company.project.module2.package1',
])
done()
})

it('getPackageNameFilter -- wildcard both sides filter', done => {
const filter = getPackageNameFilter('*.project.module3.*')

const result = packageNameList.filter(filter.apply);

expect(filter.subfilters[0].type).toBe('wildcard')
expect(result).toEqual([
'com.company.project.module3.package-name-lorem',
'com.company.project.module3.package-name-ipsum',
'com.company.project.module3.package-name-dolor'
])
done()
})


it('getPackageNameFilter -- wildcard all filter', done => {
const filter = getPackageNameFilter('*')

const result = packageNameList.filter(filter.apply);

expect(filter.subfilters[0].type).toBe('wildcard')
expect(result).toEqual(packageNameList.slice())
done()
})

it('getPackageNameFilter -- regex filter', done => {
const filter = getPackageNameFilter('/com\\.company\\.project\\.module.*\\.package1/')
const result = packageNameList.filter(filter.apply);

expect(filter.subfilters[0].type).toBe('regex')
expect(result).toEqual([
'com.company.project.module1.package1',
'com.company.project.module2.package1',
])
done()
})

it('getPackageNameFilter -- exact match filter', done => {
const filter = getPackageNameFilter('com.company.project.module1.package1')
const result = packageNameList.filter(filter.apply);

expect(filter.subfilters[0].type).toBe('string')
expect(result).toEqual(['com.company.project.module1.package1'])
done()
})


it('getPackageNameFilter -- multiple filters', done => {
const filter = getPackageNameFilter('com.company.project.module1.package1, com.company.project.module2.*, /.*module3.*-ipsum/')
const result = packageNameList.filter(filter.apply);

expect(filter.subfilters.length).toBe(3)
expect(filter.subfilters[0].type).toBe('string')
expect(filter.subfilters[1].type).toBe('wildcard')
expect(filter.subfilters[2].type).toBe('regex')
expect(result).toEqual([
'com.company.project.module1.package1',
'com.company.project.module2.package1',
'com.company.project.module2.package2',
'com.company.project.module3.package-name-ipsum'
])
done()
})


it('getPackageNameFilter -- memoization, same input shoud return same output', done => {
const filterText = 'com.company.project.module1.package1, com.company.project.module2.*, /.*module3.*-dolor/, *.-lorem'
const filter1 = getPackageNameFilter(filterText)
const filter2 = getPackageNameFilter(filterText)
expect(filter1).toBe(filter2)
expect(filter1.subfilters.length).toBe(4)
done()
})
})
13 changes: 13 additions & 0 deletions action.yml
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,19 @@ inputs:
Type of package. Can be one of container, maven, npm, nuget, or rubygems.
required: true


package-names:
description: >
Names of the package.
Can be one of the following:
- a single package name
- a group of packages that matches a wildcard at start, end, both sides, or all packages (e.g. "package*")
- a group of packages that matches a regex, must start with slash at the beginning and end (e.g. "/package.*/")
- a comma separated list of the previous cases (e.g. "package-lorem, *-ipsum, /.*dolor/")
Defaults to an empty string.
Required if `package-version-ids` or `package-name` input is not given.
required: false

num-old-versions-to-delete:
description: >
Number of versions to delete starting with the oldest version.
Expand Down
Loading