All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jpedal.io.ObjectDecoder Maven / Gradle / Ivy

There is a newer version: 7.15.25
Show newest version
/*
 * ===========================================
 * Java Pdf Extraction Decoding Access Library
 * ===========================================
 *
 * Project Info:  http://www.idrsolutions.com
 * Help section for developers at http://www.idrsolutions.com/support/
 *
 * (C) Copyright 1997-2017 IDRsolutions and Contributors.
 *
 * This file is part of JPedal/JPDF2HTML5
 *
 @LICENSE@
 *
 * ---------------
 * ObjectDecoder.java
 * ---------------
 */
package org.jpedal.io;

import java.io.Serializable;

import org.jpedal.io.security.DecryptionFactory;
import org.jpedal.io.types.*;
import org.jpedal.objects.raw.PdfDictionary;
import org.jpedal.objects.raw.PdfObject;
import org.jpedal.utils.NumberUtils;
import org.jpedal.utils.StringUtils;

/**
 *
 */
public class ObjectDecoder implements Serializable {

    public PdfFileReader objectReader;

    public DecryptionFactory decryption;

    static final byte[] endPattern = {101, 110, 100, 111, 98, 106}; //pattern endobj

    //not final in IDE but in build do our static analysis does not flag as dead debug code
    //which we  want compiler to ooptimise out
    public static boolean debugFastCode; //objRef.equals("68 0 R")

    /**
     * used in debugging output
     */
    public static String padding = "";

    boolean isInlineImage;

    private int endPt = -1;

    public ObjectDecoder(final PdfFileReader pdfFileReader) {
        init(pdfFileReader);
    }

    private void init(final PdfFileReader objectReader) {
        this.objectReader = objectReader;

        this.decryption = objectReader.getDecryptionObject();
    }

    /**
     * read a dictionary object
     */
    public final int readDictionaryAsObject(final PdfObject pdfObject, int i, final byte[] raw) {

        if (endPt == -1) {
            endPt = raw.length;
        }

        //used to debug issues by printing out details for obj
        //(set to non-final above)
        //debugFastCode =pdfObject.getObjectRefAsString().equals("5 0 R");

        if (debugFastCode) {
            padding += "   ";
        }

        //show details in debug mode
        if (debugFastCode) {
            ObjectUtils.showData(pdfObject, i, raw, padding);
        }
        
        /*
         * main loop for read all values from Object data and store in PDF object
         */
        i = readObjectDataValues(pdfObject, i, raw);
        
        /*
         * look for stream afterwards
         */
        if (!pdfObject.ignoreStream() && pdfObject.getGeneralType(-1) != PdfDictionary.ID) {
            Stream.readStreamData(pdfObject, i, raw, raw.length, objectReader);
        }
        
        /*
         * we need full names for Forms
         */
        if (pdfObject.getObjectType() == PdfDictionary.Form) {
            Form.setFieldNames(pdfObject, objectReader);
        }
        
        /*
         * reset indent in debugging
         */
        if (debugFastCode) {
            final int len = padding.length();

            if (len > 3) {
                padding = padding.substring(0, len - 3);
            }
        }

        return i;

    }

    /**
     * get the values from the data stream and store in PdfObject
     *
     * @param pdfObject
     * @param i
     * @param raw
     * @return
     */
    private int readObjectDataValues(final PdfObject pdfObject, int i, final byte[] raw) {

        final int length = raw.length;

        int level = 0;
        //allow for no << at start
        if (isInlineImage) {
            level = 1;
        }

        while (true) {

            if (i < length && raw[i] == 37) //allow for comment and ignore
            {
                i = StreamReaderUtils.skipComment(raw, i);
            }
            
            /*
             * exit conditions
             */
            if ((i >= length ||
                    (endPt != -1 && i >= endPt)) ||
                    (raw[i] == 101 && raw[i + 1] == 110 && raw[i + 2] == 100 && raw[i + 3] == 111) ||
                    (raw[i] == 's' && raw[i + 1] == 't' && raw[i + 2] == 'r' && raw[i + 3] == 'e' && raw[i + 4] == 'a' && raw[i + 5] == 'm')) {
                break;
            }
            
            /*
             * process value
             */
            if (raw[i] == 60 && raw[i + 1] == 60) {
                i++;
                level++;
            } else if (raw[i] == 62 && i + 1 != length && raw[i + 1] == 62) {
                i++;
                level--;

                if (level == 0) {
                    break;
                }
            } else if (raw[i] == 47) {

                i++;

                if (raw[i] != 47 && raw[i] != 32) { //allow for oddity of //DeviceGray  and / /DeviceGray in colorspace
                    i = getKey(i, raw, length, pdfObject);
                }
            }

            i++;

        }

        return i;
    }

    private int getKey(int i, final byte[] raw, final int length, final PdfObject pdfObject) {
        //everything from /

        final int keyStart = i;
        final int keyLength = StreamReaderUtils.findDictionaryEnd(i, raw, length);
        i += keyLength;
        final int type = pdfObject.getObjectType();

        if (debugFastCode) {
            System.out.println("type=" + type + ' ' + ' ' + pdfObject.getID() + " chars=" + (char) raw[i - 1] + (char) raw[i] + (char) raw[i + 1] + ' ' + pdfObject + " i=" + i);

        }

        int PDFkeyInt = PdfDictionary.getIntKey(keyStart, keyLength, raw);

        if (isInlineImage) {
            PDFkeyInt = PdfObjectFactory.getInlineID(PDFkeyInt);
        }

        final int pdfKeyType = PdfDictionary.getKeyType(PDFkeyInt, type);

        if (raw[i] == 47 || raw[i] == 40 || (raw[i] == 91 && raw[i + 1] != ']')) { //move back cursor
            i--;
        }

        //check for unknown value and ignore
        if (pdfKeyType == -1) {
            i = ObjectUtils.handleUnknownType(i, raw, length);
        }

        /*
        * now read value
        */
        if (PDFkeyInt == -1 || pdfKeyType == -1) {
            if (debugFastCode) {
                System.out.println(padding + pdfObject.getObjectRefAsString() + " =================Not implemented=" + PdfDictionary.getKey(keyStart, keyLength, raw) + " pdfKeyType=" + pdfKeyType);
            }
        } else {
            if (debugFastCode) {
                System.out.println(padding + pdfObject.getObjectRefAsString() + " =================Reading value for key=" + PdfDictionary.getKey(keyStart, keyLength, raw) + " (" + PDFkeyInt + ") type=" + PdfDictionary.showAsConstant(pdfKeyType) + ' ' + pdfObject);
            }
            i = setValue(pdfObject, i, raw, PDFkeyInt, pdfKeyType);
        }
        return i;
    }

    private int setValue(final PdfObject pdfObject, int i, final byte[] raw, final int PDFkeyInt, int pdfKeyType) {

        //resolve now in this case as we need to ensure all parts present
        if (pdfKeyType == PdfDictionary.VALUE_IS_UNREAD_DICTIONARY && pdfObject.isDataExternal()) {
            pdfKeyType = PdfDictionary.VALUE_IS_DICTIONARY;
        }

        switch (pdfKeyType) {

            //read text stream (this is text) <050F>
            case PdfDictionary.VALUE_IS_TEXTSTREAM: {
                i = TextStream.setTextStreamValue(pdfObject, i, raw, PDFkeyInt, objectReader);
                break;

            }
            case PdfDictionary.VALUE_IS_NAMETREE: {
                i = Name.setNameTreeValue(pdfObject, i, raw, PDFkeyInt, objectReader);
                break;

                //readDictionary keys << /A 12 0 R /B 13 0 R >>
            }
            case PdfDictionary.VALUE_IS_DICTIONARY_PAIRS: {
                i = Dictionary.setDictionaryValue(pdfObject, i, raw, objectReader, PDFkeyInt, true);
                break;

                //Strings
            }
            case PdfDictionary.VALUE_IS_STRING_ARRAY: {
                final ArrayDecoder objDecoder = new StringArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read Object Refs in [] (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_BOOLEAN_ARRAY: {
                final ArrayDecoder objDecoder = new BooleanArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read Object Refs in [] (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_KEY_ARRAY: {
                final ArrayDecoder objDecoder = new KeyArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read numbers in [] (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_MIXED_ARRAY: {
                final ArrayDecoder objDecoder = new Array(objectReader, i, PdfDictionary.VALUE_IS_MIXED_ARRAY, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read numbers in [] (may be indirect ref) same as Mixed but allow for recursion and store as objects
            }
            case PdfDictionary.VALUE_IS_OBJECT_ARRAY: {
                final ArrayDecoder objDecoder = new ObjectArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read numbers in [] (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_DOUBLE_ARRAY: {
                final ArrayDecoder objDecoder = new DoubleArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read numbers in [] (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_INT_ARRAY: {
                final ArrayDecoder objDecoder = new IntArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read numbers in [] (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_FLOAT_ARRAY: {
                final ArrayDecoder objDecoder = new FloatArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;

                //read String (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_NAME: {
                i = Name.setNameStringValue(pdfObject, i, raw, PDFkeyInt, objectReader);
                break;

                //read true or false
            }
            case PdfDictionary.VALUE_IS_BOOLEAN: {
                i = BooleanValue.set(pdfObject, i, raw, PDFkeyInt);
                break;

                //read known set of values
            }
            case PdfDictionary.VALUE_IS_STRING_CONSTANT: {
                i = StringValue.setStringConstantValue(pdfObject, i, raw, PDFkeyInt);
                break;

                //read known set of values
            }
            case PdfDictionary.VALUE_IS_STRING_KEY: {
                i = StringValue.setStringKeyValue(pdfObject, i, raw, PDFkeyInt);
                break;

                //read number (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_INT: {

                i = StreamReaderUtils.skipSpacesOrOtherCharacter(raw, i, 47);

                i = NumberValue.setNumberValue(pdfObject, i, raw, PDFkeyInt, objectReader);
                break;

                //read float number (may be indirect ref)
            }
            case PdfDictionary.VALUE_IS_FLOAT: {
                i = FloatValue.setFloatValue(pdfObject, i, raw, PDFkeyInt, objectReader);
                break;

                //read known Dictionary object which may be direct or indirect
            }
            case PdfDictionary.VALUE_IS_UNREAD_DICTIONARY: {
                i = Dictionary.setUnreadDictionaryValue(pdfObject, i, raw, PDFkeyInt, isInlineImage);
                break;

            }
            case PdfDictionary.VALUE_IS_VARIOUS: {
                if (raw.length - 5 > 0 && StreamReaderUtils.isNull(raw, i + 1)) { //ignore null value and skip (ie /N null)
                    i += 5;
                } else {
                    i = setVariousValue(pdfObject, i, raw, PDFkeyInt, objectReader);
                }
                break;

            }
            case PdfDictionary.VALUE_IS_DICTIONARY: {
                i = Dictionary.setDictionaryValue(pdfObject, i, raw, PDFkeyInt, objectReader);
                break;
            }
            case PdfDictionary.VALUE_IS_MIXED_PAIRS: {
                i = Dictionary.setDictionaryValue(pdfObject, i, raw, objectReader, PDFkeyInt, false);
                break;

                //Strings
            }
        }
        return i;
    }

    static int setVariousValue(final PdfObject pdfObject, int i, final byte[] raw, final int PDFkeyInt, final PdfFileReader objectReader) {

        if (raw[i] != '<') {
            i++;
        }

        if (debugFastCode) {
            System.out.println(padding + "Various value (first char=" + (char) raw[i] + (char) raw[i + 1] + " )");
        }

        if (raw[i] == '/') {
            i = Name.setNameStringValue(pdfObject, i, raw, PDFkeyInt, objectReader);
        } else if (raw[i] == 'f' && raw[i + 1] == 'a' && raw[i + 2] == 'l' && raw[i + 3] == 's' && raw[i + 4] == 'e') {
            pdfObject.setBoolean(PDFkeyInt, false);
            i += 4;
        } else if (raw[i] == 't' && raw[i + 1] == 'r' && raw[i + 2] == 'u' && raw[i + 3] == 'e') {
            pdfObject.setBoolean(PDFkeyInt, true);
            i += 3;
        } else if (raw[i] == '(' || (raw[i] == '<' && raw[i - 1] != '<' && raw[i + 1] != '<')) {
            i = TextStream.readTextStream(pdfObject, i, raw, PDFkeyInt, objectReader);
        } else if (raw[i] == '[') {
            i = setArray(pdfObject, i, raw, PDFkeyInt, objectReader);
        } else if ((raw[i] == '<' && raw[i + 1] == '<')) {
            i = Dictionary.readDictionary(pdfObject, i, raw, PDFkeyInt, objectReader);
        } else {
            i = General.readGeneral(pdfObject, i, raw, PDFkeyInt, objectReader);
        }

        return i;
    }

    static int setArray(final PdfObject pdfObject, int i, final byte[] raw, final int PDFkeyInt, final PdfFileReader objectReader) {
        switch (PDFkeyInt) {

            case PdfDictionary.D:
            case PdfDictionary.OpenAction:
            case PdfDictionary.K:
            case PdfDictionary.XFA: {
                final ArrayDecoder objDecoder = new Array(objectReader, i, PdfDictionary.VALUE_IS_MIXED_ARRAY, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;
            }

            case PdfDictionary.Mask: {
                final ArrayDecoder objDecoder = new IntArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);

                break;
            }
            case PdfDictionary.C:
            case PdfDictionary.IC: {
                final ArrayDecoder objDecoder = new FloatArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;
            }

            case PdfDictionary.TR:
            case PdfDictionary.OCGs: {
                final ArrayDecoder objDecoder = new KeyArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;
            }

            case PdfDictionary.DecodeParms: {
                final ArrayDecoder objDecoder = new ObjectArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;
            }


            default: {
                final ArrayDecoder objDecoder = new StringArray(objectReader, i, raw);
                i = objDecoder.readArray(pdfObject, PDFkeyInt);
                break;
            }
        }
        return i;
    }

    /**
     * used by linearization to check object fully fully available and return false if not
     *
     * @param pdfObject
     */
    public static synchronized boolean resolveFully(final PdfObject pdfObject, final PdfFileReader objectReader) {

        boolean fullyResolved = pdfObject != null;

        if (fullyResolved) {

            final byte[] raw;
            if (pdfObject.getStatus() == PdfObject.DECODED) {
                raw = StringUtils.toBytes(pdfObject.getObjectRefAsString());
            } else {
                raw = pdfObject.getUnresolvedData();
            }

            //flag now done and flush raw data
            pdfObject.setStatus(PdfObject.DECODED);

            //allow for empty object
            if (raw[0] != 'e' && raw[1] != 'n' && raw[2] != 'd' && raw[3] != 'o' && raw[4] != 'b') {

                //allow for [ref] at top level (may be followed by gap
                int j = StreamReaderUtils.skipSpacesOrOtherCharacter(raw, 0, 91);

                // get object ref
                int keyStart = j;

                //move cursor to end of reference
                j = StreamReaderUtils.skipToEndOfRef(raw, j);

                final int ref = NumberUtils.parseInt(keyStart, j, raw);

                j = StreamReaderUtils.skipSpaces(raw, j);

                // get generation number
                keyStart = j;

                //move cursor to end of reference
                j = StreamReaderUtils.skipToEndOfRef(raw, j);

                final int generation = NumberUtils.parseInt(keyStart, j, raw);

                if (raw[raw.length - 1] == 'R') //recursively validate all child objects
                {
                    fullyResolved = resolveFullyChildren(pdfObject, fullyResolved, raw, ref, generation, objectReader);
                }

                if (fullyResolved) {
                    pdfObject.ignoreRecursion(false);
                    final ObjectDecoder objDecoder = new ObjectDecoder(objectReader);
                    objDecoder.readDictionaryAsObject(pdfObject, j, raw);
                }
            }
        }

        return fullyResolved;
    }

    static boolean resolveFullyChildren(final PdfObject pdfObject, boolean fullyResolved, final byte[] raw, final int ref, final int generation, final PdfFileReader objectReader) {

        pdfObject.setRef(new String(raw));
        pdfObject.isDataExternal(true);

        final byte[] pageData = objectReader.readObjectAsByteArray(pdfObject, objectReader.isCompressed(ref, generation), ref, generation);

        //allow for data in Linear object not yet loaded
        if (pageData == null) {
            pdfObject.setFullyResolved(false);
            fullyResolved = false;
        } else {
            pdfObject.setStatus(PdfObject.UNDECODED_DIRECT);
            pdfObject.setUnresolvedData(pageData, PdfDictionary.Linearized);
            pdfObject.isDataExternal(true);

            if (!resolveFully(pdfObject, objectReader)) {
                pdfObject.setFullyResolved(false);
            }
        }

        return fullyResolved;
    }

    /**
     * read object setup to contain only ref to data
     *
     * @param pdfObject
     */
    public final void checkResolved(final PdfObject pdfObject) {

        if (pdfObject != null && pdfObject.getStatus() != PdfObject.DECODED) {

            final byte[] raw = pdfObject.getUnresolvedData();

            //flag now done and flush raw data
            pdfObject.setStatus(PdfObject.DECODED);

            //allow for empty object
            if (raw[0] == 'e' && raw[1] == 'n' && raw[2] == 'd' && raw[3] == 'o' && raw[4] == 'b') {
                //empty object
            } else if (StreamReaderUtils.isNull(raw, 0)) {
                //null object
            } else { //we need to ref from ref elsewhere which may be indirect [ref], hence loop

                String objectRef = pdfObject.getObjectRefAsString();

                if (raw[raw.length - 1] == 'R') {
                    objectRef = new String(raw);
                    pdfObject.setRef(objectRef);
                } else if (raw[0] != '<' && raw[raw.length - 1] == '>') {
                    ////see case 23155 (encrypted annot needs obj ref appended so we can decrypt string later)
                    extractRefFromEnd(raw, pdfObject, objectReader);
                    return;
                }

                Dictionary.readDictionaryFromRefOrDirect(pdfObject, objectRef, 0, raw, -1, objectReader);

            }
        }
    }

    /**
     * see case 23155 (encrypted annot needs obj ref appended so we can decrypt string later)
     */
    private static void extractRefFromEnd(byte[] raw, final PdfObject pdfObject, final PdfFileReader objectReader) {

        final String objectRef;

        //scan along to find number
        int ptr = 0;
        final int len = raw.length;
        for (int jj = 0; jj < len; jj++) {

            if (raw[jj] >= '0' && raw[jj] <= '9') {
                ptr = jj;
                jj = len;
            }
        }
        //check first non-number is R
        int end = ptr;
        while ((raw[end] >= '0' && raw[end] <= '9') || raw[end] == ' ' || raw[end] == 10 || raw[end] == 13 || raw[end] == 9) {
            end++;
        }
        //and store if it is a ref
        if (raw[end] == 'o' && raw[end + 1] == 'b' && raw[end + 2] == 'j') {
            objectRef = new String(raw, 0, end) + 'R';
            final int newArrayLen = raw.length - end - 4;
            final byte[] newArray = new byte[newArrayLen];
            System.arraycopy(raw, end + 4, newArray, 0, newArrayLen);
            raw = newArray;
            pdfObject.setRef(objectRef);

            Dictionary.readDictionaryFromRefOrDirect(pdfObject, objectRef, 0, raw, -1, objectReader);
        }
    }

    /**
     * set end if not end of data stream
     */
    public void setEndPt(final int dataPointer) {
        this.endPt = dataPointer;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy