From 5f2db3b6d74b54038d1d977c791c69c4db702c98 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Galder=20Zamarren=CC=83o?= Date: Thu, 16 Sep 2021 11:34:04 +0200 Subject: [PATCH] Add image resize support via AWT extension #19789 --- .github/native-tests.json | 8 +- bom/application/pom.xml | 10 ++ .../runtime/graal/Java2DSubstitutions.java | 20 ---- devtools/bom-descriptor-json/pom.xml | 13 +++ docs/pom.xml | 13 +++ extensions/awt/deployment/pom.xml | 43 +++++++++ .../quarkus/awt/deployment/AwtProcessor.java | 75 +++++++++++++++ extensions/awt/pom.xml | 17 ++++ extensions/awt/runtime/pom.xml | 44 +++++++++ .../quarkus/awt/runtime/graal/AwtFeature.java | 22 +++++ .../resources/META-INF/quarkus-extension.yaml | 12 +++ extensions/pom.xml | 2 + integration-tests/awt/pom.xml | 93 +++++++++++++++++++ .../io/quarkus/awt/it/AwtImageResizeIT.java | 7 ++ .../io/quarkus/awt/it/AwtImageResizeTest.java | 51 ++++++++++ integration-tests/pom.xml | 1 + 16 files changed, 410 insertions(+), 21 deletions(-) delete mode 100644 core/runtime/src/main/java/io/quarkus/runtime/graal/Java2DSubstitutions.java create mode 100644 extensions/awt/deployment/pom.xml create mode 100644 extensions/awt/deployment/src/main/java/io/quarkus/awt/deployment/AwtProcessor.java create mode 100644 extensions/awt/pom.xml create mode 100644 extensions/awt/runtime/pom.xml create mode 100644 extensions/awt/runtime/src/main/java/io/quarkus/awt/runtime/graal/AwtFeature.java create mode 100644 extensions/awt/runtime/src/main/resources/META-INF/quarkus-extension.yaml create mode 100644 integration-tests/awt/pom.xml create mode 100644 integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeIT.java create mode 100644 integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeTest.java diff --git a/.github/native-tests.json b/.github/native-tests.json index f738e83fdfe7b9..1a803bf484efc3 100644 --- a/.github/native-tests.json +++ b/.github/native-tests.json @@ -143,6 +143,12 @@ "timeout": 70, "test-modules": "devtools-registry-client", "os-name": "ubuntu-latest" - } + }, + { + "category": "AWT, ImageIO and Java2D", + "timeout": 20, + "test-modules": "awt", + "os-name": "ubuntu-latest" + } ] } diff --git a/bom/application/pom.xml b/bom/application/pom.xml index 6afb2e823a9771..cd78818cb54e0c 100644 --- a/bom/application/pom.xml +++ b/bom/application/pom.xml @@ -5414,6 +5414,16 @@ quarkus-vertx-web-deployment ${project.version} + + io.quarkus + quarkus-awt + ${project.version} + + + io.quarkus + quarkus-awt-deployment + ${project.version} + diff --git a/core/runtime/src/main/java/io/quarkus/runtime/graal/Java2DSubstitutions.java b/core/runtime/src/main/java/io/quarkus/runtime/graal/Java2DSubstitutions.java deleted file mode 100644 index 0ca2b30ab278e0..00000000000000 --- a/core/runtime/src/main/java/io/quarkus/runtime/graal/Java2DSubstitutions.java +++ /dev/null @@ -1,20 +0,0 @@ -package io.quarkus.runtime.graal; - -import java.awt.Graphics; -import java.awt.GraphicsEnvironment; - -import com.oracle.svm.core.annotate.AlwaysInline; -import com.oracle.svm.core.annotate.Substitute; -import com.oracle.svm.core.annotate.TargetClass; - -@TargetClass(GraphicsEnvironment.class) -final class Target_java_awt_GraphicsEnvironment { - @AlwaysInline("DCE for things using Java2D") - @Substitute - public static Graphics getLocalGraphicsEnvironment() { - throw new UnsupportedOperationException("Not implemented yet for GraalVM native images"); - } -} - -class Java2DSubstitutions { -} diff --git a/devtools/bom-descriptor-json/pom.xml b/devtools/bom-descriptor-json/pom.xml index 75612efc9853cc..f919a05eaaed70 100644 --- a/devtools/bom-descriptor-json/pom.xml +++ b/devtools/bom-descriptor-json/pom.xml @@ -383,6 +383,19 @@ + + io.quarkus + quarkus-awt + ${project.version} + pom + test + + + * + * + + + io.quarkus quarkus-azure-functions-http diff --git a/docs/pom.xml b/docs/pom.xml index 610684e926b421..e505d8a052b3fd 100644 --- a/docs/pom.xml +++ b/docs/pom.xml @@ -344,6 +344,19 @@ + + io.quarkus + quarkus-awt-deployment + ${project.version} + pom + test + + + * + * + + + io.quarkus quarkus-azure-functions-http-deployment diff --git a/extensions/awt/deployment/pom.xml b/extensions/awt/deployment/pom.xml new file mode 100644 index 00000000000000..a31abf40b9e4a9 --- /dev/null +++ b/extensions/awt/deployment/pom.xml @@ -0,0 +1,43 @@ + + + 4.0.0 + + io.quarkus + quarkus-awt-parent + 999-SNAPSHOT + + quarkus-awt-deployment + Quarkus - Awt - Deployment + + + io.quarkus + quarkus-arc-deployment + + + io.quarkus + quarkus-awt + + + io.quarkus + quarkus-junit5-internal + test + + + + + + maven-compiler-plugin + + + + io.quarkus + quarkus-extension-processor + ${project.version} + + + + + + + diff --git a/extensions/awt/deployment/src/main/java/io/quarkus/awt/deployment/AwtProcessor.java b/extensions/awt/deployment/src/main/java/io/quarkus/awt/deployment/AwtProcessor.java new file mode 100644 index 00000000000000..f41aadc2a79169 --- /dev/null +++ b/extensions/awt/deployment/src/main/java/io/quarkus/awt/deployment/AwtProcessor.java @@ -0,0 +1,75 @@ +package io.quarkus.awt.deployment; + +import io.quarkus.deployment.annotations.BuildStep; +import io.quarkus.deployment.builditem.FeatureBuildItem; +import io.quarkus.deployment.builditem.nativeimage.JniRuntimeAccessBuildItem; +import io.quarkus.deployment.builditem.nativeimage.ReflectiveClassBuildItem; + +class AwtProcessor { + private static final String FEATURE = "awt"; + + @BuildStep + ReflectiveClassBuildItem setupReflectionClasses() { + return new ReflectiveClassBuildItem(false, false, + "sun.awt.X11GraphicsEnvironment", + "sun.awt.X11.XToolkit"); + } + + @BuildStep + JniRuntimeAccessBuildItem setupJava2DClasses() { + return new JniRuntimeAccessBuildItem(true, true, true, + "java.awt.AlphaComposite", + "java.awt.Color", + "java.awt.geom.AffineTransform", + "java.awt.geom.Path2D", + "java.awt.geom.Path2D$Float", + "java.awt.image.BufferedImage", + "java.awt.image.ColorModel", + "java.awt.image.IndexColorModel", + "java.awt.image.Raster", + "java.awt.image.SampleModel", + "java.awt.image.SinglePixelPackedSampleModel", + "sun.awt.SunHints", + "sun.awt.image.BufImgSurfaceData$ICMColorData", + "sun.awt.image.ByteComponentRaster", + "sun.awt.image.BytePackedRaster", + "sun.awt.image.ImageRepresentation", + "sun.awt.image.IntegerComponentRaster", + "sun.java2d.Disposer", + "sun.java2d.InvalidPipeException", + "sun.java2d.NullSurfaceData", + "sun.java2d.SunGraphics2D", + "sun.java2d.SurfaceData", + "sun.java2d.loops.Blit", + "sun.java2d.loops.BlitBg", + "sun.java2d.loops.CompositeType", + "sun.java2d.loops.DrawGlyphList", + "sun.java2d.loops.DrawGlyphListAA", + "sun.java2d.loops.DrawGlyphListLCD", + "sun.java2d.loops.DrawLine", + "sun.java2d.loops.DrawParallelogram", + "sun.java2d.loops.DrawPath", + "sun.java2d.loops.DrawPolygons", + "sun.java2d.loops.DrawRect", + "sun.java2d.loops.FillParallelogram", + "sun.java2d.loops.FillPath", + "sun.java2d.loops.FillRect", + "sun.java2d.loops.FillSpans", + "sun.java2d.loops.GraphicsPrimitive", + "sun.java2d.loops.GraphicsPrimitiveMgr", + "sun.java2d.loops.GraphicsPrimitive[]", + "sun.java2d.loops.MaskBlit", + "sun.java2d.loops.MaskFill", + "sun.java2d.loops.ScaledBlit", + "sun.java2d.loops.SurfaceType", + "sun.java2d.loops.TransformHelper", + "sun.java2d.loops.XORComposite", + "sun.java2d.pipe.Region", + "sun.java2d.pipe.RegionIterator"); + } + + @BuildStep + FeatureBuildItem feature() { + return new FeatureBuildItem(FEATURE); + } +} diff --git a/extensions/awt/pom.xml b/extensions/awt/pom.xml new file mode 100644 index 00000000000000..e28ae5059bd3fc --- /dev/null +++ b/extensions/awt/pom.xml @@ -0,0 +1,17 @@ + + + 4.0.0 + + io.quarkus + quarkus-extensions-parent + 999-SNAPSHOT + + quarkus-awt-parent + pom + Quarkus - Awt - Parent + + deployment + runtime + + diff --git a/extensions/awt/runtime/pom.xml b/extensions/awt/runtime/pom.xml new file mode 100644 index 00000000000000..e4ce9c74395b26 --- /dev/null +++ b/extensions/awt/runtime/pom.xml @@ -0,0 +1,44 @@ + + + 4.0.0 + + io.quarkus + quarkus-awt-parent + 999-SNAPSHOT + + quarkus-awt + Quarkus - Awt - Runtime + Enable AWT and Java2D usage + + + io.quarkus + quarkus-arc + + + org.graalvm.nativeimage + svm + provided + + + + + + io.quarkus + quarkus-bootstrap-maven-plugin + + + maven-compiler-plugin + + + + io.quarkus + quarkus-extension-processor + ${project.version} + + + + + + + diff --git a/extensions/awt/runtime/src/main/java/io/quarkus/awt/runtime/graal/AwtFeature.java b/extensions/awt/runtime/src/main/java/io/quarkus/awt/runtime/graal/AwtFeature.java new file mode 100644 index 00000000000000..bb9dc8b4fdf4df --- /dev/null +++ b/extensions/awt/runtime/src/main/java/io/quarkus/awt/runtime/graal/AwtFeature.java @@ -0,0 +1,22 @@ +package io.quarkus.awt.runtime.graal; + +import org.graalvm.nativeimage.ImageSingletons; +import org.graalvm.nativeimage.hosted.Feature; +import org.graalvm.nativeimage.impl.RuntimeClassInitializationSupport; + +import com.oracle.svm.core.annotate.AutomaticFeature; + +@AutomaticFeature +public class AwtFeature implements Feature { + @Override + public void afterRegistration(AfterRegistrationAccess access) { + final RuntimeClassInitializationSupport runtimeInit = ImageSingletons.lookup(RuntimeClassInitializationSupport.class); + final String reason = "Quarkus run time init for AWT"; + runtimeInit.initializeAtRunTime("com.sun.imageio", reason); + runtimeInit.initializeAtRunTime("java.awt", reason); + runtimeInit.initializeAtRunTime("javax.imageio", reason); + runtimeInit.initializeAtRunTime("sun.awt", reason); + runtimeInit.initializeAtRunTime("sun.font", reason); + runtimeInit.initializeAtRunTime("sun.java2d", reason); + } +} diff --git a/extensions/awt/runtime/src/main/resources/META-INF/quarkus-extension.yaml b/extensions/awt/runtime/src/main/resources/META-INF/quarkus-extension.yaml new file mode 100644 index 00000000000000..cebd0192408062 --- /dev/null +++ b/extensions/awt/runtime/src/main/resources/META-INF/quarkus-extension.yaml @@ -0,0 +1,12 @@ +name: AWT +metadata: + keywords: + - awt + - font + - java2d + - image + - imageio +# guide: ... + categories: + - "miscellaneous" + status: "preview" \ No newline at end of file diff --git a/extensions/pom.xml b/extensions/pom.xml index 86697504fe80d9..3ee41c6ef7cbd7 100644 --- a/extensions/pom.xml +++ b/extensions/pom.xml @@ -201,6 +201,8 @@ vertx-web + + awt diff --git a/integration-tests/awt/pom.xml b/integration-tests/awt/pom.xml new file mode 100644 index 00000000000000..2f29b6b7716e47 --- /dev/null +++ b/integration-tests/awt/pom.xml @@ -0,0 +1,93 @@ + + + 4.0.0 + + io.quarkus + quarkus-integration-tests-parent + 999-SNAPSHOT + + quarkus-awt-integration-tests + Quarkus - Awt - Integration Tests + + + + io.quarkus + quarkus-awt-deployment + ${project.version} + pom + test + + + * + * + + + + + io.quarkus + quarkus-awt + + + io.quarkus + quarkus-junit5 + test + + + + + + io.quarkus + quarkus-maven-plugin + + + + build + + + + + + + + + native-image + + + native + + + + + + maven-surefire-plugin + + ${native.surefire.skip} + + + + maven-failsafe-plugin + + + + integration-test + verify + + + + ${project.build.directory}/${project.build.finalName}-runner + org.jboss.logmanager.LogManager + ${maven.home} + + + + + + + + + native + + + + diff --git a/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeIT.java b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeIT.java new file mode 100644 index 00000000000000..9c39a3d3809938 --- /dev/null +++ b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeIT.java @@ -0,0 +1,7 @@ +package io.quarkus.awt.it; + +import io.quarkus.test.junit.main.QuarkusMainIntegrationTest; + +@QuarkusMainIntegrationTest +public class AwtImageResizeIT extends AwtImageResizeTest { +} diff --git a/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeTest.java b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeTest.java new file mode 100644 index 00000000000000..70f2f500e4888a --- /dev/null +++ b/integration-tests/awt/src/test/java/io/quarkus/awt/it/AwtImageResizeTest.java @@ -0,0 +1,51 @@ +package io.quarkus.awt.it; + +import java.awt.Image; +import java.awt.image.BufferedImage; +import java.io.ByteArrayInputStream; +import java.io.ByteArrayOutputStream; +import java.io.IOException; +import java.util.Base64; + +import javax.imageio.ImageIO; + +import org.junit.jupiter.api.Assertions; +import org.junit.jupiter.api.Test; + +import io.quarkus.test.junit.main.QuarkusMainTest; + +@QuarkusMainTest +public class AwtImageResizeTest { + + @Test + public void testImageResize() throws IOException { + final String imageData = ""; + final String expected = ""; + + final byte[] imageBytes = Base64.getDecoder().decode(imageData.split(",", 2)[1]); + final String resizeImage = resizeImage(imageBytes, 100); + Assertions.assertEquals(expected, resizeImage); + } + + private String resizeImage(byte[] imageBytes, int height) throws IOException { + BufferedImage inputImage = ImageIO.read(new ByteArrayInputStream(imageBytes)); + + int currentW = inputImage.getWidth(); + int currentH = inputImage.getHeight(); + int width = currentW * height / currentH; + if (currentH < height) { + width = currentW; + height = currentH; + } + + Image originalImage = inputImage.getScaledInstance(width, height, Image.SCALE_SMOOTH); + BufferedImage resizedImage = new BufferedImage(width, height, BufferedImage.TYPE_3BYTE_BGR); + resizedImage.getGraphics().drawImage(originalImage, 0, 0, null); + + ByteArrayOutputStream outputStream = new ByteArrayOutputStream(); + ImageIO.write(resizedImage, "png", outputStream); + + return String.format("data:image/png;base64,%s", Base64.getEncoder().encodeToString(outputStream.toByteArray())); + } + +} diff --git a/integration-tests/pom.xml b/integration-tests/pom.xml index b8a59a9881b896..e7649c044d4e25 100644 --- a/integration-tests/pom.xml +++ b/integration-tests/pom.xml @@ -114,6 +114,7 @@ avro-reload + awt bouncycastle bouncycastle-fips bouncycastle-fips-jsse