CRLCertstore.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.certificate;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.Serializable;
import java.security.cert.CertStore;
import java.security.cert.CertificateFactory;
import java.security.cert.CollectionCertStoreParameters;
import java.security.cert.X509CRL;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import org.apache.commons.lang.StringUtils;
import org.openspcoop2.utils.Utilities;
import org.openspcoop2.utils.UtilsException;
/**
* CRLCertstore
*
* @author Andrea Poli (apoli@link.it)
* @author $Author$
* @version $Rev$, $Date$
*/
public class CRLCertstore implements Serializable {
/**
*
*/
private static final long serialVersionUID = 1L;
private List<byte []> crlBytes = null;
private List<String> crlPaths = null;
private transient CertificateFactory certFactory = null;
private transient List<X509CRL> caCrls = null;
private transient CertStore certStore = null;
@Override
public String toString() {
StringBuilder bf = new StringBuilder();
bf.append("CRLCertstore (");
boolean first = true;
if(this.crlPaths!=null) {
for (String crlPath : this.crlPaths) {
if(!first) {
bf.append(", ");
}
bf.append(crlPath);
first=false;
}
}
else {
bf.append("Nessuna crl definita");
}
bf.append(")");
return bf.toString();
}
public static List<String> readCrlPaths(String crlPaths){
List<String> crlPathsList = new ArrayList<>();
if(crlPaths!=null && StringUtils.isNotEmpty(crlPaths)) {
if(crlPaths.contains(",")) {
String [] tmp = crlPaths.split(",");
for (String crlPath : tmp) {
crlPathsList.add(crlPath.trim());
}
}
else {
crlPathsList.add(crlPaths.trim());
}
}
return crlPathsList;
}
public static String convertToCrlPaths(List<String> crlPathsList) {
StringBuilder sb = new StringBuilder();
if(crlPathsList==null || crlPathsList.isEmpty()) {
return null;
}
for (String path : crlPathsList) {
if(sb.length()>0) {
sb.append(",");
}
sb.append(path);
}
return sb.toString();
}
public CRLCertstore(String crlPaths) throws UtilsException {
this(crlPaths, null);
}
public CRLCertstore(String crlPaths, Map<String, byte[]> localResources) throws UtilsException {
List<String> crlPathsList = readCrlPaths(crlPaths);
this.initEngine(crlPathsList, localResources);
}
public CRLCertstore(List<String> crlPaths) throws UtilsException{
this(crlPaths, null);
}
public CRLCertstore(List<String> crlPaths, Map<String, byte[]> localResources) throws UtilsException{
this.initEngine(crlPaths, localResources);
}
private void initEngine(List<String> crlPaths, Map<String, byte[]> localResources) throws UtilsException{
try{
if(crlPaths==null || crlPaths.isEmpty()){
throw new UtilsException("crlPaths non indicato");
}
this.crlPaths = crlPaths;
this.crlBytes = new ArrayList<>();
for (String crlPath : crlPaths) {
if(localResources!=null && !localResources.isEmpty() && localResources.containsKey(crlPath)) {
byte[] r = localResources.get(crlPath);
if(r!=null && r.length>0) {
this.crlBytes.add(r);
}
continue;
}
initEngineCrl(crlPath);
}
this.initCRL();
}catch(Exception e){
throw new UtilsException(e.getMessage(),e);
}
}
private void initEngineCrl(String crlPath) throws UtilsException, IOException {
byte[] crlBytesAdd = null;
File fStore = new File(crlPath);
boolean fStoreExists = fStore.exists();
try(InputStream isStore = fStoreExists ? new FileInputStream(fStore) : CRLCertstore.class.getResourceAsStream(crlPath);){
if(isStore!=null) {
crlBytesAdd = Utilities.getAsByteArray(isStore);
}
}
if(crlBytesAdd==null && !fStoreExists) {
try(InputStream isStore = CRLCertstore.class.getResourceAsStream("/"+crlPath);){
if(isStore!=null) {
crlBytesAdd = Utilities.getAsByteArray(isStore);
}
}
}
if(crlBytesAdd==null) {
throw new UtilsException("Store ["+crlPath+"] not found");
}
this.crlBytes.add(crlBytesAdd);
}
private void checkInit() throws UtilsException{
if(this.caCrls==null) {
this.initCRL();
}
}
private synchronized void initCRL() throws UtilsException{
if(this.caCrls==null) {
// create a X509 certificate factory for later use
try {
this.certFactory = org.openspcoop2.utils.certificate.CertificateFactory.getCertificateFactory();
}catch(Exception e){
throw new UtilsException("Error getInstance CertificateFactory: "+e.getMessage(),e);
}
this.caCrls = new ArrayList<>();
for (int i = 0; i < this.crlBytes.size(); i++) {
byte [] crl = this.crlBytes.get(i);
try(ByteArrayInputStream bin = new ByteArrayInputStream(crl)){
X509CRL caCrl = (X509CRL) this.certFactory.generateCRL(bin);
this.caCrls.add(caCrl);
}
catch(Exception e){
throw new UtilsException("Error loading CRL '"+this.crlPaths.get(i)+"': "+e.getMessage(),e);
}
}
try {
CollectionCertStoreParameters certStoreParams =
new CollectionCertStoreParameters(this.caCrls);
this.certStore =
CertStore.getInstance("Collection", certStoreParams);
}catch(Exception e){
throw new UtilsException("Build CertStore failed: "+e.getMessage(),e);
}
}
}
public CertificateFactory getCertFactory() throws UtilsException {
this.checkInit(); // per ripristino da Serializable
return this.certFactory;
}
public List<X509CRL> getCaCrls() throws UtilsException {
this.checkInit(); // per ripristino da Serializable
return this.caCrls;
}
public CertStore getCertStore() throws UtilsException {
this.checkInit(); // per ripristino da Serializable
return this.certStore;
}
public int countCrls() {
return this.crlBytes!=null ? this.crlBytes.size() : 0;
}
}