mirror of https://github.com/apache/nifi.git
NIFI-10776 Added NONE and PKI AuthorizationSchemes for ElasticSearchClientService
This closes #6662 Signed-off-by: David Handermann <exceptionfactory@apache.org>
This commit is contained in:
parent
a861bab34d
commit
d23e50168f
|
@ -41,6 +41,27 @@ An example using a non-Docker version of Elasticsearch:
|
||||||
|
|
||||||
`mvn -Pintegration-tests --fail-at-end -Delasticsearch.testcontainers.enabled=false -Delasticsearch.elastic_user.password=s3cret1234 clean install`
|
`mvn -Pintegration-tests --fail-at-end -Delasticsearch.testcontainers.enabled=false -Delasticsearch.elastic_user.password=s3cret1234 clean install`
|
||||||
|
|
||||||
|
## Bash Script Example
|
||||||
|
|
||||||
|
Execute the following script from the `nifi-elasticsearch-bundle` directory:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mvn --fail-at-end -Pcontrib-check clean install
|
||||||
|
|
||||||
|
es_versions=(elasticsearch6 elasticsearch7 elasticsearch8)
|
||||||
|
it_modules=(nifi-elasticsearch-client-service nifi-elasticsearch-restapi-processors)
|
||||||
|
for v in "${es_versions[@]}"; do
|
||||||
|
for m in "${it_modules[@]}"; do
|
||||||
|
pushd "${m}"
|
||||||
|
if ! mvn -P "integration-tests,${v}" --fail-at-end failsafe:integration-test failsafe:verify; then
|
||||||
|
echo; echo; echo "Integration Tests failed for ${v} in ${m}, see Maven logs for details"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
popd
|
||||||
|
done
|
||||||
|
done
|
||||||
|
```
|
||||||
|
|
||||||
## Modules with Integration Tests (using Testcontainers)
|
## Modules with Integration Tests (using Testcontainers)
|
||||||
|
|
||||||
- [Elasticsearch Client Service](nifi-elasticsearch-client-service)
|
- [Elasticsearch Client Service](nifi-elasticsearch-client-service)
|
||||||
|
@ -50,6 +71,8 @@ An example using a non-Docker version of Elasticsearch:
|
||||||
|
|
||||||
Integration Tests with Testcontainers currently only uses the `amd64` Docker Images.
|
Integration Tests with Testcontainers currently only uses the `amd64` Docker Images.
|
||||||
|
|
||||||
`elasticsearch6` is known to **not** work with `arm64` machines (e.g. Mac M1/M2), but other Elasticsearch images (e.g. 7.x and 8.x) appear to work.
|
`elasticsearch6` is known to experience some problems with `arm64` machines (e.g. Mac M1/M2),
|
||||||
|
but other Elasticsearch images (e.g. 7.x and 8.x) appear to work. Settings have been altered for the Elasticsearch
|
||||||
|
containers in order to try and enable them on different architectures, but there may still be some inconsistencies.
|
||||||
|
|
||||||
Explicit `arm64` architecture support may be added in future where the Elasticsearch images exist.
|
Explicit `arm64` architecture support may be added in future where the Elasticsearch images exist.
|
||||||
|
|
|
@ -19,6 +19,8 @@ package org.apache.nifi.elasticsearch;
|
||||||
import org.apache.nifi.components.DescribedValue;
|
import org.apache.nifi.components.DescribedValue;
|
||||||
|
|
||||||
public enum AuthorizationScheme implements DescribedValue {
|
public enum AuthorizationScheme implements DescribedValue {
|
||||||
|
NONE("None", "No authorization scheme."),
|
||||||
|
PKI("PKI", "Mutual TLS with PKI certificate authorization scheme."),
|
||||||
BASIC("Basic", "Basic authorization scheme."),
|
BASIC("Basic", "Basic authorization scheme."),
|
||||||
API_KEY("API Key", "API key authorization scheme.");
|
API_KEY("API Key", "API key authorization scheme.");
|
||||||
|
|
||||||
|
|
|
@ -58,7 +58,7 @@ public interface ElasticSearchClientService extends ControllerService, Verifiabl
|
||||||
PropertyDescriptor AUTHORIZATION_SCHEME = new PropertyDescriptor.Builder()
|
PropertyDescriptor AUTHORIZATION_SCHEME = new PropertyDescriptor.Builder()
|
||||||
.name("authorization-scheme")
|
.name("authorization-scheme")
|
||||||
.displayName("Authorization Scheme")
|
.displayName("Authorization Scheme")
|
||||||
.description("Authorization Scheme used for authenticating to Elasticsearch using the HTTP Authorization header.")
|
.description("Authorization Scheme used for optional authentication to Elasticsearch.")
|
||||||
.allowableValues(AuthorizationScheme.class)
|
.allowableValues(AuthorizationScheme.class)
|
||||||
.defaultValue(AuthorizationScheme.BASIC.getValue())
|
.defaultValue(AuthorizationScheme.BASIC.getValue())
|
||||||
.required(true)
|
.required(true)
|
||||||
|
|
|
@ -112,33 +112,48 @@ public class ElasticSearchClientServiceImpl extends AbstractControllerService im
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected Collection<ValidationResult> customValidate(ValidationContext validationContext) {
|
protected Collection<ValidationResult> customValidate(final ValidationContext validationContext) {
|
||||||
final List<ValidationResult> results = new ArrayList<>(1);
|
final List<ValidationResult> results = new ArrayList<>(super.customValidate(validationContext));
|
||||||
|
|
||||||
|
final AuthorizationScheme authorizationScheme = AuthorizationScheme.valueOf(validationContext.getProperty(AUTHORIZATION_SCHEME).getValue());
|
||||||
|
|
||||||
final boolean usernameSet = validationContext.getProperty(USERNAME).isSet();
|
final boolean usernameSet = validationContext.getProperty(USERNAME).isSet();
|
||||||
final boolean passwordSet = validationContext.getProperty(PASSWORD).isSet();
|
final boolean passwordSet = validationContext.getProperty(PASSWORD).isSet();
|
||||||
|
|
||||||
if ((usernameSet && !passwordSet) || (!usernameSet && passwordSet)) {
|
|
||||||
results.add(new ValidationResult.Builder().subject(String.format("%s and %s", USERNAME.getDisplayName(), PASSWORD.getDisplayName()))
|
|
||||||
.valid(false).explanation(String.format("if '%s' or '%s' is set, both must be set.", USERNAME.getDisplayName(), PASSWORD.getDisplayName())).build());
|
|
||||||
}
|
|
||||||
|
|
||||||
final boolean apiKeyIdSet = validationContext.getProperty(API_KEY_ID).isSet();
|
final boolean apiKeyIdSet = validationContext.getProperty(API_KEY_ID).isSet();
|
||||||
final boolean apiKeySet = validationContext.getProperty(API_KEY).isSet();
|
final boolean apiKeySet = validationContext.getProperty(API_KEY).isSet();
|
||||||
|
|
||||||
if ((apiKeyIdSet && !apiKeySet) || (!apiKeyIdSet && apiKeySet)) {
|
final SSLContextService sslService = validationContext.getProperty(PROP_SSL_CONTEXT_SERVICE).asControllerService(SSLContextService.class);
|
||||||
results.add(new ValidationResult.Builder().subject(String.format("%s and %s", API_KEY.getDisplayName(), API_KEY_ID.getDisplayName()))
|
if (authorizationScheme == AuthorizationScheme.PKI && (sslService == null || !sslService.isKeyStoreConfigured())) {
|
||||||
.valid(false).explanation(String.format("if '%s' or '%s' is set, both must be set.", API_KEY.getDisplayName(), API_KEY_ID.getDisplayName())).build());
|
results.add(new ValidationResult.Builder().subject(PROP_SSL_CONTEXT_SERVICE.getName()).valid(false)
|
||||||
|
.explanation(String.format("if '%s' is '%s' then '%s' must be set and specify a Keystore for mutual TLS encryption.",
|
||||||
|
AUTHORIZATION_SCHEME.getDisplayName(), authorizationScheme.getDisplayName(), PROP_SSL_CONTEXT_SERVICE.getDisplayName())
|
||||||
|
).build()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (usernameSet && apiKeyIdSet) {
|
if (usernameSet && !passwordSet) {
|
||||||
results.add(new ValidationResult.Builder().subject(String.format("%s and %s", USERNAME.getDisplayName(), API_KEY_ID.getDisplayName()))
|
addAuthorizationPropertiesValidationIssue(results, USERNAME, PASSWORD);
|
||||||
.valid(false).explanation(String.format("'%s' and '%s' cannot be used together.", USERNAME.getDisplayName(), API_KEY_ID.getDisplayName())).build());
|
} else if (passwordSet && !usernameSet) {
|
||||||
|
addAuthorizationPropertiesValidationIssue(results, PASSWORD, USERNAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (apiKeyIdSet && !apiKeySet) {
|
||||||
|
addAuthorizationPropertiesValidationIssue(results, API_KEY_ID, API_KEY);
|
||||||
|
} else if (apiKeySet && !apiKeyIdSet) {
|
||||||
|
addAuthorizationPropertiesValidationIssue(results, API_KEY, API_KEY_ID);
|
||||||
}
|
}
|
||||||
|
|
||||||
return results;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void addAuthorizationPropertiesValidationIssue(final List<ValidationResult> results, final PropertyDescriptor presentProperty, final PropertyDescriptor missingProperty) {
|
||||||
|
results.add(new ValidationResult.Builder().subject(missingProperty.getName()).valid(false)
|
||||||
|
.explanation(String.format("if '%s' is then '%s' must be set.", presentProperty.getDisplayName(), missingProperty.getDisplayName()))
|
||||||
|
.build()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
@OnEnabled
|
@OnEnabled
|
||||||
public void onEnabled(final ConfigurationContext context) throws InitializationException {
|
public void onEnabled(final ConfigurationContext context) throws InitializationException {
|
||||||
try {
|
try {
|
||||||
|
|
|
@ -0,0 +1,144 @@
|
||||||
|
/*
|
||||||
|
* 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.elasticsearch.unit;
|
||||||
|
|
||||||
|
import org.apache.nifi.components.PropertyDescriptor;
|
||||||
|
import org.apache.nifi.elasticsearch.AuthorizationScheme;
|
||||||
|
import org.apache.nifi.elasticsearch.ElasticSearchClientService;
|
||||||
|
import org.apache.nifi.elasticsearch.ElasticSearchClientServiceImpl;
|
||||||
|
import org.apache.nifi.elasticsearch.TestControllerServiceProcessor;
|
||||||
|
import org.apache.nifi.reporting.InitializationException;
|
||||||
|
import org.apache.nifi.ssl.SSLContextService;
|
||||||
|
import org.apache.nifi.util.TestRunner;
|
||||||
|
import org.apache.nifi.util.TestRunners;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.opentest4j.AssertionFailedError;
|
||||||
|
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertThrows;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||||
|
import static org.mockito.Mockito.atMostOnce;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.reset;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
class ElasticSearchClientServiceImplTest {
|
||||||
|
private TestRunner runner;
|
||||||
|
|
||||||
|
private ElasticSearchClientServiceImpl service;
|
||||||
|
|
||||||
|
private static final String HOST = "http://localhost:9200";
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
void setUp() throws Exception {
|
||||||
|
runner = TestRunners.newTestRunner(TestControllerServiceProcessor.class);
|
||||||
|
service = new ElasticSearchClientServiceImpl();
|
||||||
|
runner.addControllerService("Client Service", service);
|
||||||
|
runner.setProperty(TestControllerServiceProcessor.CLIENT_SERVICE, "Client Service");
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.HTTP_HOSTS, HOST);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testTransitUrl() {
|
||||||
|
final String index = "test";
|
||||||
|
final String type = "no-type";
|
||||||
|
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.AUTHORIZATION_SCHEME, AuthorizationScheme.NONE.getValue());
|
||||||
|
runner.assertValid(service);
|
||||||
|
runner.enableControllerService(service);
|
||||||
|
|
||||||
|
assertEquals(String.format("%s/%s/%s", HOST, index, type), service.getTransitUrl(index, type));
|
||||||
|
assertEquals(String.format("%s/%s", HOST, index), service.getTransitUrl(index, null));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testValidateBasicAuth() {
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.AUTHORIZATION_SCHEME, AuthorizationScheme.BASIC.getValue());
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.USERNAME, "elastic");
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.PASSWORD, "password");
|
||||||
|
runner.assertValid(service);
|
||||||
|
|
||||||
|
runner.removeProperty(service, ElasticSearchClientService.PASSWORD);
|
||||||
|
assertAuthorizationPropertyValidationErrorMessage(ElasticSearchClientService.USERNAME, ElasticSearchClientService.PASSWORD);
|
||||||
|
|
||||||
|
runner.removeProperty(service, ElasticSearchClientService.USERNAME);
|
||||||
|
runner.assertValid(service);
|
||||||
|
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.PASSWORD, "password");
|
||||||
|
runner.removeProperty(service, ElasticSearchClientService.USERNAME);
|
||||||
|
assertAuthorizationPropertyValidationErrorMessage(ElasticSearchClientService.PASSWORD, ElasticSearchClientService.USERNAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testValidateApiKeyAuth() {
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.AUTHORIZATION_SCHEME, AuthorizationScheme.API_KEY.getValue());
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.API_KEY_ID, "api-key-id");
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.API_KEY, "api-key");
|
||||||
|
runner.assertValid(service);
|
||||||
|
|
||||||
|
runner.removeProperty(service, ElasticSearchClientService.API_KEY_ID);
|
||||||
|
assertAuthorizationPropertyValidationErrorMessage(ElasticSearchClientService.API_KEY, ElasticSearchClientService.API_KEY_ID);
|
||||||
|
|
||||||
|
runner.removeProperty(service, ElasticSearchClientService.API_KEY);
|
||||||
|
runner.assertValid(service);
|
||||||
|
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.API_KEY_ID, "api-key-id");
|
||||||
|
runner.removeProperty(service, ElasticSearchClientService.API_KEY);
|
||||||
|
assertAuthorizationPropertyValidationErrorMessage(ElasticSearchClientService.API_KEY_ID, ElasticSearchClientService.API_KEY);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void testValidatePkiAuth() throws InitializationException {
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.AUTHORIZATION_SCHEME, AuthorizationScheme.PKI.getValue());
|
||||||
|
|
||||||
|
final SSLContextService sslService = mock(SSLContextService.class);
|
||||||
|
when(sslService.getIdentifier()).thenReturn("ssl-context");
|
||||||
|
runner.addControllerService("ssl-context", sslService);
|
||||||
|
runner.setProperty(service, ElasticSearchClientService.PROP_SSL_CONTEXT_SERVICE, "ssl-context");
|
||||||
|
when(sslService.isKeyStoreConfigured()).thenReturn(true);
|
||||||
|
runner.assertValid(service);
|
||||||
|
verify(sslService, atMostOnce()).isKeyStoreConfigured();
|
||||||
|
reset(sslService);
|
||||||
|
|
||||||
|
when(sslService.isKeyStoreConfigured()).thenReturn(false);
|
||||||
|
assertPKIAuthorizationValidationErrorMessage();
|
||||||
|
verify(sslService, atMostOnce()).isKeyStoreConfigured();
|
||||||
|
reset(sslService);
|
||||||
|
|
||||||
|
runner.removeProperty(service, ElasticSearchClientService.PROP_SSL_CONTEXT_SERVICE);
|
||||||
|
assertPKIAuthorizationValidationErrorMessage();
|
||||||
|
verify(sslService, atMostOnce()).isKeyStoreConfigured();
|
||||||
|
reset(sslService);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertAuthorizationPropertyValidationErrorMessage(final PropertyDescriptor presentProperty, final PropertyDescriptor missingProperty) {
|
||||||
|
final AssertionFailedError afe = assertThrows(AssertionFailedError.class, () -> runner.assertValid(service));
|
||||||
|
assertTrue(afe.getMessage().contains(String.format("if '%s' is then '%s' must be set.", presentProperty.getDisplayName(), missingProperty.getDisplayName())));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertPKIAuthorizationValidationErrorMessage() {
|
||||||
|
final AssertionFailedError afe = assertThrows(AssertionFailedError.class, () -> runner.assertValid(service));
|
||||||
|
assertTrue(afe.getMessage().contains(String.format(
|
||||||
|
"if '%s' is '%s' then '%s' must be set and specify a Keystore for mutual TLS encryption.",
|
||||||
|
ElasticSearchClientService.AUTHORIZATION_SCHEME.getDisplayName(),
|
||||||
|
AuthorizationScheme.PKI.getDisplayName(),
|
||||||
|
ElasticSearchClientService.PROP_SSL_CONTEXT_SERVICE.getDisplayName()
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
}
|
|
@ -53,12 +53,16 @@ import java.util.Map;
|
||||||
import static org.apache.http.auth.AuthScope.ANY;
|
import static org.apache.http.auth.AuthScope.ANY;
|
||||||
|
|
||||||
public abstract class AbstractElasticsearchITBase {
|
public abstract class AbstractElasticsearchITBase {
|
||||||
|
// default Elasticsearch version should (ideally) match that in the nifi-elasticsearch-bundle#pom.xml for the integration-tests profile
|
||||||
protected static final DockerImageName IMAGE = DockerImageName
|
protected static final DockerImageName IMAGE = DockerImageName
|
||||||
.parse(System.getProperty("elasticsearch.docker.image", "docker.elastic.co/elasticsearch/elasticsearch:8.4.3"));
|
.parse(System.getProperty("elasticsearch.docker.image", "docker.elastic.co/elasticsearch/elasticsearch:8.5.0"));
|
||||||
protected static final String ELASTIC_USER_PASSWORD = System.getProperty("elasticsearch.elastic_user.password", RandomStringUtils.randomAlphanumeric(10, 20));
|
protected static final String ELASTIC_USER_PASSWORD = System.getProperty("elasticsearch.elastic_user.password", RandomStringUtils.randomAlphanumeric(10, 20));
|
||||||
protected static final ElasticsearchContainer ELASTICSEARCH_CONTAINER = new ElasticsearchContainer(IMAGE)
|
protected static final ElasticsearchContainer ELASTICSEARCH_CONTAINER = new ElasticsearchContainer(IMAGE)
|
||||||
.withPassword(ELASTIC_USER_PASSWORD)
|
.withPassword(ELASTIC_USER_PASSWORD)
|
||||||
.withEnv("xpack.security.enabled", "true");
|
.withEnv("xpack.security.enabled", "true")
|
||||||
|
// enable API Keys for integration-tests (6.x & 7.x don't enable SSL and therefore API Keys by default, so use a trial license and explicitly enable API Keys)
|
||||||
|
.withEnv("xpack.license.self_generated.type", "trial")
|
||||||
|
.withEnv("xpack.security.authc.api_key.enabled", "true");
|
||||||
protected static final String CLIENT_SERVICE_NAME = "Client Service";
|
protected static final String CLIENT_SERVICE_NAME = "Client Service";
|
||||||
protected static final String INDEX = "messages";
|
protected static final String INDEX = "messages";
|
||||||
|
|
||||||
|
@ -68,7 +72,12 @@ public abstract class AbstractElasticsearchITBase {
|
||||||
protected static String elasticsearchHost;
|
protected static String elasticsearchHost;
|
||||||
protected static void startTestcontainer() {
|
protected static void startTestcontainer() {
|
||||||
if (ENABLE_TEST_CONTAINERS) {
|
if (ENABLE_TEST_CONTAINERS) {
|
||||||
ELASTICSEARCH_CONTAINER.start();
|
if (getElasticMajorVersion() == 6) {
|
||||||
|
// disable system call filter check to allow Elasticsearch 6 to run on aarch64 machines (e.g. Mac M1/2)
|
||||||
|
ELASTICSEARCH_CONTAINER.withEnv("bootstrap.system_call_filter", "false").start();
|
||||||
|
} else {
|
||||||
|
ELASTICSEARCH_CONTAINER.start();
|
||||||
|
}
|
||||||
elasticsearchHost = String.format("http://%s", ELASTICSEARCH_CONTAINER.getHttpHostAddress());
|
elasticsearchHost = String.format("http://%s", ELASTICSEARCH_CONTAINER.getHttpHostAddress());
|
||||||
} else {
|
} else {
|
||||||
elasticsearchHost = System.getProperty("elasticsearch.endpoint", "http://localhost:9200");
|
elasticsearchHost = System.getProperty("elasticsearch.endpoint", "http://localhost:9200");
|
||||||
|
@ -89,6 +98,7 @@ public abstract class AbstractElasticsearchITBase {
|
||||||
|
|
||||||
@BeforeAll
|
@BeforeAll
|
||||||
static void beforeAll() throws IOException {
|
static void beforeAll() throws IOException {
|
||||||
|
|
||||||
startTestcontainer();
|
startTestcontainer();
|
||||||
type = getElasticMajorVersion() == 6 ? "_doc" : "";
|
type = getElasticMajorVersion() == 6 ? "_doc" : "";
|
||||||
System.out.printf("%n%n%n%n%n%n%n%n%n%n%n%n%n%n%nTYPE: %s%nIMAGE: %s:%s%n%n%n%n%n%n%n%n%n%n%n%n%n%n%n%n",
|
System.out.printf("%n%n%n%n%n%n%n%n%n%n%n%n%n%n%nTYPE: %s%nIMAGE: %s:%s%n%n%n%n%n%n%n%n%n%n%n%n%n%n%n%n",
|
||||||
|
|
|
@ -86,6 +86,7 @@ language governing permissions and limitations under the License. -->
|
||||||
<activeByDefault>false</activeByDefault>
|
<activeByDefault>false</activeByDefault>
|
||||||
</activation>
|
</activation>
|
||||||
<properties>
|
<properties>
|
||||||
|
<!-- also update the default Elasticsearch version in nifi-elasticsearch-test-utils#src/main/java/org/apache/nifi/elasticsearch/integration/AbstractElasticsearchITBase.java-->
|
||||||
<elasticsearch_docker_image>8.5.0</elasticsearch_docker_image>
|
<elasticsearch_docker_image>8.5.0</elasticsearch_docker_image>
|
||||||
<elasticsearch.elastic.password>s3cret</elasticsearch.elastic.password>
|
<elasticsearch.elastic.password>s3cret</elasticsearch.elastic.password>
|
||||||
</properties>
|
</properties>
|
||||||
|
|
Loading…
Reference in New Issue