Skip to content

Commit

Permalink
NIFI-11653 This closes #7349. Added Connection URL Validation for Dat…
Browse files Browse the repository at this point in the history
…abase Services

Signed-off-by: Joe Witt <joewitt@apache.org>
  • Loading branch information
exceptionfactory authored and joewitt committed Jun 6, 2023
1 parent d956ede commit 4faf3ea
Show file tree
Hide file tree
Showing 7 changed files with 175 additions and 2 deletions.
Expand Up @@ -20,6 +20,7 @@
import org.apache.nifi.components.PropertyValue;
import org.apache.nifi.components.resource.ResourceCardinality;
import org.apache.nifi.components.resource.ResourceType;
import org.apache.nifi.dbcp.ConnectionUrlValidator;
import org.apache.nifi.dbcp.DBCPValidator;
import org.apache.nifi.expression.ExpressionLanguageScope;
import org.apache.nifi.kerberos.KerberosUserService;
Expand All @@ -37,7 +38,7 @@ private DBCPProperties() {
.description("A database connection URL used to connect to a database. May contain database system name, host, port, database name and some parameters."
+ " The exact syntax of a database connection URL is specified by your DBMS.")
.defaultValue(null)
.addValidator(StandardValidators.NON_EMPTY_VALIDATOR)
.addValidator(new ConnectionUrlValidator())
.required(true)
.expressionLanguageSupported(ExpressionLanguageScope.VARIABLE_REGISTRY)
.build();
Expand Down
Expand Up @@ -32,5 +32,10 @@
<artifactId>nifi-utils</artifactId>
<version>2.0.0-SNAPSHOT</version>
</dependency>
<dependency>
<groupId>org.apache.nifi</groupId>
<artifactId>nifi-mock</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
</project>
@@ -0,0 +1,66 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.nifi.dbcp;

import org.apache.nifi.components.ValidationContext;
import org.apache.nifi.components.ValidationResult;
import org.apache.nifi.components.Validator;

import java.util.Collections;
import java.util.Set;

/**
* Database Connection URL Validator supports system attribute expressions and evaluates URL formatting
*/
public class ConnectionUrlValidator implements Validator {
private static final Set<String> UNSUPPORTED_SCHEMES = Collections.singleton("jdbc:h2");

@Override
public ValidationResult validate(final String subject, final String input, final ValidationContext context) {
final ValidationResult.Builder builder = new ValidationResult.Builder().subject(subject).input(input);

if (input == null || input.isEmpty()) {
builder.valid(false);
builder.explanation("Connection URL required");
} else {
final String url = context.newPropertyValue(input).evaluateAttributeExpressions().getValue();

if (isUrlUnsupported(url)) {
builder.valid(false);
builder.explanation(String.format("Connection URL starts with an unsupported scheme %s", UNSUPPORTED_SCHEMES));
} else {
builder.valid(true);
builder.explanation("Connection URL is valid");
}
}

return builder.build();
}

private boolean isUrlUnsupported(final String url) {
boolean unsupported = false;

for (final String unsupportedScheme : UNSUPPORTED_SCHEMES) {
if (url.startsWith(unsupportedScheme)) {
unsupported = true;
break;
}
}

return unsupported;
}
}
@@ -0,0 +1,77 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.nifi.dbcp;

import org.apache.nifi.components.ValidationContext;
import org.apache.nifi.components.ValidationResult;
import org.apache.nifi.util.MockProcessContext;
import org.apache.nifi.util.MockValidationContext;
import org.apache.nifi.util.NoOpProcessor;
import org.apache.nifi.util.TestRunners;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;

import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertTrue;

class ConnectionUrlValidatorTest {

private static final String SUBJECT = "Database URL";

private static final String EMPTY = "";

private static final String UNSUPPORTED_URL = "jdbc:h2:file";

private static final String VENDOR_URL = "jdbc:vendor";

private ValidationContext validationContext;

private ConnectionUrlValidator validator;

@BeforeEach
void setValidator() {
validator = new ConnectionUrlValidator();

final MockProcessContext processContext = (MockProcessContext) TestRunners.newTestRunner(NoOpProcessor.class).getProcessContext();
validationContext = new MockValidationContext(processContext);
}

@Test
void testValidateEmpty() {
final ValidationResult result = validator.validate(SUBJECT, EMPTY, validationContext);

assertNotNull(result);
assertFalse(result.isValid());
}

@Test
void testValidateUnsupportedUrl() {
final ValidationResult result = validator.validate(SUBJECT, UNSUPPORTED_URL, validationContext);

assertNotNull(result);
assertFalse(result.isValid());
}

@Test
void testValidateSupportedUrl() {
final ValidationResult result = validator.validate(SUBJECT, VENDOR_URL, validationContext);

assertNotNull(result);
assertTrue(result.isValid());
}
}
Expand Up @@ -57,6 +57,8 @@ public class DBCPServiceTest {

private static final String DERBY_SHUTDOWN_STATE = "XJ015";

private static final String INVALID_CONNECTION_URL = "jdbc:h2";

private TestRunner runner;

private File databaseDirectory;
Expand Down Expand Up @@ -99,6 +101,14 @@ public void shutdown() throws IOException {
}
}

@Test
public void testConnectionUrlInvalid() {
runner.assertValid(service);

runner.setProperty(service, DBCPProperties.DATABASE_URL, INVALID_CONNECTION_URL);
runner.assertNotValid(service);
}

@Test
public void testCustomValidateOfKerberosProperties() throws InitializationException {
// direct principal + password and no kerberos services is valid
Expand Down
Expand Up @@ -76,7 +76,7 @@ public class HikariCPConnectionPool extends AbstractControllerService implements
.description("A database connection URL used to connect to a database. May contain database system name, host, port, database name and some parameters."
+ " The exact syntax of a database connection URL is specified by your DBMS.")
.defaultValue(null)
.addValidator(StandardValidators.NON_EMPTY_VALIDATOR)
.addValidator(new ConnectionUrlValidator())
.required(true)
.expressionLanguageSupported(ExpressionLanguageScope.VARIABLE_REGISTRY)
.build();
Expand Down
Expand Up @@ -33,13 +33,27 @@
public class HikariCPConnectionPoolTest {
private final static String SERVICE_ID = HikariCPConnectionPoolTest.class.getSimpleName();

private static final String INVALID_CONNECTION_URL = "jdbc:h2";

private TestRunner runner;

@BeforeEach
public void setup() {
runner = TestRunners.newTestRunner(NoOpProcessor.class);
}

@Test
public void testConnectionUrlInvalid() throws InitializationException {
final HikariCPConnectionPool service = new HikariCPConnectionPool();

runner.addControllerService(SERVICE_ID, service);
setDatabaseProperties(service);
runner.assertValid(service);

runner.setProperty(service, HikariCPConnectionPool.DATABASE_URL, INVALID_CONNECTION_URL);
runner.assertNotValid(service);
}

@Test
public void testMissingPropertyValues() throws InitializationException {
final HikariCPConnectionPool service = new HikariCPConnectionPool();
Expand Down

0 comments on commit 4faf3ea

Please sign in to comment.