org.apache.pdfbox.pdmodel.interactive.digitalsignature.PDSignature Maven / Gradle / Ivy
/*
* 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.interactive.digitalsignature;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Calendar;
import org.apache.pdfbox.cos.COSArray;
import org.apache.pdfbox.cos.COSDictionary;
import org.apache.pdfbox.cos.COSInteger;
import org.apache.pdfbox.cos.COSName;
import org.apache.pdfbox.cos.COSString;
import org.apache.pdfbox.pdmodel.common.COSObjectable;
/**
* This represents a digital signature that can be attached to a document.
*
* @author Ben Litchfield
* @author Thomas Chojecki
*/
public class PDSignature implements COSObjectable
{
private final COSDictionary dictionary;
/**
* A signature filter value.
*/
public static final COSName FILTER_ADOBE_PPKLITE = COSName.ADOBE_PPKLITE;
/**
* A signature filter value.
*/
public static final COSName FILTER_ENTRUST_PPKEF = COSName.ENTRUST_PPKEF;
/**
* A signature filter value.
*/
public static final COSName FILTER_CICI_SIGNIT = COSName.CICI_SIGNIT;
/**
* A signature filter value.
*/
public static final COSName FILTER_VERISIGN_PPKVS = COSName.VERISIGN_PPKVS;
/**
* A signature subfilter value.
*/
public static final COSName SUBFILTER_ADBE_X509_RSA_SHA1 = COSName.ADBE_X509_RSA_SHA1;
/**
* A signature subfilter value.
*/
public static final COSName SUBFILTER_ADBE_PKCS7_DETACHED = COSName.ADBE_PKCS7_DETACHED;
/**
* A signature subfilter value.
*/
public static final COSName SUBFILTER_ETSI_CADES_DETACHED = COSName.getPDFName("ETSI.CAdES.detached");
/**
* A signature subfilter value.
*/
public static final COSName SUBFILTER_ADBE_PKCS7_SHA1 = COSName.ADBE_PKCS7_SHA1;
/**
* Default constructor.
*/
public PDSignature()
{
dictionary = new COSDictionary();
dictionary.setItem(COSName.TYPE, COSName.SIG);
}
/**
* Constructor.
*
* @param dict The signature dictionary.
*/
public PDSignature(COSDictionary dict)
{
dictionary = dict;
}
/**
* Convert this standard java object to a COS dictionary.
*
* @return The COS dictionary that matches this Java object.
*/
public COSDictionary getCOSObject()
{
return dictionary;
}
/**
* Set the dictionary type.
*
* @param type is the dictionary type.
*/
public void setType(COSName type)
{
dictionary.setItem(COSName.TYPE, type);
}
/**
* Set the filter.
*
* @param filter the filter to be used
*/
public void setFilter(COSName filter)
{
dictionary.setItem(COSName.FILTER, filter);
}
/**
* Set a subfilter that specify the signature that should be used.
*
* @param subfilter the subfilter that shall be used.
*/
public void setSubFilter(COSName subfilter)
{
dictionary.setItem(COSName.SUB_FILTER, subfilter);
}
/**
* Sets the name.
* @param name the name to be used
*/
public void setName(String name)
{
dictionary.setString(COSName.NAME, name);
}
/**
* Sets the location.
* @param location the location to be used
*/
public void setLocation(String location)
{
dictionary.setString(COSName.LOCATION, location);
}
/**
* Sets the reason.
*
* @param reason the reason to be used
*/
public void setReason(String reason)
{
dictionary.setString(COSName.REASON, reason);
}
/**
* Sets the contact info.
*
* @param contactInfo the contact info to be used
*/
public void setContactInfo(String contactInfo)
{
dictionary.setString(COSName.CONTACT_INFO, contactInfo);
}
/**
* Set the sign date.
*
* @param cal the date to be used as sign date
*/
public void setSignDate(Calendar cal)
{
dictionary.setDate(COSName.M, cal);
}
/**
* Returns the filter.
* @return the filter
*/
public String getFilter()
{
return dictionary.getNameAsString(COSName.FILTER);
}
/**
* Returns the subfilter.
*
* @return the subfilter
*/
public String getSubFilter()
{
return dictionary.getNameAsString(COSName.SUB_FILTER);
}
/**
* Returns the name.
*
* @return the name
*/
public String getName()
{
return dictionary.getString(COSName.NAME);
}
/**
* Returns the location.
*
* @return the location
*/
public String getLocation()
{
return dictionary.getString(COSName.LOCATION);
}
/**
* Returns the reason.
*
* @return the reason
*/
public String getReason()
{
return dictionary.getString(COSName.REASON);
}
/**
* Returns the contact info.
*
* @return teh contact info
*/
public String getContactInfo()
{
return dictionary.getString(COSName.CONTACT_INFO);
}
/**
* Returns the sign date.
*
* @return the sign date
*/
public Calendar getSignDate()
{
return dictionary.getDate(COSName.M);
}
/**
* Sets the byte range.
*
* @param range the byte range to be used
*/
public void setByteRange(int[] range)
{
if (range.length!=4)
{
return;
}
COSArray ary = new COSArray();
for ( int i : range )
{
ary.add(COSInteger.get(i));
}
dictionary.setItem(COSName.BYTERANGE, ary);
}
/**
* Read out the byterange from the file.
*
* @return a integer array with the byterange
*/
public int[] getByteRange()
{
COSArray byteRange = (COSArray)dictionary.getDictionaryObject(COSName.BYTERANGE);
int[] ary = new int[byteRange.size()];
for (int i = 0; i and )
else if(buffer[c-1]==0x3E || buffer[c-1]==0x29)
{
byteOS.write(buffer, 0, c-1);
}
else
{
byteOS.write(buffer, 0, c);
}
}
fis.close();
return COSString.parseHex(byteOS.toString("ISO-8859-1")).getBytes();
}
/**
* Sets the contents.
*
* @param bytes contents to be used
*/
public void setContents(byte[] bytes)
{
COSString string = new COSString(bytes);
string.setForceHexForm(true);
dictionary.setItem(COSName.CONTENTS, string);
}
/**
* Will return the signed content of the document.
*
* @param pdfFile The signed pdf file as InputStream
* @return a byte array containing only the signed part of the content
* @throws IOException if the pdfFile can't be read
*/
public byte[] getSignedContent(InputStream pdfFile) throws IOException
{
COSFilterInputStream fis=null;
try
{
fis = new COSFilterInputStream(pdfFile,getByteRange());
return fis.toByteArray();
}
finally
{
if (fis != null)
{
fis.close();
}
}
}
/**
* Will return the signed content of the document.
*
* @param pdfFile The signed pdf file as byte array
* @return a byte array containing only the signed part of the content
* @throws IOException if the pdfFile can't be read
*/
public byte[] getSignedContent(byte[] pdfFile) throws IOException
{
COSFilterInputStream fis=null;
try
{
fis = new COSFilterInputStream(pdfFile,getByteRange());
return fis.toByteArray();
}
finally
{
if (fis != null)
{
fis.close();
}
}
}
/**
* PDF signature build dictionary. Provides informations about the signature handler.
*
* @return the pdf signature build dictionary.
*/
public PDPropBuild getPropBuild()
{
PDPropBuild propBuild = null;
COSDictionary propBuildDic = (COSDictionary)dictionary.getDictionaryObject(COSName.PROP_BUILD);
if (propBuildDic != null)
{
propBuild = new PDPropBuild(propBuildDic);
}
return propBuild;
}
/**
* PDF signature build dictionary. Provides informations about the signature handler.
*
* @param propBuild the prop build
*/
public void setPropBuild(PDPropBuild propBuild)
{
dictionary.setItem(COSName.PROP_BUILD, propBuild);
}
}