From 19a84dd3dd1e40a41cf8e2daf88092d946043aac Mon Sep 17 00:00:00 2001 From: Tamas Palfy Date: Fri, 26 Jun 2020 18:21:47 +0200 Subject: [PATCH] NIFI-7581 Added ControllerService-based credential settings support for ADLS processors NIFI-7581 Separated Controller Service for providing Azure credentials for ADLS (ADLSCredentialsControllerService) form the one that does the same for Blob storages (AzureStorageCredentialsDetails). (This was done due to the considerable difference in the APIs of the libraries used to connect to both.) NIFI-7581 Fix: Register controller service in META-INF. Minor fixes. NIFI-7581 Minor changes (documentation, type etc.) NIFI-7581 Updated integration tests. NIFI-7581 Minor changes (renaming). This closes #4369. Signed-off-by: Peter Turcsanyi --- .../nifi-azure-processors/pom.xml | 2 +- ...AbstractAzureDataLakeStorageProcessor.java | 160 +++--------- .../storage/utils/AzureStorageUtils.java | 15 +- .../ADLSCredentialsControllerService.java | 153 +++++++++++ ...g.apache.nifi.controller.ControllerService | 1 + .../AbstractAzureDataLakeStorageIT.java | 14 + .../azure/storage/AbstractAzureStorageIT.java | 6 +- .../TestAbstractAzureDataLakeStorage.java | 59 +---- .../TestADLSCredentialsControllerService.java | 242 ++++++++++++++++++ .../nifi-azure-services-api/pom.xml | 5 + .../azure/storage/ADLSCredentialsDetails.java | 120 +++++++++ .../azure/storage/ADLSCredentialsService.java | 35 +++ nifi-nar-bundles/nifi-azure-bundle/pom.xml | 1 + 13 files changed, 639 insertions(+), 174 deletions(-) create mode 100644 nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsControllerService.java create mode 100644 nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/services/azure/storage/TestADLSCredentialsControllerService.java create mode 100644 nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsDetails.java create mode 100644 nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsService.java diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/pom.xml b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/pom.xml index 2cff8eed68..f0da2880a0 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/pom.xml +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/pom.xml @@ -59,7 +59,7 @@ com.azure azure-core - 1.5.0 + ${azure.core.version} com.azure diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/AbstractAzureDataLakeStorageProcessor.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/AbstractAzureDataLakeStorageProcessor.java index af75f99720..fd0be044a5 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/AbstractAzureDataLakeStorageProcessor.java +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/AbstractAzureDataLakeStorageProcessor.java @@ -16,15 +16,15 @@ */ package org.apache.nifi.processors.azure; -import java.util.ArrayList; import java.util.Arrays; -import java.util.Collection; import java.util.Collections; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Set; +import com.azure.core.credential.AccessToken; +import com.azure.core.credential.TokenCredential; import com.azure.identity.ManagedIdentityCredential; import com.azure.identity.ManagedIdentityCredentialBuilder; import com.azure.storage.common.StorageSharedKeyCredential; @@ -33,8 +33,6 @@ import com.azure.storage.file.datalake.DataLakeServiceClientBuilder; import org.apache.commons.lang3.StringUtils; import org.apache.nifi.components.PropertyDescriptor; -import org.apache.nifi.components.ValidationContext; -import org.apache.nifi.components.ValidationResult; import org.apache.nifi.components.Validator; import org.apache.nifi.context.PropertyContext; import org.apache.nifi.expression.ExpressionLanguageScope; @@ -42,57 +40,19 @@ import org.apache.nifi.flowfile.FlowFile; import org.apache.nifi.processor.AbstractProcessor; import org.apache.nifi.processor.Relationship; import org.apache.nifi.processor.util.StandardValidators; +import org.apache.nifi.services.azure.storage.ADLSCredentialsDetails; +import org.apache.nifi.services.azure.storage.ADLSCredentialsService; +import reactor.core.publisher.Mono; public abstract class AbstractAzureDataLakeStorageProcessor extends AbstractProcessor { - public static final PropertyDescriptor ACCOUNT_NAME = new PropertyDescriptor.Builder() - .name("storage-account-name").displayName("Storage Account Name") - .description("The storage account name. There are certain risks in allowing the account name to be stored as a flowfile " + - "attribute. While it does provide for a more flexible flow by allowing the account name to " + - "be fetched dynamically from a flowfile attribute, care must be taken to restrict access to " + - "the event provenance data (e.g. by strictly controlling the policies governing provenance for this Processor). " + - "In addition, the provenance repositories may be put on encrypted disk partitions." + - " Instead of defining the Storage Account Name, Storage Account Key and SAS Token properties directly on the processor, " + - "the preferred way is to configure them through a controller service") - .addValidator(StandardValidators.NON_EMPTY_VALIDATOR) - .expressionLanguageSupported(ExpressionLanguageScope.FLOWFILE_ATTRIBUTES) - .required(true) - .sensitive(true).build(); - - public static final PropertyDescriptor ACCOUNT_KEY = new PropertyDescriptor.Builder() - .name("storage-account-key").displayName("Storage Account Key") - .description("The storage account key. This is an admin-like password providing access to every container in this account. It is recommended " + - "one uses Shared Access Signature (SAS) token instead for fine-grained control with policies. " + - "There are certain risks in allowing the account key to be stored as a flowfile " + - "attribute. While it does provide for a more flexible flow by allowing the account key to " + - "be fetched dynamically from a flow file attribute, care must be taken to restrict access to " + - "the event provenance data (e.g. by strictly controlling the policies governing provenance for this Processor). " + - "In addition, the provenance repositories may be put on encrypted disk partitions.") - .addValidator(StandardValidators.NON_EMPTY_VALIDATOR) - .expressionLanguageSupported(ExpressionLanguageScope.FLOWFILE_ATTRIBUTES) - .required(false) - .sensitive(true).build(); - - public static final PropertyDescriptor SAS_TOKEN = new PropertyDescriptor.Builder() - .name("storage-sas-token").displayName("SAS Token") - .description("Shared Access Signature token, including the leading '?'. Specify either SAS Token (recommended) or Account Key. " + - "There are certain risks in allowing the SAS token to be stored as a flowfile " + - "attribute. While it does provide for a more flexible flow by allowing the account name to " + - "be fetched dynamically from a flowfile attribute, care must be taken to restrict access to " + - "the event provenance data (e.g. by strictly controlling the policies governing provenance for this Processor). " + - "In addition, the provenance repositories may be put on encrypted disk partitions.") - .required(false) - .expressionLanguageSupported(ExpressionLanguageScope.FLOWFILE_ATTRIBUTES) - .sensitive(true) - .addValidator(StandardValidators.NON_EMPTY_VALIDATOR) - .build(); - - public static final PropertyDescriptor USE_MANAGED_IDENTITY = new PropertyDescriptor.Builder() - .name("use-managed-identity") - .displayName("Use Azure Managed Identity") - .description("Choose whether or not to use the managed identity of Azure VM/VMSS ") - .required(false).defaultValue("false").allowableValues("true", "false") - .addValidator(StandardValidators.BOOLEAN_VALIDATOR).build(); + public static final PropertyDescriptor ADLS_CREDENTIALS_SERVICE = new PropertyDescriptor.Builder() + .name("adls-credentials-service") + .displayName("ADLS Credentials") + .description("Controller Service used to obtain Azure Credentials.") + .identifiesControllerService(ADLSCredentialsService.class) + .required(true) + .build(); public static final PropertyDescriptor FILESYSTEM = new PropertyDescriptor.Builder() .name("filesystem-name").displayName("Filesystem Name") @@ -119,15 +79,6 @@ public abstract class AbstractAzureDataLakeStorageProcessor extends AbstractProc .defaultValue("${azure.filename}") .build(); - public static final PropertyDescriptor ENDPOINT_SUFFIX = new PropertyDescriptor.Builder() - .name("endpoint-suffix").displayName("Endpoint Suffix") - .description("Endpoint Suffix") - .addValidator(StandardValidators.NON_EMPTY_VALIDATOR) - .expressionLanguageSupported(ExpressionLanguageScope.FLOWFILE_ATTRIBUTES) - .required(false) - .defaultValue("dfs.core.windows.net") - .build(); - public static final Relationship REL_SUCCESS = new Relationship.Builder().name("success").description( "Files that have been successfully written to Azure storage are transferred to this relationship") .build(); @@ -135,66 +86,38 @@ public abstract class AbstractAzureDataLakeStorageProcessor extends AbstractProc "Files that could not be written to Azure storage for some reason are transferred to this relationship") .build(); - private static final List PROPERTIES = Collections.unmodifiableList( - Arrays.asList(AbstractAzureDataLakeStorageProcessor.ACCOUNT_NAME, - AbstractAzureDataLakeStorageProcessor.ACCOUNT_KEY, - AbstractAzureDataLakeStorageProcessor.SAS_TOKEN, - AbstractAzureDataLakeStorageProcessor.USE_MANAGED_IDENTITY, - AbstractAzureDataLakeStorageProcessor.ENDPOINT_SUFFIX, - AbstractAzureDataLakeStorageProcessor.FILESYSTEM, - AbstractAzureDataLakeStorageProcessor.DIRECTORY, - AbstractAzureDataLakeStorageProcessor.FILE)); + private static final List PROPERTIES = Collections.unmodifiableList(Arrays.asList( + ADLS_CREDENTIALS_SERVICE, + FILESYSTEM, + DIRECTORY, + FILE + )); - private static final Set RELATIONSHIPS = Collections.unmodifiableSet( - new HashSet<>(Arrays.asList( - AbstractAzureBlobProcessor.REL_SUCCESS, - AbstractAzureBlobProcessor.REL_FAILURE))); + private static final Set RELATIONSHIPS = Collections.unmodifiableSet(new HashSet<>(Arrays.asList( + REL_SUCCESS, + REL_FAILURE + ))); @Override protected List getSupportedPropertyDescriptors() { return PROPERTIES; } - public static Collection validateCredentialProperties(final ValidationContext validationContext) { - final List results = new ArrayList<>(); - - final boolean useManagedIdentity = validationContext.getProperty(USE_MANAGED_IDENTITY).asBoolean(); - final boolean accountKeyIsSet = validationContext.getProperty(ACCOUNT_KEY).isSet(); - final boolean sasTokenIsSet = validationContext.getProperty(SAS_TOKEN).isSet(); - - int credential_config_found = 0; - if(useManagedIdentity) credential_config_found++; - if(accountKeyIsSet) credential_config_found++; - if(sasTokenIsSet) credential_config_found++; - - if(credential_config_found == 0){ - final String msg = String.format( - "At least one of ['%s', '%s', '%s'] should be set", - ACCOUNT_KEY.getDisplayName(), - SAS_TOKEN.getDisplayName(), - USE_MANAGED_IDENTITY.getDisplayName() - ); - results.add(new ValidationResult.Builder().subject("Credentials config").valid(false).explanation(msg).build()); - } else if(credential_config_found > 1) { - final String msg = String.format( - "Only one of ['%s', '%s', '%s'] should be set", - ACCOUNT_KEY.getDisplayName(), - SAS_TOKEN.getDisplayName(), - USE_MANAGED_IDENTITY.getDisplayName() - ); - results.add(new ValidationResult.Builder().subject("Credentials config").valid(false).explanation(msg).build()); - } - return results; - } - public static DataLakeServiceClient getStorageClient(PropertyContext context, FlowFile flowFile) { final Map attributes = flowFile != null ? flowFile.getAttributes() : Collections.emptyMap(); - final String accountName = context.getProperty(ACCOUNT_NAME).evaluateAttributeExpressions(attributes).getValue(); - final String accountKey = context.getProperty(ACCOUNT_KEY).evaluateAttributeExpressions(attributes).getValue(); - final String sasToken = context.getProperty(SAS_TOKEN).evaluateAttributeExpressions(attributes).getValue(); - final String endpointSuffix = context.getProperty(ENDPOINT_SUFFIX).evaluateAttributeExpressions(attributes).getValue(); + + final ADLSCredentialsService credentialsService = context.getProperty(ADLS_CREDENTIALS_SERVICE).asControllerService(ADLSCredentialsService.class); + + ADLSCredentialsDetails credentialsDetails = credentialsService.getCredentialsDetails(attributes); + + final String accountName = credentialsDetails.getAccountName(); + final String accountKey = credentialsDetails.getAccountKey(); + final String sasToken = credentialsDetails.getSasToken(); + final AccessToken accessToken = credentialsDetails.getAccessToken(); + final String endpointSuffix = credentialsDetails.getEndpointSuffix(); + final boolean useManagedIdentity = credentialsDetails.getUseManagedIdentity(); + final String endpoint = String.format("https://%s.%s", accountName,endpointSuffix); - final boolean useManagedIdentity = context.getProperty(USE_MANAGED_IDENTITY).asBoolean(); DataLakeServiceClient storageClient; if (StringUtils.isNotBlank(accountKey)) { final StorageSharedKeyCredential credential = new StorageSharedKeyCredential(accountName, @@ -204,6 +127,11 @@ public abstract class AbstractAzureDataLakeStorageProcessor extends AbstractProc } else if (StringUtils.isNotBlank(sasToken)) { storageClient = new DataLakeServiceClientBuilder().endpoint(endpoint).sasToken(sasToken) .buildClient(); + } else if (accessToken != null) { + TokenCredential credential = tokenRequestContext -> Mono.just(accessToken); + + storageClient = new DataLakeServiceClientBuilder().endpoint(endpoint).credential(credential) + .buildClient(); } else if(useManagedIdentity){ final ManagedIdentityCredential misCrendential = new ManagedIdentityCredentialBuilder() .build(); @@ -212,16 +140,10 @@ public abstract class AbstractAzureDataLakeStorageProcessor extends AbstractProc .credential(misCrendential) .buildClient(); } else { - throw new IllegalArgumentException(String.format("Either '%s' or '%s' must be defined.", - ACCOUNT_KEY.getDisplayName(), SAS_TOKEN.getDisplayName())); + throw new IllegalArgumentException("No valid credentials were provided"); } - return storageClient; - } - @Override - protected Collection customValidate(final ValidationContext validationContext) { - final Collection results = validateCredentialProperties(validationContext); - return results; + return storageClient; } @Override diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/storage/utils/AzureStorageUtils.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/storage/utils/AzureStorageUtils.java index ffd732ab34..34be7a305a 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/storage/utils/AzureStorageUtils.java +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/processors/azure/storage/utils/AzureStorageUtils.java @@ -49,8 +49,13 @@ public final class AzureStorageUtils { public static final String BLOCK = "Block"; public static final String PAGE = "Page"; + public static final String STORAGE_ACCOUNT_NAME_PROPERTY_DESCRIPTOR_NAME = "storage-account-name"; + public static final String STORAGE_ACCOUNT_KEY_PROPERTY_DESCRIPTOR_NAME = "storage-account-key"; + public static final String STORAGE_SAS_TOKEN_PROPERTY_DESCRIPTOR_NAME = "storage-sas-token"; + public static final String STORAGE_ENDPOINT_SUFFIX_PROPERTY_DESCRIPTOR_NAME = "storage-endpoint-suffix"; + public static final PropertyDescriptor ACCOUNT_KEY = new PropertyDescriptor.Builder() - .name("storage-account-key") + .name(STORAGE_ACCOUNT_KEY_PROPERTY_DESCRIPTOR_NAME) .displayName("Storage Account Key") .description("The storage account key. This is an admin-like password providing access to every container in this account. It is recommended " + "one uses Shared Access Signature (SAS) token instead for fine-grained control with policies. " + @@ -73,7 +78,7 @@ public final class AzureStorageUtils { "In addition, the provenance repositories may be put on encrypted disk partitions."; public static final PropertyDescriptor ACCOUNT_NAME = new PropertyDescriptor.Builder() - .name("storage-account-name") + .name(STORAGE_ACCOUNT_NAME_PROPERTY_DESCRIPTOR_NAME) .displayName("Storage Account Name") .description(ACCOUNT_NAME_BASE_DESCRIPTION + " Instead of defining the Storage Account Name, Storage Account Key and SAS Token properties directly on the processor, " + @@ -87,11 +92,11 @@ public final class AzureStorageUtils { .build(); public static final PropertyDescriptor ENDPOINT_SUFFIX = new PropertyDescriptor.Builder() - .name("storage-endpoint-suffix") + .name(STORAGE_ENDPOINT_SUFFIX_PROPERTY_DESCRIPTOR_NAME) .displayName("Common Storage Account Endpoint Suffix") .description( "Storage accounts in public Azure always use a common FQDN suffix. " + - "Override this endpoint suffix with a different suffix in certain circumsances (like Azure Stack or non-public Azure regions). " + + "Override this endpoint suffix with a different suffix in certain circumstances (like Azure Stack or non-public Azure regions). " + "The preferred way is to configure them through a controller service specified in the Storage Credentials property. " + "The controller service can provide a common/shared configuration for multiple/all Azure processors. Furthermore, the credentials " + "can also be looked up dynamically with the 'Lookup' version of the service.") @@ -111,7 +116,7 @@ public final class AzureStorageUtils { .build(); public static final PropertyDescriptor PROP_SAS_TOKEN = new PropertyDescriptor.Builder() - .name("storage-sas-token") + .name(STORAGE_SAS_TOKEN_PROPERTY_DESCRIPTOR_NAME) .displayName("SAS Token") .description("Shared Access Signature token, including the leading '?'. Specify either SAS Token (recommended) or Account Key. " + "There are certain risks in allowing the SAS token to be stored as a flowfile " + diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsControllerService.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsControllerService.java new file mode 100644 index 0000000000..ec86f9ce3f --- /dev/null +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsControllerService.java @@ -0,0 +1,153 @@ +/* + * 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.services.azure.storage; + +import org.apache.commons.lang3.StringUtils; +import org.apache.nifi.annotation.documentation.CapabilityDescription; +import org.apache.nifi.annotation.documentation.Tags; +import org.apache.nifi.annotation.lifecycle.OnEnabled; +import org.apache.nifi.components.PropertyDescriptor; +import org.apache.nifi.components.PropertyValue; +import org.apache.nifi.components.ValidationContext; +import org.apache.nifi.components.ValidationResult; +import org.apache.nifi.controller.AbstractControllerService; +import org.apache.nifi.controller.ConfigurationContext; +import org.apache.nifi.processor.util.StandardValidators; +import org.apache.nifi.processors.azure.storage.utils.AzureStorageUtils; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collection; +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.StringJoiner; +import java.util.function.BiConsumer; +import java.util.function.Function; + +/** + * Provides credentials details for ADLS + * + * @see AbstractControllerService + */ +@Tags({"azure", "microsoft", "cloud", "storage", "adls", "credentials"}) +@CapabilityDescription("Defines credentials for ADLS processors.") +public class ADLSCredentialsControllerService extends AbstractControllerService implements ADLSCredentialsService { + + public static final PropertyDescriptor ACCOUNT_NAME = new PropertyDescriptor.Builder() + .fromPropertyDescriptor(AzureStorageUtils.ACCOUNT_NAME) + .description(AzureStorageUtils.ACCOUNT_NAME_BASE_DESCRIPTION) + .required(true) + .build(); + + public static final PropertyDescriptor ENDPOINT_SUFFIX = new PropertyDescriptor.Builder() + .fromPropertyDescriptor(AzureStorageUtils.ENDPOINT_SUFFIX) + .displayName("Endpoint Suffix") + .description( + "Storage accounts in public Azure always use a common FQDN suffix. " + + "Override this endpoint suffix with a different suffix in certain circumstances (like Azure Stack or non-public Azure regions).") + .required(true) + .defaultValue("dfs.core.windows.net") + .build(); + + public static final PropertyDescriptor USE_MANAGED_IDENTITY = new PropertyDescriptor.Builder() + .name("storage-use-managed-identity") + .displayName("Use Azure Managed Identity") + .description("Choose whether or not to use the managed identity of Azure VM/VMSS ") + .required(false) + .defaultValue("false") + .allowableValues("true", "false") + .addValidator(StandardValidators.BOOLEAN_VALIDATOR) + .build(); + + private static final List PROPERTIES = Collections.unmodifiableList(Arrays.asList( + ACCOUNT_NAME, + ENDPOINT_SUFFIX, + AzureStorageUtils.ACCOUNT_KEY, + AzureStorageUtils.PROP_SAS_TOKEN, + USE_MANAGED_IDENTITY + )); + + private ConfigurationContext context; + + @Override + protected List getSupportedPropertyDescriptors() { + return PROPERTIES; + } + + @Override + protected Collection customValidate(ValidationContext validationContext) { + final List results = new ArrayList<>(); + + boolean accountKeySet = StringUtils.isNotBlank(validationContext.getProperty(AzureStorageUtils.ACCOUNT_KEY).getValue()); + boolean sasTokenSet = StringUtils.isNotBlank(validationContext.getProperty(AzureStorageUtils.PROP_SAS_TOKEN).getValue()); + boolean useManagedIdentitySet = validationContext.getProperty(USE_MANAGED_IDENTITY).asBoolean(); + + if (!onlyOneSet(accountKeySet, sasTokenSet, useManagedIdentitySet)) { + StringJoiner options = new StringJoiner(", ") + .add(AzureStorageUtils.ACCOUNT_KEY.getDisplayName()) + .add(AzureStorageUtils.PROP_SAS_TOKEN.getDisplayName()) + .add(USE_MANAGED_IDENTITY.getDisplayName()); + + results.add(new ValidationResult.Builder().subject(this.getClass().getSimpleName()) + .valid(false) + .explanation("one and only one of [" + options + "] should be set") + .build()); + } + + return results; + } + + private boolean onlyOneSet(Boolean... checks) { + long nrOfSet = Arrays.stream(checks) + .filter(check -> check) + .count(); + + return nrOfSet == 1; + } + + @OnEnabled + public void onEnabled(ConfigurationContext context) { + this.context = context; + } + + @Override + public ADLSCredentialsDetails getCredentialsDetails(Map attributes) { + ADLSCredentialsDetails.Builder credentialsBuilder = ADLSCredentialsDetails.Builder.newBuilder(); + + setValue(credentialsBuilder, ACCOUNT_NAME, PropertyValue::getValue, ADLSCredentialsDetails.Builder::setAccountName); + setValue(credentialsBuilder, AzureStorageUtils.ACCOUNT_KEY, PropertyValue::getValue, ADLSCredentialsDetails.Builder::setAccountKey); + setValue(credentialsBuilder, AzureStorageUtils.PROP_SAS_TOKEN, PropertyValue::getValue, ADLSCredentialsDetails.Builder::setSasToken); + setValue(credentialsBuilder, ENDPOINT_SUFFIX, PropertyValue::getValue, ADLSCredentialsDetails.Builder::setEndpointSuffix); + setValue(credentialsBuilder, USE_MANAGED_IDENTITY, PropertyValue::asBoolean, ADLSCredentialsDetails.Builder::setUseManagedIdentity); + + return credentialsBuilder.build(); + } + + private void setValue( + ADLSCredentialsDetails.Builder credentialsBuilder, + PropertyDescriptor propertyDescriptor, Function getPropertyValue, + BiConsumer setBuilderValue + ) { + PropertyValue property = context.getProperty(propertyDescriptor); + + if (property.isSet()) { + T value = getPropertyValue.apply(property); + setBuilderValue.accept(credentialsBuilder, value); + } + } +} diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/resources/META-INF/services/org.apache.nifi.controller.ControllerService b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/resources/META-INF/services/org.apache.nifi.controller.ControllerService index 659452b5c9..4179771dfc 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/resources/META-INF/services/org.apache.nifi.controller.ControllerService +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/main/resources/META-INF/services/org.apache.nifi.controller.ControllerService @@ -14,3 +14,4 @@ # limitations under the License. org.apache.nifi.services.azure.storage.AzureStorageCredentialsControllerService org.apache.nifi.services.azure.storage.AzureStorageCredentialsControllerServiceLookup +org.apache.nifi.services.azure.storage.ADLSCredentialsControllerService \ No newline at end of file diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureDataLakeStorageIT.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureDataLakeStorageIT.java index e0b591a5d2..da03eae96b 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureDataLakeStorageIT.java +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureDataLakeStorageIT.java @@ -23,6 +23,9 @@ import com.azure.storage.file.datalake.DataLakeFileSystemClient; import com.azure.storage.file.datalake.DataLakeServiceClient; import com.azure.storage.file.datalake.DataLakeServiceClientBuilder; import org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor; +import org.apache.nifi.processors.azure.storage.utils.AzureStorageUtils; +import org.apache.nifi.services.azure.storage.ADLSCredentialsControllerService; +import org.apache.nifi.services.azure.storage.ADLSCredentialsService; import org.junit.After; import org.junit.Before; @@ -36,6 +39,17 @@ public abstract class AbstractAzureDataLakeStorageIT extends AbstractAzureStorag protected String fileSystemName; protected DataLakeFileSystemClient fileSystemClient; + @Override + protected void setUpCredentials() throws Exception { + ADLSCredentialsService service = new ADLSCredentialsControllerService(); + runner.addControllerService("ADLSCredentials", service); + runner.setProperty(service, ADLSCredentialsControllerService.ACCOUNT_NAME, getAccountName()); + runner.setProperty(service, AzureStorageUtils.ACCOUNT_KEY, getAccountKey()); + runner.enableControllerService(service); + + runner.setProperty(AbstractAzureDataLakeStorageProcessor.ADLS_CREDENTIALS_SERVICE, "ADLSCredentials"); + } + @Before public void setUpAzureDataLakeStorageIT() { fileSystemName = String.format("%s-%s", FILESYSTEM_NAME_PREFIX, UUID.randomUUID()); diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureStorageIT.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureStorageIT.java index 7689b96d47..0ee349bcb2 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureStorageIT.java +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/AbstractAzureStorageIT.java @@ -68,9 +68,13 @@ public abstract class AbstractAzureStorageIT { protected TestRunner runner; @Before - public void setUpAzureStorageIT() { + public void setUpAzureStorageIT() throws Exception { runner = TestRunners.newTestRunner(getProcessorClass()); + setUpCredentials(); + } + + protected void setUpCredentials() throws Exception { runner.setProperty(AzureStorageUtils.ACCOUNT_NAME, getAccountName()); runner.setProperty(AzureStorageUtils.ACCOUNT_KEY, getAccountKey()); } diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/TestAbstractAzureDataLakeStorage.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/TestAbstractAzureDataLakeStorage.java index 960b1fea4f..94fc5761a8 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/TestAbstractAzureDataLakeStorage.java +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/processors/azure/storage/TestAbstractAzureDataLakeStorage.java @@ -16,14 +16,14 @@ */ package org.apache.nifi.processors.azure.storage; -import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.ACCOUNT_KEY; -import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.ACCOUNT_NAME; +import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.ADLS_CREDENTIALS_SERVICE; import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.DIRECTORY; import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.FILE; import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.FILESYSTEM; -import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.SAS_TOKEN; -import static org.apache.nifi.processors.azure.AbstractAzureDataLakeStorageProcessor.USE_MANAGED_IDENTITY; +import static org.mockito.Mockito.mock; +import static org.mockito.Mockito.when; +import org.apache.nifi.services.azure.storage.ADLSCredentialsService; import org.apache.nifi.util.TestRunner; import org.apache.nifi.util.TestRunners; import org.junit.Before; @@ -34,57 +34,20 @@ public class TestAbstractAzureDataLakeStorage { private TestRunner runner; @Before - public void setUp() { + public void setUp() throws Exception { // test the property validation in the abstract class via the put processor runner = TestRunners.newTestRunner(PutAzureDataLakeStorage.class); - runner.setProperty(ACCOUNT_NAME, "accountName"); - runner.setProperty(ACCOUNT_KEY, "accountKey"); + ADLSCredentialsService credentialsService = mock(ADLSCredentialsService.class); + when(credentialsService.getIdentifier()).thenReturn("credentials_service"); + runner.addControllerService("credentials_service", credentialsService); + runner.enableControllerService(credentialsService); + runner.setProperty(FILESYSTEM, "filesystem"); runner.setProperty(DIRECTORY, "directory"); runner.setProperty(FILE, "file"); - } + runner.setProperty(ADLS_CREDENTIALS_SERVICE, "credentials_service"); - @Test - public void testValidWhenAccountNameAndAccountKeySpecified() { - runner.assertValid(); - } - - @Test - public void testValidWhenAccountNameAndSasTokenSpecified() { - runner.removeProperty(ACCOUNT_KEY); - runner.setProperty(SAS_TOKEN, "sasToken"); - - runner.assertValid(); - } - - @Test - public void testValidWhenAccountNameAndUseManagedIdentity() { - runner.removeProperty(ACCOUNT_KEY); - runner.setProperty(USE_MANAGED_IDENTITY, "true"); - - runner.assertValid(); - } - - @Test - public void testNotValidWhenNoAccountNameSpecified() { - runner.removeProperty(ACCOUNT_NAME); - - runner.assertNotValid(); - } - - @Test - public void testNotValidWhenNoAccountKeyNorSasTokenSpecified() { - runner.removeProperty(ACCOUNT_KEY); - - runner.assertNotValid(); - } - - @Test - public void testNotValidWhenBothAccountKeyAndSasTokenSpecified() { - runner.setProperty(SAS_TOKEN, "sasToken"); - - runner.assertNotValid(); } @Test diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/services/azure/storage/TestADLSCredentialsControllerService.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/services/azure/storage/TestADLSCredentialsControllerService.java new file mode 100644 index 0000000000..aa51c4c861 --- /dev/null +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-processors/src/test/java/org/apache/nifi/services/azure/storage/TestADLSCredentialsControllerService.java @@ -0,0 +1,242 @@ +/* + * 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.services.azure.storage; + +import org.apache.nifi.processors.azure.storage.utils.AzureStorageUtils; +import org.apache.nifi.reporting.InitializationException; +import org.apache.nifi.util.NoOpProcessor; +import org.apache.nifi.util.TestRunner; +import org.apache.nifi.util.TestRunners; +import org.junit.Before; +import org.junit.Test; + +import java.util.HashMap; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertTrue; + +public class TestADLSCredentialsControllerService { + + public static final String CREDENTIALS_SERVICE_IDENTIFIER = "credentials-service"; + + private static final String ACCOUNT_NAME_VALUE = "AccountName"; + private static final String ACCOUNT_KEY_VALUE = "AccountKey"; + private static final String SAS_TOKEN_VALUE = "SasToken"; + public static final String END_POINT_SUFFIX_VALUE = "end.point.suffix"; + + private TestRunner runner; + private ADLSCredentialsControllerService credentialsService; + + @Before + public void setUp() throws InitializationException { + runner = TestRunners.newTestRunner(NoOpProcessor.class); + credentialsService = new ADLSCredentialsControllerService(); + runner.addControllerService(CREDENTIALS_SERVICE_IDENTIFIER, credentialsService); + } + + @Test + public void testNotValidBecauseAccountNameMissing() { + configureAccountKey(); + + runner.assertNotValid(credentialsService); + } + + @Test + public void testNotValidBecauseNoCredentialsIsSet() { + configureAccountName(); + + runner.assertNotValid(credentialsService); + } + + @Test + public void testNotValidBecauseBothAccountKeyAndSasTokenSpecified() { + configureAccountName(); + + configureAccountKey(); + configureSasToken(); + + runner.assertNotValid(credentialsService); + } + + @Test + public void testNotValidBecauseBothAccountKeyAndUseManagedIdentitySpecified() { + configureAccountName(); + + configureAccountKey(); + configureUseManagedIdentity(); + + runner.assertNotValid(credentialsService); + } + + @Test + public void testNotValidBecauseBothSasTokenAndUseManagedIdentitySpecified() { + configureAccountName(); + + configureSasToken(); + configureUseManagedIdentity(); + + runner.assertNotValid(credentialsService); + } + + @Test + public void testNotValidBecauseAllCredentialsSpecified() { + configureAccountName(); + + configureAccountKey(); + configureSasToken(); + configureUseManagedIdentity(); + + runner.assertNotValid(credentialsService); + } + + @Test + public void testNotValidWithEmptyEndpointSuffix() { + configureAccountName(); + configureAccountKey(); + + runner.setProperty(credentialsService, ADLSCredentialsControllerService.ENDPOINT_SUFFIX, ""); + runner.assertNotValid(credentialsService); + } + @Test + public void testNotValidWithWhitespaceEndpointSuffix() { + configureAccountName(); + configureAccountKey(); + + runner.setProperty(credentialsService, ADLSCredentialsControllerService.ENDPOINT_SUFFIX, " "); + runner.assertNotValid(credentialsService); + } + + @Test + public void testValidWithAccountNameAndAccountKey() { + configureAccountName(); + configureAccountKey(); + + runner.assertValid(credentialsService); + } + + @Test + public void testValidWithAccountNameAndSasToken() { + configureAccountName(); + configureSasToken(); + + runner.assertValid(credentialsService); + } + + @Test + public void testValidWithAccountNameAndUseManagedIdentity() { + configureAccountName(); + configureUseManagedIdentity(); + + runner.assertValid(credentialsService); + } + + @Test + public void testGetCredentialsDetailsWithAccountKey() throws Exception { + // GIVEN + configureAccountName(); + configureAccountKey(); + + runner.enableControllerService(credentialsService); + + // WHEN + ADLSCredentialsDetails actual = credentialsService.getCredentialsDetails(new HashMap<>()); + + // THEN + assertEquals(ACCOUNT_NAME_VALUE, actual.getAccountName()); + assertEquals(ACCOUNT_KEY_VALUE, actual.getAccountKey()); + assertNull(actual.getSasToken()); + assertFalse(actual.getUseManagedIdentity()); + assertNotNull(actual.getEndpointSuffix()); + } + + @Test + public void testGetCredentialsDetailsWithSasToken() throws Exception { + // GIVEN + configureAccountName(); + configureSasToken(); + + runner.enableControllerService(credentialsService); + + // WHEN + ADLSCredentialsDetails actual = credentialsService.getCredentialsDetails(new HashMap<>()); + + // THEN + assertEquals(ACCOUNT_NAME_VALUE, actual.getAccountName()); + assertEquals(SAS_TOKEN_VALUE, actual.getSasToken()); + assertNull(actual.getAccountKey()); + assertFalse(actual.getUseManagedIdentity()); + assertNotNull(actual.getEndpointSuffix()); + } + + @Test + public void testGetCredentialsDetailsWithUseManagedIdentity() throws Exception { + // GIVEN + configureAccountName(); + configureUseManagedIdentity(); + + runner.enableControllerService(credentialsService); + + // WHEN + ADLSCredentialsDetails actual = credentialsService.getCredentialsDetails(new HashMap<>()); + + // THEN + assertEquals(ACCOUNT_NAME_VALUE, actual.getAccountName()); + assertTrue(actual.getUseManagedIdentity()); + assertNull(actual.getAccountKey()); + assertNull(actual.getSasToken()); + assertNotNull(actual.getEndpointSuffix()); + } + + @Test + public void testGetCredentialsDetailsWithSetEndpointSuffix() throws Exception { + // GIVEN + configureAccountName(); + configureAccountKey(); + configureEndpointSuffix(); + + runner.enableControllerService(credentialsService); + + // WHEN + ADLSCredentialsDetails actual = credentialsService.getCredentialsDetails(new HashMap<>()); + + // THEN + assertEquals(END_POINT_SUFFIX_VALUE, actual.getEndpointSuffix()); + } + + private void configureAccountName() { + runner.setProperty(credentialsService, ADLSCredentialsControllerService.ACCOUNT_NAME, ACCOUNT_NAME_VALUE); + } + + private void configureAccountKey() { + runner.setProperty(credentialsService, AzureStorageUtils.ACCOUNT_KEY, ACCOUNT_KEY_VALUE); + } + + private void configureSasToken() { + runner.setProperty(credentialsService, AzureStorageUtils.PROP_SAS_TOKEN, SAS_TOKEN_VALUE); + } + + private void configureUseManagedIdentity() { + runner.setProperty(credentialsService, ADLSCredentialsControllerService.USE_MANAGED_IDENTITY, "true"); + } + + private void configureEndpointSuffix() { + runner.setProperty(credentialsService, ADLSCredentialsControllerService.ENDPOINT_SUFFIX, END_POINT_SUFFIX_VALUE); + } +} diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/pom.xml b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/pom.xml index d3e8cd90e4..99a13dbd08 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/pom.xml +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/pom.xml @@ -28,6 +28,11 @@ com.microsoft.azure azure-storage + + com.azure + azure-core + ${azure.core.version} + com.fasterxml.jackson.core diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsDetails.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsDetails.java new file mode 100644 index 0000000000..cd1111e5be --- /dev/null +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsDetails.java @@ -0,0 +1,120 @@ +/* + * 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.services.azure.storage; + +import com.azure.core.credential.AccessToken; + +public class ADLSCredentialsDetails { + private final String accountName; + + private final String accountKey; + private final String sasToken; + private final String endpointSuffix; + + private final AccessToken accessToken; + + private final boolean useManagedIdentity; + + public ADLSCredentialsDetails( + String accountName, + String accountKey, + String sasToken, + String endpointSuffix, + AccessToken accessToken, + boolean useManagedIdentity + ) { + this.accountName = accountName; + this.accountKey = accountKey; + this.sasToken = sasToken; + this.endpointSuffix = endpointSuffix; + this.accessToken = accessToken; + this.useManagedIdentity = useManagedIdentity; + } + + public String getAccountName() { + return accountName; + } + + public String getEndpointSuffix() { + return endpointSuffix; + } + + public String getAccountKey() { + return accountKey; + } + + public String getSasToken() { + return sasToken; + } + + public AccessToken getAccessToken() { + return accessToken; + } + + public boolean getUseManagedIdentity() { + return useManagedIdentity; + } + + public static class Builder { + private String accountName; + private String accountKey; + private String sasToken; + private String endpointSuffix; + private AccessToken accessToken; + private boolean useManagedIdentity; + + private Builder() {} + + public static Builder newBuilder() { + return new Builder(); + } + + public Builder setAccountName(String accountName) { + this.accountName = accountName; + return this; + } + + public Builder setAccountKey(String accountKey) { + this.accountKey = accountKey; + return this; + } + + public Builder setSasToken(String sasToken) { + this.sasToken = sasToken; + return this; + } + + public Builder setEndpointSuffix(String endpointSuffix) { + this.endpointSuffix = endpointSuffix; + return this; + } + + public Builder setAccessToken(AccessToken accessToken) { + this.accessToken = accessToken; + return this; + } + + public Builder setUseManagedIdentity(boolean useManagedIdentity) { + this.useManagedIdentity = useManagedIdentity; + return this; + } + + public ADLSCredentialsDetails build() { + return new ADLSCredentialsDetails(accountName, accountKey, sasToken, endpointSuffix, accessToken, useManagedIdentity); + } + } +} diff --git a/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsService.java b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsService.java new file mode 100644 index 0000000000..53d17a4ac6 --- /dev/null +++ b/nifi-nar-bundles/nifi-azure-bundle/nifi-azure-services-api/src/main/java/org/apache/nifi/services/azure/storage/ADLSCredentialsService.java @@ -0,0 +1,35 @@ +/* + * 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.services.azure.storage; + +import org.apache.nifi.controller.ControllerService; + +import java.util.Map; + +/** + * + * Service interface to provide ADLS credentials details. + */ +public interface ADLSCredentialsService extends ControllerService { + + /** + * Get ADLSCredentialsDetails object which contains the Storage Account Name and one or more credentials type + * @param attributes FlowFile attributes (typically) + * @return ADLSCredentialsDetails object + */ + ADLSCredentialsDetails getCredentialsDetails(Map attributes); +} diff --git a/nifi-nar-bundles/nifi-azure-bundle/pom.xml b/nifi-nar-bundles/nifi-azure-bundle/pom.xml index c5ecc49a56..13e1e6d4ac 100644 --- a/nifi-nar-bundles/nifi-azure-bundle/pom.xml +++ b/nifi-nar-bundles/nifi-azure-bundle/pom.xml @@ -27,6 +27,7 @@ 8.4.0 + 1.5.0 2.10.3