NIFI-6008: When a Processor is Terminated, call its #onPropertyModified method for properties whose value differs from the default value. Previously, it was called if the value differs from the current value (which it never will since we are only restoring the current values for the Processor). This results in ensuring that #onPr #onPropertyModified is correctly called when a component is reloaded

This closes #3296.

Signed-off-by: Bryan Bende <bbende@apache.org>
This commit is contained in:
Mark Payne 2019-02-07 16:56:22 -05:00 committed by Bryan Bende
parent e4be06c7c8
commit 4e914cea1f
No known key found for this signature in database
GPG Key ID: A0DDA9ED50711C39
1 changed files with 15 additions and 6 deletions

View File

@ -56,6 +56,7 @@ import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicReference; import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; import java.util.concurrent.locks.ReentrantLock;
import java.util.function.Function;
import java.util.stream.Collectors; import java.util.stream.Collectors;
public abstract class AbstractComponentNode implements ComponentNode { public abstract class AbstractComponentNode implements ComponentNode {
@ -185,7 +186,8 @@ public abstract class AbstractComponentNode implements ComponentNode {
if (entry.getKey() != null && entry.getValue() == null) { if (entry.getKey() != null && entry.getValue() == null) {
removeProperty(entry.getKey(), allowRemovalOfRequiredProperties); removeProperty(entry.getKey(), allowRemovalOfRequiredProperties);
} else if (entry.getKey() != null) { } else if (entry.getKey() != null) {
setProperty(entry.getKey(), CharacterFilterUtils.filterInvalidXmlCharacters(entry.getValue())); final String updatedValue = CharacterFilterUtils.filterInvalidXmlCharacters(entry.getValue());
setProperty(entry.getKey(), updatedValue, this.properties::get);
} }
} }
@ -210,16 +212,16 @@ public abstract class AbstractComponentNode implements ComponentNode {
} }
// Keep setProperty/removeProperty private so that all calls go through setProperties // Keep setProperty/removeProperty private so that all calls go through setProperties
private void setProperty(final String name, final String value) { private void setProperty(final String name, final String value, final Function<PropertyDescriptor, String> valueToCompareFunction) {
if (null == name || null == value) { if (null == name || null == value) {
throw new IllegalArgumentException("Name or Value can not be null"); throw new IllegalArgumentException("Name or Value can not be null");
} }
final PropertyDescriptor descriptor = getComponent().getPropertyDescriptor(name); final PropertyDescriptor descriptor = getComponent().getPropertyDescriptor(name);
final String propertyModComparisonValue = valueToCompareFunction.apply(descriptor);
final String oldValue = properties.put(descriptor, value); final String oldValue = properties.put(descriptor, value);
if (!value.equals(oldValue)) {
if (!value.equals(oldValue)) {
if (descriptor.getControllerServiceDefinition() != null) { if (descriptor.getControllerServiceDefinition() != null) {
if (oldValue != null) { if (oldValue != null) {
final ControllerServiceNode oldNode = serviceProvider.getControllerServiceNode(oldValue); final ControllerServiceNode oldNode = serviceProvider.getControllerServiceNode(oldValue);
@ -233,7 +235,12 @@ public abstract class AbstractComponentNode implements ComponentNode {
newNode.addReference(this); newNode.addReference(this);
} }
} }
}
// In the case of a component "reload", we want to call onPropertyModified when the value is changed from the descriptor's default.
// However, we do not want to update any controller service references because those are tied to the ComponentNode. We only want to
// allow the newly created component's internal state to be updated.
if (!value.equals(propertyModComparisonValue)) {
try { try {
onPropertyModified(descriptor, oldValue, value); onPropertyModified(descriptor, oldValue, value);
} catch (final Exception e) { } catch (final Exception e) {
@ -309,10 +316,12 @@ public abstract class AbstractComponentNode implements ComponentNode {
@Override @Override
public void refreshProperties() { public void refreshProperties() {
// use setProperty instead of setProperties so we can bypass the class loading logic // use setProperty instead of setProperties so we can bypass the class loading logic.
// Consider value changed if it is different than the PropertyDescriptor's default value because we need to call the #onPropertiesModified
// method on the component if the current value is not the default value, since the component itself is being reloaded.
getProperties().entrySet().stream() getProperties().entrySet().stream()
.filter(e -> e.getKey() != null && e.getValue() != null) .filter(e -> e.getKey() != null && e.getValue() != null)
.forEach(e -> setProperty(e.getKey().getName(), e.getValue())); .forEach(e -> setProperty(e.getKey().getName(), e.getValue(), PropertyDescriptor::getDefaultValue));
} }
/** /**