Search code examples
c#pdfpdf-generationitext7wmf

iText7 WmfImage from byte array throws IO Exception


can someone please confirm this is a bug?

Loading a WMF file as WmfImage from a file (see code, method 1) works, but loading it from a byte array (method 2) fails.

    PdfWriter writer = new PdfWriter(dest);
    PdfDocument pdf = new PdfDocument(writer);
    Document document = new Document(pdf);

    // Method 1: WmfImageData from file (works).
    WmfImageData imageData1 = new WmfImageData("test.wmf");
    PdfFormXObject xObject1 = new PdfFormXObject(imageData1, pdf);
    document.Add(new Image(xObject1));

    // Method 2: WmfImageData from byte[] (fails).
    byte[] wmfBytes = File.ReadAllBytes("test.wmf");
    WmfImageData imageData2 = new WmfImageData(wmfBytes);
    PdfFormXObject xObject2 = new PdfFormXObject(imageData2, pdf);
    document.Add(new Image(xObject2));

    document.Close();

In my case the second method is useful because I would be able to generate a Microsoft Chart and convert it to a WMF byte array, and place it in the PDF without having to save it to a file first.

Method 2 throws this error:

    System.NullReferenceException
  HResult=0x80004003
  Message=Object reference not set to an instance of an object.
  Source=itext.io
  StackTrace:
   at iText.IO.Util.UrlUtil.OpenStream(Uri url) in itext7-dotnet\itext\itext.io\itext\io\util\UrlUtil.cs:line 73
   at iText.Kernel.Pdf.Canvas.Wmf.WmfImageData.ReadImageType(Uri source) in itext7-dotnet\itext\itext.kernel\itext\kernel\pdf\canvas\wmf\WmfImageData.cs:line 94
   at iText.Kernel.Pdf.Canvas.Wmf.WmfImageData..ctor(Byte[] bytes) in itext7-dotnet\itext\itext.kernel\itext\kernel\pdf\canvas\wmf\WmfImageData.cs:line 76

The cause seems to be the constructor accepting a byte array in the iText.Kernel.Pdf.Canvas.Wmf.WmfImageData class. It tries to check whether the argument is a proper WMF image, but does so by trying to load bytes from an URI which does not exist.

My suggestion for a fix is the addition of the following function in the WmfImageData class, and changing one line in the constructor from ReadImageType(url) to ReadImageType(bytes).

    private static byte[] ReadImageType(byte[] bytes) {
        if (bytes.Length > 1) {
            return new byte[] { bytes[0], bytes[1] };
        }
        return null;
    }

Solution

  • Assuming this source: https://github.com/itext/itext7/blob/develop/kernel/src/main/java/com/itextpdf/kernel/pdf/canvas/wmf/WmfImageData.java

    Yes, I can confirm the constructor which takes a byte[] errantly references the base class field url without that field being initialized.

    I will leave the how to fix up to the community of developers working on that project as they have more familiarity with the inner workings of WMF images. However, if I'm reading their ReadImageType(url) function properly, then it uses the first 8 bytes as a type descriptor.