mirror of https://github.com/apache/nifi.git
NIFI-9035 Refactored isKeystoreValid() to avoid NullPointerException
- Removed unnecessary debug logging from StandardTlsConfiguration - Replaced internal string labels with StoreType enum Signed-off-by: Matthew Burgess <mattyb149@apache.org> This closes #5297
This commit is contained in:
parent
2e1f276f06
commit
f27cd012f3
|
@ -18,8 +18,6 @@ package org.apache.nifi.security.util;
|
||||||
|
|
||||||
import org.apache.nifi.util.NiFiProperties;
|
import org.apache.nifi.util.NiFiProperties;
|
||||||
import org.apache.nifi.util.StringUtils;
|
import org.apache.nifi.util.StringUtils;
|
||||||
import org.slf4j.Logger;
|
|
||||||
import org.slf4j.LoggerFactory;
|
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.net.MalformedURLException;
|
import java.net.MalformedURLException;
|
||||||
|
@ -28,6 +26,7 @@ import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
import java.net.URL;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -36,8 +35,6 @@ import java.util.Properties;
|
||||||
* building {@link javax.net.ssl.SSLContext}s.
|
* building {@link javax.net.ssl.SSLContext}s.
|
||||||
*/
|
*/
|
||||||
public class StandardTlsConfiguration implements TlsConfiguration {
|
public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
private static final Logger logger = LoggerFactory.getLogger(StandardTlsConfiguration.class);
|
|
||||||
|
|
||||||
private static final String TLS_PROTOCOL_VERSION = TlsConfiguration.getHighestCurrentSupportedTlsProtocolVersion();
|
private static final String TLS_PROTOCOL_VERSION = TlsConfiguration.getHighestCurrentSupportedTlsProtocolVersion();
|
||||||
private static final String MASKED_PASSWORD_LOG = "********";
|
private static final String MASKED_PASSWORD_LOG = "********";
|
||||||
private static final String NULL_LOG = "null";
|
private static final String NULL_LOG = "null";
|
||||||
|
@ -192,22 +189,17 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
* @return a populated TlsConfiguration container object
|
* @return a populated TlsConfiguration container object
|
||||||
*/
|
*/
|
||||||
public static TlsConfiguration fromNiFiProperties(final Properties niFiProperties) {
|
public static TlsConfiguration fromNiFiProperties(final Properties niFiProperties) {
|
||||||
Objects.requireNonNull("The NiFi properties cannot be null");
|
Objects.requireNonNull(niFiProperties, "Properties required");
|
||||||
|
return new StandardTlsConfiguration(
|
||||||
String keystorePath = niFiProperties.getProperty(NiFiProperties.SECURITY_KEYSTORE);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_KEYSTORE),
|
||||||
String keystorePassword = niFiProperties.getProperty(NiFiProperties.SECURITY_KEYSTORE_PASSWD);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_KEYSTORE_PASSWD),
|
||||||
String keyPassword = niFiProperties.getProperty(NiFiProperties.SECURITY_KEY_PASSWD);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_KEY_PASSWD),
|
||||||
String keystoreType = niFiProperties.getProperty(NiFiProperties.SECURITY_KEYSTORE_TYPE);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_KEYSTORE_TYPE),
|
||||||
String truststorePath = niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE),
|
||||||
String truststorePassword = niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_PASSWD);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_PASSWD),
|
||||||
String truststoreType = niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_TYPE);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_TYPE),
|
||||||
String protocol = TLS_PROTOCOL_VERSION;
|
TLS_PROTOCOL_VERSION
|
||||||
|
);
|
||||||
final StandardTlsConfiguration tlsConfiguration = new StandardTlsConfiguration(keystorePath, keystorePassword, keyPassword,
|
|
||||||
keystoreType, truststorePath, truststorePassword,
|
|
||||||
truststoreType, protocol);
|
|
||||||
|
|
||||||
return tlsConfiguration;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -218,39 +210,19 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
* @param niFiProperties the NiFi properties
|
* @param niFiProperties the NiFi properties
|
||||||
* @return a populated TlsConfiguration container object
|
* @return a populated TlsConfiguration container object
|
||||||
*/
|
*/
|
||||||
public static StandardTlsConfiguration fromNiFiPropertiesTruststoreOnly(NiFiProperties niFiProperties) {
|
public static StandardTlsConfiguration fromNiFiPropertiesTruststoreOnly(final NiFiProperties niFiProperties) {
|
||||||
if (niFiProperties == null) {
|
Objects.requireNonNull(niFiProperties, "Properties required");
|
||||||
throw new IllegalArgumentException("The NiFi properties cannot be null");
|
return new StandardTlsConfiguration(
|
||||||
}
|
null,
|
||||||
|
null,
|
||||||
String truststorePath = niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE);
|
null,
|
||||||
String truststorePassword = niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_PASSWD);
|
null,
|
||||||
String truststoreType = niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_TYPE);
|
niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE),
|
||||||
String protocol = TLS_PROTOCOL_VERSION;
|
niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_PASSWD),
|
||||||
|
niFiProperties.getProperty(NiFiProperties.SECURITY_TRUSTSTORE_TYPE),
|
||||||
final StandardTlsConfiguration tlsConfiguration = new StandardTlsConfiguration(null, null, null, null, truststorePath, truststorePassword,
|
TLS_PROTOCOL_VERSION);
|
||||||
truststoreType, protocol);
|
|
||||||
if (logger.isDebugEnabled()) {
|
|
||||||
logger.debug("Instantiating TlsConfiguration from NiFi properties: null x4, {}, {}, {}, {}",
|
|
||||||
truststorePath, tlsConfiguration.getTruststorePasswordForLogging(), truststoreType, protocol);
|
|
||||||
}
|
|
||||||
|
|
||||||
return tlsConfiguration;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// /**
|
|
||||||
// * Returns {@code true} if the provided TlsConfiguration is {@code null} or <em>empty</em>
|
|
||||||
// * (i.e. neither any of the keystore nor truststore properties are populated).
|
|
||||||
// *
|
|
||||||
// * @param tlsConfiguration the container object to check
|
|
||||||
// * @return true if this container is empty or null
|
|
||||||
// */
|
|
||||||
// public static boolean isEmpty(org.apache.nifi.security.util.TlsConfiguration tlsConfiguration) {
|
|
||||||
// return tlsConfiguration == null || !(tlsConfiguration.isAnyKeystorePopulated() || tlsConfiguration.isAnyTruststorePopulated());
|
|
||||||
// }
|
|
||||||
|
|
||||||
// Getters & setters
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getKeystorePath() {
|
public String getKeystorePath() {
|
||||||
return keystorePath;
|
return keystorePath;
|
||||||
|
@ -350,7 +322,7 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean isKeystorePopulated() {
|
public boolean isKeystorePopulated() {
|
||||||
return isStorePopulated(keystorePath, keystorePassword, keystoreType, "keystore");
|
return isStorePopulated(keystorePath, keystorePassword, keystoreType, StoreType.KEY_STORE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -370,19 +342,12 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean isKeystoreValid() {
|
public boolean isKeystoreValid() {
|
||||||
boolean simpleCheck = isStoreValid(keystorePath, keystorePassword, keystoreType, "keystore");
|
if (isStoreValid(keystorePath, keystorePassword, keystoreType, StoreType.KEY_STORE)) {
|
||||||
if (simpleCheck) {
|
|
||||||
return true;
|
return true;
|
||||||
} else if (StringUtils.isNotBlank(keyPassword) && !keystorePassword.equals(keyPassword)) {
|
} else if (StringUtils.isNotBlank(keyPassword) && !keyPassword.equals(keystorePassword)) {
|
||||||
logger.debug("Simple keystore validity check failed; trying with separate key password");
|
return isKeystorePopulated()
|
||||||
try {
|
&& KeyStoreUtils.isKeyPasswordCorrect(getFileUrl(keystorePath), keystoreType, keystorePassword.toCharArray(),
|
||||||
return isKeystorePopulated()
|
getFunctionalKeyPassword().toCharArray());
|
||||||
&& KeyStoreUtils.isKeyPasswordCorrect(new File(keystorePath).toURI().toURL(), keystoreType, keystorePassword.toCharArray(),
|
|
||||||
getFunctionalKeyPassword().toCharArray());
|
|
||||||
} catch (MalformedURLException e) {
|
|
||||||
logger.error("Encountered an error validating the keystore: " + e.getLocalizedMessage());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -395,7 +360,7 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean isTruststorePopulated() {
|
public boolean isTruststorePopulated() {
|
||||||
return isStorePopulated(truststorePath, truststorePassword, truststoreType, "truststore");
|
return isStorePopulated(truststorePath, truststorePassword, truststoreType, StoreType.TRUST_STORE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -415,7 +380,7 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean isTruststoreValid() {
|
public boolean isTruststoreValid() {
|
||||||
return isStoreValid(truststorePath, truststorePassword, truststoreType, "truststore");
|
return isStoreValid(truststorePath, truststorePassword, truststoreType, StoreType.TRUST_STORE);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -462,21 +427,19 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
enabledProtocols.add(configuredProtocol);
|
enabledProtocols.add(configuredProtocol);
|
||||||
}
|
}
|
||||||
|
|
||||||
return enabledProtocols.toArray(new String[enabledProtocols.size()]);
|
return enabledProtocols.toArray(new String[0]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
StringBuilder sb = new StringBuilder("[TlsConfiguration]");
|
return "[TlsConfiguration]" + "keystorePath=" + keystorePath +
|
||||||
sb.append("keystorePath=").append(keystorePath);
|
",keystorePassword=" + getKeystorePasswordForLogging() +
|
||||||
sb.append(",keystorePassword=").append(getKeystorePasswordForLogging());
|
",keyPassword=" + getKeyPasswordForLogging() +
|
||||||
sb.append(",keyPassword=").append(getKeyPasswordForLogging());
|
",keystoreType=" + keystoreType +
|
||||||
sb.append(",keystoreType=").append(keystoreType);
|
",truststorePath=" + truststorePath +
|
||||||
sb.append(",truststorePath=").append(truststorePath);
|
",truststorePassword=" + getTruststorePasswordForLogging() +
|
||||||
sb.append(",truststorePassword=").append(getTruststorePasswordForLogging());
|
",truststoreType=" + truststoreType +
|
||||||
sb.append(",truststoreType=").append(truststoreType);
|
",protocol=" + protocol;
|
||||||
sb.append(",protocol=").append(protocol);
|
|
||||||
return sb.toString();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -507,32 +470,32 @@ public class StandardTlsConfiguration implements TlsConfiguration {
|
||||||
return StringUtils.isNotBlank(path) || StringUtils.isNotBlank(password) || type != null;
|
return StringUtils.isNotBlank(path) || StringUtils.isNotBlank(password) || type != null;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isStorePopulated(String path, String password, KeystoreType type, String label) {
|
private boolean isStorePopulated(final String path, final String password, final KeystoreType type, final StoreType storeType) {
|
||||||
boolean isPopulated;
|
boolean populated;
|
||||||
String passwordForLogging;
|
|
||||||
|
|
||||||
// Legacy truststores can be populated without a password; only check the path and type
|
// Legacy trust stores such as JKS can be populated without a password; only check the path and type
|
||||||
isPopulated = StringUtils.isNotBlank(path) && type != null;
|
populated = StringUtils.isNotBlank(path) && type != null;
|
||||||
if ("truststore".equalsIgnoreCase(label)) {
|
if (StoreType.KEY_STORE == storeType) {
|
||||||
passwordForLogging = getTruststorePasswordForLogging();
|
populated = populated && StringUtils.isNotBlank(password);
|
||||||
} else {
|
|
||||||
// Keystores require a password
|
|
||||||
isPopulated = isPopulated && StringUtils.isNotBlank(password);
|
|
||||||
passwordForLogging = getKeystorePasswordForLogging();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (logger.isDebugEnabled()) {
|
return populated;
|
||||||
logger.debug("TLS config {} is {}: {}, {}, {}", label, isPopulated ? "populated" : "not populated", path, passwordForLogging, type);
|
|
||||||
}
|
|
||||||
return isPopulated;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isStoreValid(String path, String password, KeystoreType type, String label) {
|
private boolean isStoreValid(final String path, final String password, final KeystoreType type, final StoreType storeType) {
|
||||||
|
return isStorePopulated(path, password, type, storeType) && KeyStoreUtils.isStoreValid(getFileUrl(path), type, password.toCharArray());
|
||||||
|
}
|
||||||
|
|
||||||
|
private URL getFileUrl(final String path) {
|
||||||
try {
|
try {
|
||||||
return isStorePopulated(path, password, type, label) && KeyStoreUtils.isStoreValid(new File(path).toURI().toURL(), type, password.toCharArray());
|
return new File(path).toURI().toURL();
|
||||||
} catch (MalformedURLException e) {
|
} catch (final MalformedURLException e) {
|
||||||
logger.error("Encountered an error validating the " + label + ": " + e.getLocalizedMessage());
|
throw new IllegalArgumentException(String.format("File Path [%s] URL conversion failed", path), e);
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private enum StoreType {
|
||||||
|
KEY_STORE,
|
||||||
|
TRUST_STORE
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -170,6 +170,12 @@ class StandardTlsConfigurationTest extends GroovyTestCase {
|
||||||
assert noPasswordIsPopulated
|
assert noPasswordIsPopulated
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testIsKeystoreValidFalseNullKeystorePassword() {
|
||||||
|
TlsConfiguration configuration = new StandardTlsConfiguration(KEYSTORE_PATH, null, KEY_PASSWORD, KEYSTORE_TYPE, TRUSTSTORE_PATH, TRUSTSTORE_PASSWORD, TRUSTSTORE_TYPE)
|
||||||
|
assert !configuration.isKeystoreValid()
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
void testShouldValidateKeystoreConfiguration() {
|
void testShouldValidateKeystoreConfiguration() {
|
||||||
// Arrange
|
// Arrange
|
||||||
|
@ -234,7 +240,7 @@ class StandardTlsConfigurationTest extends GroovyTestCase {
|
||||||
TlsConfiguration configuration = getTlsConfiguration(currentProtocol)
|
TlsConfiguration configuration = getTlsConfiguration(currentProtocol)
|
||||||
|
|
||||||
String[] enabledProtocols = configuration.enabledProtocols
|
String[] enabledProtocols = configuration.enabledProtocols
|
||||||
assert enabledProtocols == [currentProtocol]
|
assert enabledProtocols == [currentProtocol] as String[]
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
Loading…
Reference in New Issue