ApiKeyUtilities.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.pdd.core.autenticazione;
import java.util.List;
import org.openspcoop2.core.config.ServizioApplicativo;
import org.openspcoop2.core.config.driver.db.DriverConfigurazioneDB;
import org.openspcoop2.core.id.IDServizioApplicativo;
import org.openspcoop2.core.id.IDSoggetto;
import org.openspcoop2.core.registry.Soggetto;
import org.openspcoop2.core.registry.driver.db.DriverRegistroServiziDB;
import org.openspcoop2.pdd.core.PdDContext;
import org.openspcoop2.pdd.core.connettori.InfoConnettoreIngresso;
import org.openspcoop2.utils.crypt.PasswordGenerator;
import org.openspcoop2.utils.io.Base64Utilities;
/**
* ApiKeyUtilities
*
* @author Andrea Poli (apoli@link.it)
* @author $Author$
* @version $Rev$, $Date$
*/
public class ApiKeyUtilities {
public static String getKey(boolean apiKey,
boolean header, boolean cookie, boolean queryParameter,
String nomeHeader, String nomeCookie, String nomeQueryParameter,
InfoConnettoreIngresso infoConnettore, PdDContext pddContext, boolean throwException,
StringBuilder fullCredential) throws AutenticazioneException {
String tipo = apiKey ? "ApiKey" : "AppId";
String key = null;
// viene usato l'ordine della specifica
if(queryParameter) {
if(nomeQueryParameter==null && throwException) {
throw new AutenticazioneException("Nome del parametro della query, da cui estrarre l'"+tipo+", non indicato");
}
if(nomeQueryParameter!=null && infoConnettore!=null && infoConnettore.getUrlProtocolContext()!=null) {
key = infoConnettore.getUrlProtocolContext().getParameterFirstValue(nomeQueryParameter);
if(key!=null && "".equals(key.trim())) {
key = null;
}
}
}
if(key!=null) {
if(fullCredential.length()>0) {
fullCredential.append("\n");
}
fullCredential.append(tipo).append(" (query) '").append(nomeQueryParameter);
if(!apiKey) { // l'api key non deve essere tracciata come per la password.
fullCredential.append(": ").append(key);
}
fullCredential.append("'");
return key;
}
if(header) {
if(nomeHeader==null && throwException) {
throw new AutenticazioneException("Nome dell'header, da cui estrarre l'"+tipo+", non indicato");
}
if(nomeHeader!=null && infoConnettore!=null && infoConnettore.getUrlProtocolContext()!=null) {
key = infoConnettore.getUrlProtocolContext().getHeaderFirstValue(nomeHeader);
if(key!=null && "".equals(key.trim())) {
key = null;
}
}
}
if(key!=null) {
if(fullCredential.length()>0) {
fullCredential.append("\n");
}
fullCredential.append(tipo).append(" (http) '").append(nomeHeader);
if(!apiKey) { // l'api key non deve essere tracciata come per la password.
fullCredential.append(": ").append(key);
}
fullCredential.append("'");
return key;
}
if(cookie) {
if(nomeCookie==null && throwException) {
throw new AutenticazioneException("Nome del cookie, da cui estrarre l'"+tipo+", non indicato");
}
if(nomeCookie!=null && infoConnettore!=null && infoConnettore.getUrlProtocolContext()!=null) {
key = infoConnettore.getUrlProtocolContext().getCookieValue(nomeCookie);
if(key!=null && "".equals(key.trim())) {
key = null;
}
}
}
if(key!=null) {
if(fullCredential.length()>0) {
fullCredential.append("\n");
}
fullCredential.append(tipo).append(" (cookie) '").append(nomeCookie);
if(!apiKey) { // l'api key non deve essere tracciata come per la password.
fullCredential.append(": ").append(key);
}
fullCredential.append("'");
return key;
}
if(throwException) {
throw new AutenticazioneException(tipo+" non presente nella richiesta");
}
return null;
}
private static final String SOGGETTO_SEPARATOR = ".";
public static final String APPLICATIVO_SOGGETTO_SEPARATOR = "@";
private static final int MAX_ALREADY_EXISTS = 5000;
private static final String SEPARATOR_ALREADY_EXISTS = ".";
public static String toAppId(String protocollo, IDSoggetto idSoggetto, boolean multipleApiKeys, DriverRegistroServiziDB driver) throws Exception {
String appId = _toAppId(protocollo, idSoggetto);
if(!_existsAppId(appId, multipleApiKeys, idSoggetto, driver)) {
return appId;
}
for (int i = 2; i < MAX_ALREADY_EXISTS; i++) {
IDSoggetto idSoggettoAlreadyExists = new IDSoggetto(idSoggetto.getTipo(), idSoggetto.getNome()+SEPARATOR_ALREADY_EXISTS+i);
String appIdAlreadyExists = _toAppId(protocollo, idSoggettoAlreadyExists);
if(!_existsAppId(appIdAlreadyExists, multipleApiKeys, idSoggetto, driver)) {
return appIdAlreadyExists;
}
}
throw new Exception("Generazione appId univoco non riuscita dopo "+MAX_ALREADY_EXISTS+" tentativi");
}
private static String _toAppId(String protocollo, IDSoggetto idSoggetto) throws Exception {
// non va bene perchè due nomi uguali di due protocolli differenti, vengono identici.
//return org.openspcoop2.protocol.engine.utils.NamingUtils.getLabelSoggetto(protocollo, idSoggetto.getTipo(), idSoggetto.getNome());
if(idSoggetto==null || idSoggetto.getTipo()==null || idSoggetto.getNome()==null) {
throw new Exception("Identificativo soggetto non definito");
}
return idSoggetto.getNome()+SOGGETTO_SEPARATOR+idSoggetto.getTipo();
}
private static boolean _existsAppId(String appId, boolean multipleApiKeys, IDSoggetto idSoggetto, DriverRegistroServiziDB driver) throws Exception {
Soggetto soggetto = driver.soggettoWithCredenzialiApiKey(appId, multipleApiKeys);
if(soggetto==null) {
return false;
}
boolean isSame = false;
if(soggetto.getTipo().equals(idSoggetto.getTipo()) &&
soggetto.getNome().equals(idSoggetto.getNome()) ) {
isSame = true;
}
return !isSame;
}
public static String toAppId(String protocollo, IDServizioApplicativo idSA, boolean multipleApiKeys, DriverConfigurazioneDB driver) throws Exception {
String appId = _toAppId(protocollo, idSA.getNome(), idSA.getIdSoggettoProprietario());
if(!_existsAppId(appId, multipleApiKeys, idSA, driver)) {
return appId;
}
for (int i = 2; i < MAX_ALREADY_EXISTS; i++) {
String appIdAlreadyExists = _toAppId(protocollo, idSA.getNome()+SEPARATOR_ALREADY_EXISTS+i, idSA.getIdSoggettoProprietario());
if(!_existsAppId(appIdAlreadyExists, multipleApiKeys, idSA, driver)) {
return appIdAlreadyExists;
}
}
throw new Exception("Generazione appId univoco non riuscita dopo "+MAX_ALREADY_EXISTS+" tentativi");
}
private static String _toAppId(String protocollo, String nomeSA, IDSoggetto idSoggetto) throws Exception {
return nomeSA+APPLICATIVO_SOGGETTO_SEPARATOR+ _toAppId(protocollo, idSoggetto);
}
private static boolean _existsAppId(String appId, boolean multipleApiKeys, IDServizioApplicativo idSA, DriverConfigurazioneDB driver) throws Exception {
List<ServizioApplicativo> saList = driver.servizioApplicativoWithCredenzialiApiKeyList(appId, multipleApiKeys);
if (saList==null || saList.isEmpty()) {
return false;
}
boolean isSame = false;
for (ServizioApplicativo servizioApplicativo : saList) {
if(!servizioApplicativo.getNome().equals(idSA.getNome())) {
continue;
}
if(!servizioApplicativo.getTipoSoggettoProprietario().equals(idSA.getIdSoggettoProprietario().getTipo())) {
continue;
}
if(!servizioApplicativo.getNomeSoggettoProprietario().equals(idSA.getIdSoggettoProprietario().getNome())) {
continue;
}
isSame = true;
break;
}
return !isSame;
}
private static final String PREFIX_SEPARATOR_API_KEY = ".";
private static final String PREFIX_SEPARATOR_API_KEY_REGEXP = "\\.";
private static String getPrefixForApiKey(String protocollo, IDSoggetto idSoggetto, DriverRegistroServiziDB driver) throws Exception {
boolean multipleApiKeys = false;
String appId = toAppId(protocollo, idSoggetto, multipleApiKeys, driver);
return Base64Utilities.encodeAsString(appId.getBytes())+PREFIX_SEPARATOR_API_KEY;
}
public static ApiKey newApiKey(String protocollo, IDSoggetto idSoggetto, int length, DriverRegistroServiziDB driver) throws Exception {
ApiKey apiKey = new ApiKey();
String password = getPassword(length);
String apiKeyToken = getPrefixForApiKey(protocollo, idSoggetto, driver)+Base64Utilities.encodeAsString(password.getBytes());
apiKey.setApiKey(apiKeyToken);
apiKey.setPassword(password);
return apiKey;
}
private static String getPrefixForApiKey(String protocollo, IDServizioApplicativo idSA, DriverConfigurazioneDB driver) throws Exception {
boolean multipleApiKeys = false;
String appId = toAppId(protocollo, idSA, multipleApiKeys, driver);
return Base64Utilities.encodeAsString(appId.getBytes())+PREFIX_SEPARATOR_API_KEY;
}
public static ApiKey newApiKey(String protocollo, IDServizioApplicativo idSA, int length, DriverConfigurazioneDB driver) throws Exception {
ApiKey apiKey = new ApiKey();
String password = getPassword(length);
String apiKeyToken = getPrefixForApiKey(protocollo, idSA, driver)+Base64Utilities.encodeAsString(password.getBytes());
apiKey.setApiKey(apiKeyToken);
apiKey.setPassword(password);
return apiKey;
}
public static String encodeApiKey(String appId, String password) throws Exception {
return Base64Utilities.encodeAsString(appId.getBytes())+
PREFIX_SEPARATOR_API_KEY+
Base64Utilities.encodeAsString(password.getBytes());
}
public static String[] decodeApiKey(String apiKeyBase64) throws Exception {
if(!apiKeyBase64.contains(PREFIX_SEPARATOR_API_KEY)) {
throw new Exception("Formato non corretto");
}
String [] tmp = apiKeyBase64.split(PREFIX_SEPARATOR_API_KEY_REGEXP);
if(tmp==null || tmp.length!=2){
throw new Exception("Formato non corretto (.)");
}
try {
tmp[0] = new String(Base64Utilities.decode(tmp[0]));
}catch(Exception e) {
throw new Exception("Formato non corretto (appId)",e);
}
try {
tmp[1] = new String(Base64Utilities.decode(tmp[1]));
}catch(Exception e) {
throw new Exception("Formato non corretto (appKey)",e);
}
return tmp;
}
public static ApiKey newMultipleApiKey(int length) throws Exception {
ApiKey apiKey = new ApiKey();
String password = getPassword(length);
String apiKeyToken = encodeMultipleApiKey(password);
apiKey.setApiKey(apiKeyToken);
apiKey.setPassword(password);
return apiKey;
}
public static String decodeMultipleApiKey(String apiKeyBase64) throws Exception {
return new String(Base64Utilities.decode(apiKeyBase64));
}
public static String encodeMultipleApiKey(String password) throws Exception {
return Base64Utilities.encodeAsString(password.getBytes());
}
private static String getPassword(int length) throws Exception {
PasswordGenerator pwdGenerator = PasswordGenerator.DEFAULT;
return pwdGenerator.generate(length);
}
}