From eb5db5b492dae1b4e41e78fe58c661b596ae020a Mon Sep 17 00:00:00 2001 From: Shawn Smith <chezsmithy@me.com> Date: Sun, 5 Mar 2023 10:29:06 -0800 Subject: [PATCH] fix(datasource/docker): Artifactory next link is broken for tags api (#20745) Co-authored-by: Rhys Arkins <rhys@arkins.net> Co-authored-by: Michael Kriese <michael.kriese@visualon.de> --- lib/modules/datasource/docker/common.ts | 4 +- lib/modules/datasource/docker/index.spec.ts | 66 +++++++++++++-------- lib/modules/datasource/docker/index.ts | 10 +++- 3 files changed, 52 insertions(+), 28 deletions(-) diff --git a/lib/modules/datasource/docker/common.ts b/lib/modules/datasource/docker/common.ts index f91b85c1e2..467755d977 100644 --- a/lib/modules/datasource/docker/common.ts +++ b/lib/modules/datasource/docker/common.ts @@ -10,6 +10,8 @@ export const gitRefLabel = 'org.opencontainers.image.revision'; const JFROG_ARTIFACTORY_RES_HEADER = 'x-jfrog-version'; -export function isArtifactoryServer(res: HttpResponse | undefined): boolean { +export function isArtifactoryServer<T = unknown>( + res: HttpResponse<T> | undefined +): boolean { return is.string(res?.headers[JFROG_ARTIFACTORY_RES_HEADER]); } diff --git a/lib/modules/datasource/docker/index.spec.ts b/lib/modules/datasource/docker/index.spec.ts index e19389b4bb..321f65932f 100644 --- a/lib/modules/datasource/docker/index.spec.ts +++ b/lib/modules/datasource/docker/index.spec.ts @@ -5,6 +5,7 @@ import { } from '@aws-sdk/client-ecr'; import { mockClient } from 'aws-sdk-client-mock'; import { getDigest, getPkgReleases } from '..'; +import { range } from '../../../../lib/util/range'; import * as httpMock from '../../../../test/http-mock'; import { logger, mocked } from '../../../../test/util'; import { @@ -1193,32 +1194,45 @@ describe('modules/datasource/docker/index', () => { await expect(getPkgReleases(config)).rejects.toThrow(EXTERNAL_HOST_ERROR); }); - it.each([[true], [false]])( - 'jfrog artifactory - retry tags for official images by injecting `/library` after repository and before image, abortOnError=%p', - async (abortOnError) => { - hostRules.find.mockReturnValue({ abortOnError }); - const tags = ['18.0.0']; - httpMock - .scope('https://org.jfrog.io/v2') - .get('/virtual-mirror/node/tags/list?n=10000') - .reply(200, '', {}) - .get('/virtual-mirror/node/tags/list?n=10000') - .reply(404, '', { 'x-jfrog-version': 'Artifactory/7.42.2 74202900' }) - .get('/virtual-mirror/library/node/tags/list?n=10000') - .reply(200, '', {}) - .get('/virtual-mirror/library/node/tags/list?n=10000') - .reply(200, { tags }, {}) - .get('/') - .reply(200, '', {}) - .get('/virtual-mirror/node/manifests/18.0.0') - .reply(200, '', {}); - const res = await getPkgReleases({ - datasource: DockerDatasource.id, - depName: 'org.jfrog.io/virtual-mirror/node', - }); - expect(res?.releases).toHaveLength(1); - } - ); + it('jfrog artifactory - retry tags for official images by injecting `/library` after repository and before image', async () => { + const tags1 = [...range(1, 10000)].map((i) => `${i}.0.0`); + const tags2 = [...range(10000, 10050)].map((i) => `${i}.0.0`); + httpMock + .scope('https://org.jfrog.io/v2') + .get('/virtual-mirror/node/tags/list?n=10000') + .reply(200, '', { 'x-jfrog-version': 'Artifactory/7.42.2 74202900' }) + .get('/virtual-mirror/node/tags/list?n=10000') + .reply(404, '', { 'x-jfrog-version': 'Artifactory/7.42.2 74202900' }) + .get('/virtual-mirror/library/node/tags/list?n=10000') + .reply(200, '', {}) + .get('/virtual-mirror/library/node/tags/list?n=10000') + // Note the Link is incorrect and should be `</virtual-mirror/library/node/tags/list?n=10000&last=10000>; rel="next", ` + // Artifactory incorrectly returns a next link without the virtual repository name + // this is due to a bug in Artifactory https://jfrog.atlassian.net/browse/RTFACT-18971 + .reply( + 200, + { tags: tags1 }, + { + 'x-jfrog-version': 'Artifactory/7.42.2 74202900', + link: '</library/node/tags/list?n=10000&last=10000>; rel="next", ', + } + ) + .get('/virtual-mirror/library/node/tags/list?n=10000&last=10000') + .reply( + 200, + { tags: tags2 }, + { 'x-jfrog-version': 'Artifactory/7.42.2 74202900' } + ) + .get('/') + .reply(200, '', {}) + .get('/virtual-mirror/node/manifests/10050.0.0') + .reply(200, '', {}); + const res = await getPkgReleases({ + datasource: DockerDatasource.id, + depName: 'org.jfrog.io/virtual-mirror/node', + }); + expect(res?.releases).toHaveLength(10050); + }); it('uses lower tag limit for ECR deps', async () => { httpMock diff --git a/lib/modules/datasource/docker/index.ts b/lib/modules/datasource/docker/index.ts index 83bcd6dabf..c2192fec8d 100644 --- a/lib/modules/datasource/docker/index.ts +++ b/lib/modules/datasource/docker/index.ts @@ -883,7 +883,15 @@ export class DockerDatasource extends Datasource { } tags = tags.concat(res.body.tags); const linkHeader = parseLinkHeader(res.headers.link); - url = linkHeader?.next ? URL.resolve(url, linkHeader.next.url) : null; + if (isArtifactoryServer(res)) { + // Artifactory incorrectly returns a next link without the virtual repository name + // this is due to a bug in Artifactory https://jfrog.atlassian.net/browse/RTFACT-18971 + url = linkHeader?.next?.last + ? `${url}&last=${linkHeader.next.last}` + : null; + } else { + url = linkHeader?.next ? URL.resolve(url, linkHeader.next.url) : null; + } page += 1; } while (url && page < 20); return tags; -- GitLab