se.idsec.signservice.security.sign.pdf.impl.BasicPDFSignatureValidator Maven / Gradle / Ivy
/*
* Copyright 2019-2024 IDsec Solutions AB
*
* Licensed 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 se.idsec.signservice.security.sign.pdf.impl;
import lombok.extern.slf4j.Slf4j;
import org.apache.pdfbox.Loader;
import org.apache.pdfbox.pdmodel.PDDocument;
import org.apache.pdfbox.pdmodel.interactive.digitalsignature.PDSignature;
import org.bouncycastle.asn1.ASN1Encodable;
import org.bouncycastle.asn1.ASN1Encoding;
import org.bouncycastle.asn1.ASN1ObjectIdentifier;
import org.bouncycastle.asn1.ASN1Sequence;
import org.bouncycastle.asn1.ASN1TaggedObject;
import org.bouncycastle.asn1.ASN1UTCTime;
import org.bouncycastle.asn1.cms.Attribute;
import org.bouncycastle.asn1.cms.AttributeTable;
import org.bouncycastle.asn1.x509.AlgorithmIdentifier;
import org.bouncycastle.cert.X509CertificateHolder;
import org.bouncycastle.cms.CMSException;
import org.bouncycastle.cms.CMSSignedDataParser;
import org.bouncycastle.cms.CMSTypedStream;
import org.bouncycastle.cms.CMSVerifierCertificateNotValidException;
import org.bouncycastle.cms.SignerInformation;
import org.bouncycastle.cms.SignerInformationStore;
import org.bouncycastle.cms.SignerInformationVerifier;
import org.bouncycastle.cms.jcajce.JcaSimpleSignerInfoVerifierBuilder;
import org.bouncycastle.operator.OperatorCreationException;
import org.bouncycastle.operator.bc.BcDigestCalculatorProvider;
import org.bouncycastle.util.CollectionStore;
import se.idsec.signservice.security.certificate.CertificateUtils;
import se.idsec.signservice.security.certificate.CertificateValidator;
import se.idsec.signservice.security.sign.SignatureValidationResult;
import se.idsec.signservice.security.sign.SignatureValidationResult.Status;
import se.idsec.signservice.security.sign.impl.InternalSignatureValidationException;
import se.idsec.signservice.security.sign.pdf.PDFSignatureValidationResult;
import se.idsec.signservice.security.sign.pdf.PDFSignatureValidator;
import se.idsec.signservice.security.sign.pdf.configuration.PDFAlgorithmRegistry;
import se.idsec.signservice.security.sign.pdf.configuration.PDFObjectIdentifiers;
import se.idsec.signservice.security.sign.pdf.utils.PDFBoxSignatureUtils;
import se.idsec.signservice.utils.Pair;
import se.swedenconnect.security.algorithms.SignatureAlgorithm;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.security.SignatureException;
import java.security.cert.CertificateEncodingException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;
import java.text.ParseException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Calendar;
import java.util.Collection;
import java.util.Collections;
import java.util.Date;
import java.util.Iterator;
import java.util.List;
/**
* Verifies the signature(s) on a PDF document.
*
*
* This is a basic implementation that just verifies that the actual signatures validates correctly and reports what
* certificates that was supplied to provide the matching public key. No attempts are made to validate the certificates
* or any timestamps associated with the signature.
*
*
* @author Martin Lindström ([email protected])
* @author Stefan Santesson ([email protected])
*/
@Slf4j
public class BasicPDFSignatureValidator implements PDFSignatureValidator {
/**
* Constructor.
*/
public BasicPDFSignatureValidator() {
}
/** {@inheritDoc} */
@Override
public List validate(final byte[] document) throws SignatureException {
try (final PDDocument pdfDocument = Loader.loadPDF(document)) {
final List results = new ArrayList<>();
final List signatureDictionaries = pdfDocument.getSignatureDictionaries();
for (final PDSignature signature : signatureDictionaries) {
log.debug("Validating PDF signature [name:{}] ...", signature.getName());
final PDFSignatureValidationResult result = this.validatePdfSignature(document, signature);
results.add(result);
log.debug("PDF signature validation result for [name:{}]: {}", signature.getName(), result.toString());
}
return results;
}
catch (final IOException e) {
final String msg = String.format("Internal error while verifying PDF signature - %s", e.getMessage());
log.error("{}", msg, e);
throw new SignatureException(msg, e);
}
}
/**
* Validates the supplied signature.
*
* @param document the PDF document holding the signature
* @param signature the signature
* @return a validation result
*/
protected PDFSignatureValidationResult validatePdfSignature(final byte[] document, final PDSignature signature) {
final DefaultPDFSignatureValidationResult result = new DefaultPDFSignatureValidationResult();
result.setPdfSignature(signature);
try {
final byte[] signedContentBytes = signature.getSignedContent(new ByteArrayInputStream(document));
final byte[] signedDataBytes = signature.getContents(new ByteArrayInputStream(document));
final CMSSignedDataParser signedDataParser = new CMSSignedDataParser(new BcDigestCalculatorProvider(),
new CMSTypedStream(new ByteArrayInputStream(signedContentBytes)), signedDataBytes);
final CMSTypedStream signedContent = signedDataParser.getSignedContent();
signedContent.drain();
// Get hold of all certificates found in the document.
//
final CollectionStore> certStore = (CollectionStore>) signedDataParser.getCertificates();
final Iterator> ci = certStore.iterator();
final List certList = new ArrayList<>();
while (ci.hasNext()) {
final X509CertificateHolder ch = (X509CertificateHolder) ci.next();
certList.add(CertificateUtils.decodeCertificate(ch.getEncoded()));
}
result.setAdditionalCertificates(certList);
// Extract the signer information
//
final SignerInformationStore signerInformationStore = signedDataParser.getSignerInfos();
final Collection signerInformationCollection = signerInformationStore.getSigners();
if (signerInformationCollection.isEmpty()) {
throw new CMSException("No SignerInformation available");
}
else if (signerInformationCollection.size() > 1) {
throw new CMSException("More than one SignerInformation available");
}
final SignerInformation signerInformation = signerInformationCollection.iterator().next();
// Claimed signing time
//
final Date claimedSigningTime = getClaimedSigningTime(signerInformation);
if (claimedSigningTime != null) {
result.setClaimedSigningTime(claimedSigningTime);
}
else {
final Calendar dictionarySignDate = signature.getSignDate();
if (dictionarySignDate != null) {
log.debug("Claimed signing time was not present in signer information using time from PDF dictionary");
result.setClaimedSigningTime(dictionarySignDate.getTime());
}
else {
log.warn("Claimed signing time was not present in signer information OR the PDF dictionary");
}
}
// Get hold of signer certificate ...
//
final Collection> certCollection = certStore.getMatches(signerInformation.getSID());
if (certCollection.isEmpty()) {
throw new CMSException("No signer certificate available");
}
final X509CertificateHolder certHolder = (X509CertificateHolder) certCollection.iterator().next();
result.setSignerCertificate(CertificateUtils.decodeCertificate(certHolder.getEncoded()));
// Verify the signature
//
final SignerInformationVerifier signerInformationVerifier =
new JcaSimpleSignerInfoVerifierBuilder().setProvider("BC").build(certHolder);
try {
final boolean signatureVerificationResult = signerInformation.verify(signerInformationVerifier);
if (!signatureVerificationResult) {
result.setError(Status.ERROR_INVALID_SIGNATURE, "Invalid PDF signature");
return result;
}
}
catch (final CMSVerifierCertificateNotValidException e) {
result.setError(Status.ERROR_SIGNER_INVALID, "Signing certificate was not valid at time of signing", e);
return result;
}
// Collect signature algorithm data
//
final String digestAlgoOid = signerInformation.getDigestAlgOID();
final String encryptionAlgorithmOid = signerInformation.getEncryptionAlgOID();
final String signatureAlgorithm = PDFAlgorithmRegistry.getAlgorithmURI(
new ASN1ObjectIdentifier(encryptionAlgorithmOid), new ASN1ObjectIdentifier(digestAlgoOid));
result.setSignatureAlgorithm(signatureAlgorithm);
// Check if the CMS algorithm protection attribute has been set, and if so, assert algorithm consistency.
//
final Attribute cmsAlgorithmProtection = signerInformation.getSignedAttributes()
.get(new ASN1ObjectIdentifier(PDFObjectIdentifiers.ID_AA_CMS_ALGORITHM_PROTECTION));
if (cmsAlgorithmProtection != null) {
final Pair cmsAlgorithmProtectionAlgs =
getCmsAlgoritmProtectionData(cmsAlgorithmProtection);
final AlgorithmIdentifier protSignAlgo = cmsAlgorithmProtectionAlgs.getFirst();
final AlgorithmIdentifier protHashAlgo = cmsAlgorithmProtectionAlgs.getSecond();
result.setCmsAlgorithmProtection(true);
// Make sure the algorithms are consistent with our signature algorithm.
//
final SignatureAlgorithm algorithmProperties = PDFAlgorithmRegistry.getAlgorithmProperties(signatureAlgorithm);
if (!algorithmProperties.getAlgorithmIdentifier().getAlgorithm().equals(protSignAlgo.getAlgorithm())) {
final String msg = String.format(
"CMS algorithm protection signature algorithm (%s) is not consistent with signature algorithm used to sign data (%s)",
protSignAlgo.getAlgorithm().getId(), signatureAlgorithm);
result.setError(Status.ERROR_INVALID_SIGNATURE, msg);
return result;
}
if (!algorithmProperties.getMessageDigestAlgorithm().getAlgorithmIdentifier().getAlgorithm()
.equals(protHashAlgo.getAlgorithm())) {
final String msg = String.format(
"CMS algorithm protection hash algorithm (%s) is not consistent with signature algorithm used to sign data (%s)",
protHashAlgo.getAlgorithm().getId(), signatureAlgorithm);
result.setError(Status.ERROR_INVALID_SIGNATURE, msg);
return result;
}
log.debug(
"CMS algorithm protection attribute attributes ('{}', '{}') are consistent with signature algorithm used ('{}')",
protSignAlgo.getAlgorithm().getId(), protHashAlgo.getAlgorithm().getId(), signatureAlgorithm);
}
// Check if this is a PAdES signature, and if so, validate the PAdES properties.
//
result.setEtsiAdes(this.verifyPadesProperties(signature, signerInformation, result.getSignerCertificate()));
result.setStatus(Status.SUCCESS);
}
catch (final InternalSignatureValidationException e) {
log.error("{}", e.getMessage(), e);
result.setError(e);
}
catch (final IOException | CMSException | CertificateException | NoSuchAlgorithmException |
OperatorCreationException e) {
final String msg = String.format("PDF signature validation processing error - %s", e.getMessage());
log.error("{}", msg, e);
result.setError(Status.ERROR_BAD_FORMAT, msg, e);
}
return result;
}
/**
* Verifies PAdES properties.
*
* @param signature the PDF signature
* @param signerInformation the signer information
* @param signerCertificate the signer certificate
* @return true if this is a PAdES signature, and it was successfully validated and false if this is not a PAdES
* signature
* @throws InternalSignatureValidationException for PAdES validation errors
*/
private boolean verifyPadesProperties(final PDSignature signature, final SignerInformation signerInformation,
final X509Certificate signerCertificate) throws InternalSignatureValidationException {
try {
final String subFilter = signature.getSubFilter();
if (PDSignature.SUBFILTER_ETSI_CADES_DETACHED.getName().equals(subFilter)) {
final PDFBoxSignatureUtils.SignedCertRef signedCertificateRef = PDFBoxSignatureUtils.getSignedCertRefAttribute(
signerInformation.getSignedAttributes().toASN1Structure().getEncoded(ASN1Encoding.DER));
if (signedCertificateRef == null) {
throw new InternalSignatureValidationException(Status.ERROR_BAD_FORMAT,
"Signature subfilter indicates that the signature is a PAdES signature but no signed certificate reference is present");
}
final MessageDigest md = MessageDigest.getInstance(signedCertificateRef.getHashAlgorithm().getId());
final byte[] certificateHash = md.digest(signerCertificate.getEncoded());
if (!Arrays.equals(certificateHash, signedCertificateRef.getSignedCertHash())) {
throw new InternalSignatureValidationException(Status.ERROR_INVALID_SIGNATURE,
"PAdES signed certificate reference mismatch");
}
log.debug("PAdES signature successfully verified");
return true;
}
else {
log.debug("No PAdES signature available");
return false;
}
}
catch (final IOException | NoSuchAlgorithmException | CertificateEncodingException e) {
final String msg = String.format("Failure verifying PAdES signature - %s", e.getMessage());
throw new InternalSignatureValidationException(Status.ERROR_BAD_FORMAT, msg, e);
}
}
/**
* Obtains the claimed signing time from signed attributes.
*
* @param signer signer information
* @return claimed signing time if present or null if absent
*/
private static Date getClaimedSigningTime(final SignerInformation signer) {
final AttributeTable signedAttributes = signer.getSignedAttributes();
final Attribute sigTimeAttr = signedAttributes.get(new ASN1ObjectIdentifier("1.2.840.113549.1.9.5"));
if (sigTimeAttr == null) {
return null;
}
final ASN1Encodable[] attributeValues = sigTimeAttr.getAttributeValues();
if (attributeValues.length > 0 && attributeValues[0] instanceof ASN1UTCTime) {
try {
return ((ASN1UTCTime) attributeValues[0]).getDate();
}
catch (final ParseException e) {
log.error("Imvalid format for claimed signing time - {}", e.getMessage(), e);
return null;
}
}
else {
return null;
}
}
/**
* Gets the signature and hash algorithms from the CMS algorithm protection attribute.
*
* @param cmsAlgorithmProtectionAttribute protection attribute
* @return a pair of the signature algorithm ID and the hash algoritm id
* @throws InternalSignatureValidationException for invalid CMS algorithm protection attributes
*/
private static Pair getCmsAlgoritmProtectionData(
final Attribute cmsAlgorithmProtectionAttribute) throws InternalSignatureValidationException {
try {
final ASN1Sequence cmsapSeq =
ASN1Sequence.getInstance(cmsAlgorithmProtectionAttribute.getAttrValues().getObjectAt(0));
// Get Hash algorithm
final AlgorithmIdentifier hashAlgorithmId = AlgorithmIdentifier.getInstance(cmsapSeq.getObjectAt(0));
if (hashAlgorithmId == null) {
throw new InternalSignatureValidationException(
Status.ERROR_BAD_FORMAT, "Missing hash algorithm in CMS algorithm protection attribute");
}
// Get Signature algorithm
AlgorithmIdentifier signatureAlgorithmId = null;
for (int objIdx = 1; objIdx < cmsapSeq.size(); objIdx++) {
final ASN1Encodable asn1Encodable = cmsapSeq.getObjectAt(objIdx);
if (asn1Encodable instanceof ASN1TaggedObject) {
final ASN1TaggedObject taggedObj = ASN1TaggedObject.getInstance(asn1Encodable);
if (taggedObj.getTagNo() == 1) {
signatureAlgorithmId = AlgorithmIdentifier.getInstance(taggedObj, false);
break;
}
}
}
if (signatureAlgorithmId == null) {
throw new InternalSignatureValidationException(
Status.ERROR_BAD_FORMAT, "Missing signature algorithm in CMS algorithm protection attribute");
}
return new Pair<>(signatureAlgorithmId, hashAlgorithmId);
}
catch (final IllegalArgumentException e) {
throw new InternalSignatureValidationException(
Status.ERROR_BAD_FORMAT, "Error processing CMS algorithm protection attribute - " + e.getMessage(), e);
}
}
/** {@inheritDoc} */
@Override
public boolean isSigned(final byte[] document) throws IllegalArgumentException {
try (final PDDocument pdfDocument = Loader.loadPDF(document)) {
return !pdfDocument.getSignatureDictionaries().isEmpty();
}
catch (final IOException e) {
throw new IllegalArgumentException("Invalid document", e);
}
}
/**
* The basic implementation will always return an empty list.
*/
@Override
public List getRequiredSignerCertificates() {
return Collections.emptyList();
}
/**
* This basic implementation will always return {@code null}.
*/
@Override
public CertificateValidator getCertificateValidator() {
return null;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy