From de68c8249a7fe61ccabc5dc55bc0371726882526 Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 15:20:51 -0500 Subject: [PATCH 1/8] Add support to `--no-prune` --- .../docker/tasks/image/DockerRemoveImage.java | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java index 14e323c14..d93d96062 100644 --- a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java +++ b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java @@ -22,13 +22,26 @@ public class DockerRemoveImage extends DockerExistingImage { + /** + * Configures `--force` option when removing the images. + */ @Input @Optional public final Property getForce() { return force; } + /** + * Configures `--no-prune` option when removing the images. + */ + @Input + @Optional + public final Property getNoPrune() { + return noPrune; + } + private final Property force = getProject().getObjects().property(Boolean.class); + private final Property noPrune = getProject().getObjects().property(Boolean.class); @Override public void runRemoteCommand() { From bba702a469abbcf99b44feb2dbb5fed2b777ad1b Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 15:21:09 -0500 Subject: [PATCH 2/8] Remove unused imports --- .../gradle/docker/internal/DefaultDockerConfigResolver.java | 3 --- 1 file changed, 3 deletions(-) diff --git a/src/main/java/com/bmuschko/gradle/docker/internal/DefaultDockerConfigResolver.java b/src/main/java/com/bmuschko/gradle/docker/internal/DefaultDockerConfigResolver.java index 16dd8d647..ad6eee01a 100644 --- a/src/main/java/com/bmuschko/gradle/docker/internal/DefaultDockerConfigResolver.java +++ b/src/main/java/com/bmuschko/gradle/docker/internal/DefaultDockerConfigResolver.java @@ -1,8 +1,5 @@ package com.bmuschko.gradle.docker.internal; -import org.gradle.api.logging.Logger; -import org.gradle.api.logging.Logging; - import javax.annotation.Nullable; import java.io.File; From c029597c263ba6285252fb5cdd2fd00b7f29fc1b Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 15:21:30 -0500 Subject: [PATCH 3/8] Add support to list multiple images to remove --- .../DockerRemoveImageFunctionalTest.groovy | 79 ++++++++++- .../docker/tasks/image/DockerRemoveImage.java | 125 +++++++++++++++++- 2 files changed, 194 insertions(+), 10 deletions(-) diff --git a/src/functTest/groovy/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImageFunctionalTest.groovy b/src/functTest/groovy/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImageFunctionalTest.groovy index 65b4c49de..35cbee250 100644 --- a/src/functTest/groovy/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImageFunctionalTest.groovy +++ b/src/functTest/groovy/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImageFunctionalTest.groovy @@ -21,13 +21,13 @@ class DockerRemoveImageFunctionalTest extends AbstractGroovyDslFunctionalTest { dependsOn dockerfile inputDir = file("build/docker") } - + task removeImage(type: DockerRemoveImage) { dependsOn buildImage force = true targetImageId buildImage.getImageId() } - + task removeImageAndCheckRemoval(type: DockerListImages) { dependsOn removeImage showAll = true @@ -42,6 +42,75 @@ class DockerRemoveImageFunctionalTest extends AbstractGroovyDslFunctionalTest { !result.output.contains("repository") } + def "can remove multiple images"() { + // Given a few Official Images that are small... :-) + String firstImage = "alpine:3" + String secondImage = "busybox:1.36" + String thirdImage = "bash:5.2.21" + + // And a build script that uses them... + buildFile << """ + import com.bmuschko.gradle.docker.tasks.image.Dockerfile + import com.bmuschko.gradle.docker.tasks.image.DockerBuildImage + import com.bmuschko.gradle.docker.tasks.image.DockerListImages + import com.bmuschko.gradle.docker.tasks.image.DockerRemoveImage + + task firstDockerfile(type: Dockerfile) { + from '$firstImage' + label(['maintainer': 'jane.doe@example.com']) + } + task buildFirstImage(type: DockerBuildImage) { + dependsOn firstDockerfile + inputDir = file("build/docker") + } + + task secondDockerfile(type: Dockerfile) { + from '$secondImage' + label(['maintainer': 'jane.doe@example.com']) + } + task buildSecondImage(type: DockerBuildImage) { + dependsOn secondDockerfile + inputDir = file("build/docker") + } + + task thirdDockerfile(type: Dockerfile) { + from '$thirdImage' + label(['maintainer': 'jane.doe@example.com']) + } + task buildThirdImage(type: DockerBuildImage) { + dependsOn thirdDockerfile + inputDir = file("build/docker") + } + + task removeImages(type: DockerRemoveImage) { + dependsOn buildFirstImage + dependsOn buildSecondImage + dependsOn buildThirdImage + + force = true + + images( + buildFirstImage.getImageId(), + buildSecondImage.getImageId(), + buildThirdImage.getImageId() + ) + } + + task removeImageAndCheckRemoval(type: DockerListImages) { + dependsOn removeImages + showAll = true + dangling = true + } + """ + + when: + BuildResult result = build('removeImageAndCheckRemoval') + + then: + !result.output.contains("repository") + } + + def "can remove image tagged in multiple repositories"() { buildFile << """ import com.bmuschko.gradle.docker.tasks.image.Dockerfile @@ -59,21 +128,21 @@ class DockerRemoveImageFunctionalTest extends AbstractGroovyDslFunctionalTest { dependsOn dockerfile inputDir = file("build/docker") } - + task tagImage(type: DockerTagImage) { dependsOn buildImage repository = "repository" tag = "tag2" targetImageId buildImage.getImageId() } - + task tagImageSecondTime(type: DockerTagImage) { dependsOn tagImage repository = "repository" tag = "tag2" targetImageId buildImage.getImageId() } - + task removeImage(type: DockerRemoveImage) { dependsOn tagImageSecondTime force = true diff --git a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java index d93d96062..908dbc583 100644 --- a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java +++ b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java @@ -16,10 +16,16 @@ package com.bmuschko.gradle.docker.tasks.image; import com.github.dockerjava.api.command.RemoveImageCmd; +import org.gradle.api.provider.ListProperty; import org.gradle.api.provider.Property; +import org.gradle.api.provider.Provider; import org.gradle.api.tasks.Input; import org.gradle.api.tasks.Optional; +import java.util.Arrays; +import java.util.List; +import java.util.concurrent.Callable; + public class DockerRemoveImage extends DockerExistingImage { /** @@ -43,15 +49,124 @@ public final Property getNoPrune() { private final Property force = getProject().getObjects().property(Boolean.class); private final Property noPrune = getProject().getObjects().property(Boolean.class); + private final ListProperty images = getProject().getObjects().listProperty(String.class); + + /** + * Sets the IDs or names of the images to be removed. + * + * @param images the names or IDs of the images. + * @see #images(ListProperty) + */ + public void images(List images) { + this.images.set(images); + } + + /** + * Sets the IDs or names of the images to be removed. + * + * @param images the names or IDs of the images. + * @see #images(ListProperty) + */ + public void images(String... images) { + this.images.set(Arrays.asList(images)); + } + + /** + * Sets the IDs or names of the images to be removed. + * + * @param images Image ID or name as {@link Callable} + * @see #images(String...) + * @see #images(ListProperty) + */ + public void images(Callable> images) { + // TODO: How to do this, and is it necessary? + // images(providers.provider(images)); + } + + /** + * Sets the IDs or names of the images to be removed. + * + * @param images Image ID or name as {@link Provider} + * @see #images(String...) + */ + public void images(ListProperty images) { + this.images.set(images); + } + + // Overriding methods to provide a warning message. + /** + * Sets the target image ID or name. + * + * @param imageId Image ID or name + * @see #targetImageId(Callable) + * @see #targetImageId(Provider) + */ + @Override + public void targetImageId(String imageId) { + logWarning(); + super.targetImageId(imageId); + } + + /** + * Sets the target image ID or name. + * + * @param imageId Image ID or name as Callable + * @see #targetImageId(String) + * @see #targetImageId(Provider) + */ + @Override + public void targetImageId(Callable imageId) { + logWarning(); + super.targetImageId(imageId); + } + + /** + * Sets the target image ID or name. + * + * @param imageId Image ID or name as Provider + * @see #targetImageId(String) + * @see #targetImageId(Callable) + */ + @Override + public void targetImageId(Provider imageId) { + logWarning(); + super.targetImageId(imageId); + } + + private void logWarning() { + getLogger().warn("Use property 'images' instead of 'targetImageId' when removing images"); + } + @Override public void runRemoteCommand() { - getLogger().quiet("Removing image with ID \'" + getImageId().get() + "\'."); - RemoveImageCmd removeImageCmd = getDockerClient().removeImageCmd(getImageId().get()); + if (mixesBothProperties()) { + throw new IllegalStateException("Project sets both properties 'images' and 'targetImageId', but only one is allowed."); + } + + if (this.getImageId().isPresent()) { + removeImage(this.getImageId().get()); + } - if (Boolean.TRUE.equals(force.getOrNull())) { - removeImageCmd.withForce(force.get()); + if (this.images.isPresent()) { + this.images.get().forEach(this::removeImage); } + } + + private void removeImage(String imageId) { + getLogger().quiet("Removing image with ID \'" + imageId + "\'."); + try (RemoveImageCmd removeImageCmd = getDockerClient().removeImageCmd(imageId)) { + if (Boolean.TRUE.equals(force.getOrNull())) { + removeImageCmd.withForce(force.get()); + } + + if (Boolean.TRUE.equals(noPrune.getOrNull())) { + removeImageCmd.withNoPrune(noPrune.get()); + } + removeImageCmd.exec(); + } + } - removeImageCmd.exec(); + private boolean mixesBothProperties() { + return this.images.isPresent() && this.getImageId().isPresent(); } } From aa9ede15c67b58c156c72f9c06ddb3dd605ae281 Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 15:31:51 -0500 Subject: [PATCH 4/8] ci: Store reports for failed functionalTests --- .github/workflows/linux-build-release.yml | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/.github/workflows/linux-build-release.yml b/.github/workflows/linux-build-release.yml index c41f1466b..43859da63 100644 --- a/.github/workflows/linux-build-release.yml +++ b/.github/workflows/linux-build-release.yml @@ -36,6 +36,11 @@ jobs: HARBOR_USERNAME: ${{ secrets.HARBOR_USERNAME }} HARBOR_PASSWORD: ${{ secrets.HARBOR_PASSWORD }} run: ./gradlew functionalTest + - name: Store build reports for functionalTest + uses: actions/upload-artifact@v3 + with: + name: functionalTest-reports + path: build/reports/tests/functionalTest/ - name: Documentation tests run: ./gradlew docTest - name: Assemble artifacts From 8bfdf04af6c2c747b388fbc0279d1ab0b6811bb3 Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 15:37:45 -0500 Subject: [PATCH 5/8] Store test reports only in case of failure --- .github/workflows/linux-build-release.yml | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/.github/workflows/linux-build-release.yml b/.github/workflows/linux-build-release.yml index 43859da63..6faa77891 100644 --- a/.github/workflows/linux-build-release.yml +++ b/.github/workflows/linux-build-release.yml @@ -36,13 +36,14 @@ jobs: HARBOR_USERNAME: ${{ secrets.HARBOR_USERNAME }} HARBOR_PASSWORD: ${{ secrets.HARBOR_PASSWORD }} run: ./gradlew functionalTest - - name: Store build reports for functionalTest - uses: actions/upload-artifact@v3 - with: - name: functionalTest-reports - path: build/reports/tests/functionalTest/ - name: Documentation tests run: ./gradlew docTest + - name: Store Test Reports + if: ${{ failure() }} + uses: actions/upload-artifact@v3 + with: + name: test-reports + path: build/reports/tests - name: Assemble artifacts run: ./gradlew assemble javadoc asciidoctorAllGuides - name: Upload binaries From 2d53e7770980995f4e28de01658b73f3cc72f47a Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 15:44:04 -0500 Subject: [PATCH 6/8] Fix checking if both properties are used --- .../bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java index 908dbc583..8ba1956ab 100644 --- a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java +++ b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java @@ -167,6 +167,6 @@ private void removeImage(String imageId) { } private boolean mixesBothProperties() { - return this.images.isPresent() && this.getImageId().isPresent(); + return !this.images.get().isEmpty() && java.util.Optional.ofNullable(this.getImageId().getOrNull()).isPresent(); } } From de98052486232d0cca2f619d06eb0e242aee3265 Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 15:45:21 -0500 Subject: [PATCH 7/8] Also upload test repors for windows --- .github/workflows/linux-build-release.yml | 2 +- .github/workflows/windows-build.yml | 6 ++++++ 2 files changed, 7 insertions(+), 1 deletion(-) diff --git a/.github/workflows/linux-build-release.yml b/.github/workflows/linux-build-release.yml index 6faa77891..acb50b589 100644 --- a/.github/workflows/linux-build-release.yml +++ b/.github/workflows/linux-build-release.yml @@ -38,7 +38,7 @@ jobs: run: ./gradlew functionalTest - name: Documentation tests run: ./gradlew docTest - - name: Store Test Reports + - name: Upload test reports if: ${{ failure() }} uses: actions/upload-artifact@v3 with: diff --git a/.github/workflows/windows-build.yml b/.github/workflows/windows-build.yml index 657e39d05..395e27d66 100644 --- a/.github/workflows/windows-build.yml +++ b/.github/workflows/windows-build.yml @@ -23,5 +23,11 @@ jobs: run: ./gradlew test - name: Integration tests run: ./gradlew integrationTest + - name: Upload test reports + if: ${{ failure() }} + uses: actions/upload-artifact@v3 + with: + name: test-reports + path: build/reports/tests - name: Assemble artifacts run: ./gradlew assemble javadoc asciidoctorAllGuides From e9d923e63675bbe859051ec3d7c7baf3d7e1fe61 Mon Sep 17 00:00:00 2001 From: Marcos Pereira Date: Thu, 30 Nov 2023 16:00:27 -0500 Subject: [PATCH 8/8] Small refactoring --- .../docker/tasks/image/DockerRemoveImage.java | 22 +++++++------------ 1 file changed, 8 insertions(+), 14 deletions(-) diff --git a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java index 8ba1956ab..e940d2298 100644 --- a/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java +++ b/src/main/java/com/bmuschko/gradle/docker/tasks/image/DockerRemoveImage.java @@ -134,26 +134,24 @@ public void targetImageId(Provider imageId) { } private void logWarning() { - getLogger().warn("Use property 'images' instead of 'targetImageId' when removing images"); + getLogger().warn("Use property 'images' instead of 'targetImageId' when listing images to remove"); } @Override public void runRemoteCommand() { - if (mixesBothProperties()) { - throw new IllegalStateException("Project sets both properties 'images' and 'targetImageId', but only one is allowed."); - } + java.util.Optional imageId = java.util.Optional.ofNullable(this.getImageId().getOrNull()); + List imagesIds = this.images.get(); - if (this.getImageId().isPresent()) { - removeImage(this.getImageId().get()); + if (imageId.isPresent() && !imagesIds.isEmpty()) { + throw new IllegalStateException("Project sets both properties 'images' and 'targetImageId', but only one is allowed. Please use 'images'."); } - if (this.images.isPresent()) { - this.images.get().forEach(this::removeImage); - } + imageId.ifPresent(this::removeImage); + imagesIds.forEach(this::removeImage); } private void removeImage(String imageId) { - getLogger().quiet("Removing image with ID \'" + imageId + "\'."); + getLogger().quiet("Removing image with ID '{}'.", imageId); try (RemoveImageCmd removeImageCmd = getDockerClient().removeImageCmd(imageId)) { if (Boolean.TRUE.equals(force.getOrNull())) { removeImageCmd.withForce(force.get()); @@ -165,8 +163,4 @@ private void removeImage(String imageId) { removeImageCmd.exec(); } } - - private boolean mixesBothProperties() { - return !this.images.get().isEmpty() && java.util.Optional.ofNullable(this.getImageId().getOrNull()).isPresent(); - } }