PKCS7Signature.java
/*
* GovWay - A customizable API Gateway
* https://govway.org
*
* Copyright (c) 2005-2025 Link.it srl (https://link.it).
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 3, as published by
* the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
package org.openspcoop2.utils.security;
import java.security.PrivateKey;
import java.security.Provider;
import java.security.Security;
import java.security.cert.Certificate;
import java.util.ArrayList;
import java.util.List;
import javax.xml.crypto.dsig.SignatureMethod;
import org.bouncycastle.cert.X509CertificateHolder;
import org.bouncycastle.cert.jcajce.JcaCertStore;
import org.bouncycastle.cms.CMSProcessableByteArray;
import org.bouncycastle.cms.CMSSignedData;
import org.bouncycastle.cms.CMSSignedDataGenerator;
import org.bouncycastle.cms.CMSTypedData;
import org.bouncycastle.cms.jcajce.JcaSignerInfoGeneratorBuilder;
import org.bouncycastle.operator.ContentSigner;
import org.bouncycastle.operator.jcajce.JcaContentSignerBuilder;
import org.bouncycastle.operator.jcajce.JcaDigestCalculatorProviderBuilder;
import org.bouncycastle.util.Store;
import org.openspcoop2.utils.UtilsException;
import org.openspcoop2.utils.certificate.KeyStore;
import org.openspcoop2.utils.certificate.KeystoreType;
/**
* Signature
*
* @author Poli Andrea (apoli@link.it)
* @author $Author$
* @version $Rev$, $Date$
*/
public class PKCS7Signature {
public static final String DEFAULT_SIGNATURE_METHOD = SignatureMethod.RSA_SHA256;
private PrivateKey privateKey;
private Certificate certificate;
private Provider provider;
public PKCS7Signature(KeyStore keystore, String alias, String passwordPrivateKey) throws UtilsException{
boolean useKeystoreProvider = false;
if(keystore!=null && KeystoreType.PKCS11.getNome().equalsIgnoreCase(keystore.getKeystoreType())) {
useKeystoreProvider = true;
// Prendo il provider dal keystore per PKCS11, il cui provider implementa l'algoritmo di firma specifica per la chiave memorizzata nel dispositivo.
}
init(keystore, alias, passwordPrivateKey, useKeystoreProvider);
}
public PKCS7Signature(KeyStore keystore, String alias, String passwordPrivateKey, boolean useKeystoreProvider) throws UtilsException{
init(keystore, alias, passwordPrivateKey, useKeystoreProvider);
}
public PKCS7Signature(KeyStore keystore, String alias, String passwordPrivateKey, boolean useBouncyCastle, boolean addBouncyCastleProvider) throws UtilsException{
this(keystore, alias, passwordPrivateKey);
if(useBouncyCastle) {
if(addBouncyCastleProvider) {
this.provider = Security.getProvider(org.bouncycastle.jce.provider.BouncyCastleProvider.PROVIDER_NAME);
if(this.provider==null) {
Security.addProvider(new org.bouncycastle.jce.provider.BouncyCastleProvider());
this.provider = Security.getProvider(org.bouncycastle.jce.provider.BouncyCastleProvider.PROVIDER_NAME);
}
}
else {
this.provider = Security.getProvider(org.bouncycastle.jce.provider.BouncyCastleProvider.PROVIDER_NAME);
}
}
}
private void init(KeyStore keystore, String alias, String passwordPrivateKey, boolean useKeystoreProvider) throws UtilsException {
if(keystore==null) {
throw new UtilsException("Keystore undefined");
}
this.privateKey = keystore.getPrivateKey(alias, passwordPrivateKey);
this.certificate = keystore.getCertificate(alias);
if(useKeystoreProvider) {
this.provider = keystore.getKeystoreProvider();
// Prendere il provider dal keystore serve per PKCS11 il cui provider implementa l'algoritmo di firma specifica per la chiave memorizzata nel dispositivo.
// Per gli altri tipi di keystore, nei quali tipicamente il provider è la SUN, non deve essere usata questa opzione altrimenti si ottiene l'errore:
// no such algorithm: SHA256WITHRSA for provider SUN
// Per quei tipi di keystore devono essere usati i costruttori in cui viene usato bouncy castle
// o il costruttore senza parametri relativi al provider, il quale utilizza quello il provider più corretto preente nella JVM.
}
}
public byte[] sign(String data, String charsetName, String algorithm) throws UtilsException{
try{
return this.sign(data.getBytes(charsetName), algorithm);
}catch(Exception e){
throw new UtilsException(e.getMessage(),e);
}
}
public byte[] sign(byte[] data, String algorithm) throws UtilsException{
try{
if(algorithm==null) {
algorithm = DEFAULT_SIGNATURE_METHOD;
}
List<X509CertificateHolder> certList = new ArrayList<>();
CMSTypedData msg = new CMSProcessableByteArray(data); // Data to sign
X509CertificateHolder cert = new X509CertificateHolder(this.certificate.getEncoded());
certList.add(cert); // Adding the X509 Certificate
Store<?> certs = new JcaCertStore(certList);
CMSSignedDataGenerator gen = new CMSSignedDataGenerator();
// Initializing the the BC's Signer
JcaContentSignerBuilder cs = new JcaContentSignerBuilder(algorithm);
if(this.provider!=null) {
cs.setProvider(this.provider);
}
ContentSigner sha1Signer = cs.build(this.privateKey);
JcaDigestCalculatorProviderBuilder builder = new JcaDigestCalculatorProviderBuilder();
if(this.provider!=null) {
builder.setProvider(this.provider);
}
gen.addSignerInfoGenerator(
new JcaSignerInfoGeneratorBuilder(builder.build())
.build(sha1Signer, cert));
// adding the certificate
gen.addCertificates(certs);
// Getting the signed data
CMSSignedData sigData = gen.generate(msg, true);
return sigData.getEncoded();
}catch(Exception e){
throw new UtilsException(e.getMessage(),e);
}
}
}