PNGConverterTest.java
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.pdfbox.pdmodel.graphics.image;
import java.awt.Color;
import java.awt.color.ColorSpace;
import java.awt.color.ICC_ColorSpace;
import java.awt.color.ICC_Profile;
import java.awt.image.BufferedImage;
import java.awt.image.ColorModel;
import java.awt.image.ComponentColorModel;
import java.awt.image.WritableRaster;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Hashtable;
import javax.imageio.ImageIO;
import org.apache.pdfbox.cos.COSName;
import org.apache.pdfbox.pdmodel.PDDocument;
import org.apache.pdfbox.pdmodel.PDPage;
import org.apache.pdfbox.pdmodel.PDPageContentStream;
import org.apache.pdfbox.pdmodel.graphics.color.PDICCBased;
import org.apache.pdfbox.pdmodel.graphics.color.PDIndexed;
import static org.apache.pdfbox.pdmodel.graphics.image.LosslessFactoryTest.checkIdentRaw;
import static org.apache.pdfbox.pdmodel.graphics.image.ValidateXImage.checkIdent;
import static org.junit.jupiter.api.Assertions.assertArrayEquals;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertTrue;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.parallel.Execution;
import org.junit.jupiter.api.parallel.ExecutionMode;
@Execution(ExecutionMode.CONCURRENT)
class PNGConverterTest
{
private static final File parentDir = new File("target/test-output/graphics/graphics");
@BeforeAll
static void setup()
{
//noinspection ResultOfMethodCallIgnored
parentDir.mkdirs();
}
/**
* This "test" just dumps the list of constants for the PNGConverter CHUNK_??? types, so that
* it can just be copy&pasted into the PNGConverter class.
*/
//@Test
public void dumpChunkTypes()
{
final String[] chunkTypes = { "IHDR", "IDAT", "PLTE", "IEND", "tRNS", "cHRM", "gAMA",
"iCCP", "sBIT", "sRGB", "tEXt", "zTXt", "iTXt", "kBKG", "hIST", "pHYs", "sPLT",
"tIME" };
for (String chunkType : chunkTypes)
{
byte[] bytes = chunkType.getBytes();
assertEquals(4, bytes.length);
System.out.println(String.format("\tprivate static final int CHUNK_" + chunkType
+ " = 0x%02X%02X%02X%02X; // %s: %d %d %d %d", (int) bytes[0] & 0xFF,
(int) bytes[1] & 0xFF, (int) bytes[2] & 0xFF, (int) bytes[3] & 0xFF, chunkType,
(int) bytes[0] & 0xFF, (int) bytes[1] & 0xFF, (int) bytes[2] & 0xFF,
(int) bytes[3] & 0xFF));
}
}
@Test
void testImageConversionRGB() throws IOException
{
checkImageConvert("png.png");
}
@Test
void testImageConversionRGBGamma() throws IOException
{
checkImageConvert("png_rgb_gamma.png");
}
@Test
void testImageConversionRGB16BitICC() throws IOException
{
checkImageConvert("png_rgb_romm_16bit.png");
}
@Test
void testImageConversionRGBIndexed() throws IOException
{
checkImageConvert("png_indexed.png");
}
@Test
void testImageConversionRGBIndexedAlpha1Bit() throws IOException
{
checkImageConvert("png_indexed_1bit_alpha.png");
}
@Test
void testImageConversionRGBIndexedAlpha2Bit() throws IOException
{
checkImageConvert("png_indexed_2bit_alpha.png");
}
@Test
void testImageConversionRGBIndexedAlpha4Bit() throws IOException
{
checkImageConvert("png_indexed_4bit_alpha.png");
}
@Test
void testImageConversionRGBIndexedAlpha8Bit() throws IOException
{
checkImageConvert("png_indexed_8bit_alpha.png");
}
@Test
void testImageConversionRGBAlpha() throws IOException
{
// We can't handle Alpha RGB
checkImageConvertFail("png_alpha_rgb.png");
}
@Test
void testImageConversionGrayAlpha() throws IOException
{
// We can't handle Alpha RGB
checkImageConvertFail("png_alpha_gray.png");
}
@Test
void testImageConversionGray() throws IOException
{
checkImageConvertFail("png_gray.png");
}
@Test
void testImageConversionGrayGamma() throws IOException
{
checkImageConvertFail("png_gray_with_gama.png");
}
private void checkImageConvertFail(String name) throws IOException
{
try (PDDocument doc = new PDDocument())
{
InputStream in = PNGConverterTest.class.getResourceAsStream(name);
byte[] imageBytes = in.readAllBytes();
PDImageXObject pdImageXObject = PNGConverter.convertPNGImage(doc, imageBytes);
assertNull(pdImageXObject);
}
}
private void checkImageConvert(String name) throws IOException
{
try (PDDocument doc = new PDDocument())
{
InputStream in = PNGConverterTest.class.getResourceAsStream(name);
byte[] imageBytes = in.readAllBytes();
PDImageXObject pdImageXObject = PNGConverter.convertPNGImage(doc, imageBytes);
assertNotNull(pdImageXObject);
ICC_Profile imageProfile = null;
if (pdImageXObject.getColorSpace() instanceof PDICCBased)
{
// Make sure that ICC profile is a valid one
PDICCBased iccColorSpace = (PDICCBased) pdImageXObject.getColorSpace();
imageProfile = ICC_Profile.getInstance(iccColorSpace.getPDStream().toByteArray());
}
PDPage page = new PDPage();
doc.addPage(page);
try (PDPageContentStream contentStream = new PDPageContentStream(doc, page))
{
contentStream.setNonStrokingColor(Color.PINK);
contentStream.addRect(0, 0, page.getCropBox().getWidth(), page.getCropBox().getHeight());
contentStream.fill();
contentStream.drawImage(pdImageXObject, 0, 0, pdImageXObject.getWidth(),
pdImageXObject.getHeight());
}
doc.save(new File(parentDir, name + ".pdf"));
BufferedImage image = pdImageXObject.getImage();
assertNotNull(pdImageXObject.getRawRaster());
BufferedImage expectedImage = ImageIO.read(new ByteArrayInputStream(imageBytes));
if (imageProfile != null && expectedImage.getColorModel().getColorSpace().isCS_sRGB())
{
// The image has an embedded ICC Profile, but the default java PNG
// reader does not correctly read that.
expectedImage = getImageWithProfileData(expectedImage, imageProfile);
}
checkIdent(expectedImage, image);
BufferedImage rawImage = pdImageXObject.getRawImage();
if (rawImage != null)
{
assertEquals(rawImage.getWidth(), pdImageXObject.getWidth());
assertEquals(rawImage.getHeight(), pdImageXObject.getHeight());
// We compare the raw data
checkIdentRaw(expectedImage, pdImageXObject);
}
}
}
public static BufferedImage getImageWithProfileData(BufferedImage sourceImage,
ICC_Profile realProfile)
{
Hashtable<String, Object> properties = new Hashtable<>();
String[] propertyNames = sourceImage.getPropertyNames();
if (propertyNames != null)
{
for (String propertyName : propertyNames)
{
properties.put(propertyName, sourceImage.getProperty(propertyName));
}
}
ComponentColorModel oldColorModel = (ComponentColorModel) sourceImage.getColorModel();
boolean hasAlpha = oldColorModel.hasAlpha();
int transparency = oldColorModel.getTransparency();
boolean alphaPremultiplied = oldColorModel.isAlphaPremultiplied();
WritableRaster raster = sourceImage.getRaster();
int dataType = raster.getDataBuffer().getDataType();
int[] componentSize = oldColorModel.getComponentSize();
final ColorModel colorModel = new ComponentColorModel(new ICC_ColorSpace(realProfile),
componentSize, hasAlpha, alphaPremultiplied, transparency, dataType);
return new BufferedImage(colorModel, raster, sourceImage.isAlphaPremultiplied(),
properties);
}
@Test
void testCheckConverterState()
{
assertFalse(PNGConverter.checkConverterState(null));
PNGConverter.PNGConverterState state = new PNGConverter.PNGConverterState();
assertFalse(PNGConverter.checkConverterState(state));
PNGConverter.Chunk invalidChunk = new PNGConverter.Chunk();
invalidChunk.bytes = new byte[0];
assertFalse(PNGConverter.checkChunkSane(invalidChunk));
// Valid Dummy Chunk
PNGConverter.Chunk validChunk = new PNGConverter.Chunk();
validChunk.bytes = new byte[16];
validChunk.start = 4;
validChunk.length = 8;
validChunk.crc = 2077607535;
assertTrue(PNGConverter.checkChunkSane(validChunk));
state.IHDR = invalidChunk;
assertFalse(PNGConverter.checkConverterState(state));
state.IDATs = Collections.singletonList(validChunk);
assertFalse(PNGConverter.checkConverterState(state));
state.IHDR = validChunk;
assertTrue(PNGConverter.checkConverterState(state));
state.IDATs = new ArrayList<>();
assertFalse(PNGConverter.checkConverterState(state));
state.IDATs = Collections.singletonList(validChunk);
assertTrue(PNGConverter.checkConverterState(state));
state.PLTE = invalidChunk;
assertFalse(PNGConverter.checkConverterState(state));
state.PLTE = validChunk;
assertTrue(PNGConverter.checkConverterState(state));
state.cHRM = invalidChunk;
assertFalse(PNGConverter.checkConverterState(state));
state.cHRM = validChunk;
assertTrue(PNGConverter.checkConverterState(state));
state.tRNS = invalidChunk;
assertFalse(PNGConverter.checkConverterState(state));
state.tRNS = validChunk;
assertTrue(PNGConverter.checkConverterState(state));
state.iCCP = invalidChunk;
assertFalse(PNGConverter.checkConverterState(state));
state.iCCP = validChunk;
assertTrue(PNGConverter.checkConverterState(state));
state.sRGB = invalidChunk;
assertFalse(PNGConverter.checkConverterState(state));
state.sRGB = validChunk;
assertTrue(PNGConverter.checkConverterState(state));
state.gAMA = invalidChunk;
assertFalse(PNGConverter.checkConverterState(state));
state.gAMA = validChunk;
assertTrue(PNGConverter.checkConverterState(state));
state.IDATs = Arrays.asList(validChunk, invalidChunk);
assertFalse(PNGConverter.checkConverterState(state));
}
@Test
void testChunkSane()
{
PNGConverter.Chunk chunk = new PNGConverter.Chunk();
assertTrue(PNGConverter.checkChunkSane(null));
chunk.bytes = "IHDRsomedummyvaluesDummyValuesAtEnd".getBytes();
chunk.length = 19;
assertEquals(35, chunk.bytes.length);
assertEquals("IHDRsomedummyvalues", new String(chunk.getData()));
assertFalse(PNGConverter.checkChunkSane(chunk));
chunk.start = 4;
assertEquals("somedummyvaluesDumm", new String(chunk.getData()));
assertFalse(PNGConverter.checkChunkSane(chunk));
chunk.crc = -1729802258;
assertTrue(PNGConverter.checkChunkSane(chunk));
chunk.start = 6;
assertFalse(PNGConverter.checkChunkSane(chunk));
chunk.length = 60;
assertFalse(PNGConverter.checkChunkSane(chunk));
}
@Test
void testCRCImpl()
{
byte[] b1 = "Hello World!".getBytes();
assertEquals(472456355, PNGConverter.crc(b1, 0, b1.length));
assertEquals(-632335482, PNGConverter.crc(b1, 2, b1.length - 4));
}
@Test
void testMapPNGRenderIntent()
{
assertEquals(COSName.PERCEPTUAL, PNGConverter.mapPNGRenderIntent(0));
assertEquals(COSName.RELATIVE_COLORIMETRIC, PNGConverter.mapPNGRenderIntent(1));
assertEquals(COSName.SATURATION, PNGConverter.mapPNGRenderIntent(2));
assertEquals(COSName.ABSOLUTE_COLORIMETRIC, PNGConverter.mapPNGRenderIntent(3));
assertNull(PNGConverter.mapPNGRenderIntent(-1));
assertNull(PNGConverter.mapPNGRenderIntent(4));
}
/**
* Test code coverage for /Intent /Perceptual and for sRGB icc profile in indexed colorspace.
*
* @throws IOException
*/
@Test
void testImageConversionIntentIndexed() throws IOException
{
checkImageConvert("929316.png");
try (PDDocument doc = new PDDocument())
{
InputStream in = PNGConverterTest.class.getResourceAsStream("929316.png");
byte[] imageBytes = in.readAllBytes();
PDImageXObject pdImageXObject = PNGConverter.convertPNGImage(doc, imageBytes);
assertEquals(COSName.PERCEPTUAL, pdImageXObject.getCOSObject().getItem(COSName.INTENT));
// Check that this image gets an indexed colorspace with sRGB ICC based colorspace
PDIndexed indexedColorspace = (PDIndexed) pdImageXObject.getColorSpace();
PDICCBased iccColorspace = (PDICCBased) indexedColorspace.getBaseColorSpace();
// validity of ICC CS is tested in checkImageConvert
// should be an sRGB profile. Or at least, the data that is in ColorSpace.CS_sRGB and
// that was assigned in PNGConvert.
// (PDICCBased.is_sRGB() fails in openjdk on that data, maybe it is not a "real" sRGB)
ICC_Profile rgbProfile = ICC_Profile.getInstance(ColorSpace.CS_sRGB);
byte[] sRGB_bytes = rgbProfile.getData();
assertArrayEquals(sRGB_bytes, iccColorspace.getPDStream().toByteArray());
}
}
}