NIFI-12982 Extend test suite of MockProcessSession

This closes #8589

Signed-off-by: David Handermann <exceptionfactory@apache.org>
This commit is contained in:
EndzeitBegins 2024-03-31 22:06:23 +02:00 committed by exceptionfactory
parent 98c4061cfe
commit 6939ffc08d
No known key found for this signature in database
1 changed files with 556 additions and 150 deletions

View File

@ -16,6 +16,9 @@
*/ */
package org.apache.nifi.util; package org.apache.nifi.util;
import org.apache.nifi.annotation.behavior.Stateful;
import org.apache.nifi.components.state.Scope;
import org.apache.nifi.components.state.StateMap;
import org.apache.nifi.flowfile.FlowFile; import org.apache.nifi.flowfile.FlowFile;
import org.apache.nifi.flowfile.attributes.CoreAttributes; import org.apache.nifi.flowfile.attributes.CoreAttributes;
import org.apache.nifi.processor.AbstractProcessor; import org.apache.nifi.processor.AbstractProcessor;
@ -26,17 +29,21 @@ import org.apache.nifi.processor.Relationship;
import org.apache.nifi.processor.exception.FlowFileHandlingException; import org.apache.nifi.processor.exception.FlowFileHandlingException;
import org.apache.nifi.processor.exception.ProcessException; import org.apache.nifi.processor.exception.ProcessException;
import org.apache.nifi.state.MockStateManager; import org.apache.nifi.state.MockStateManager;
import org.junit.jupiter.api.Disabled;
import org.junit.jupiter.api.Nested;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import java.io.EOFException;
import java.io.IOException; import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.util.Collections; import java.io.OutputStream;
import java.util.Collection;
import java.util.List; import java.util.List;
import java.util.Map;
import java.util.Set; import java.util.Set;
import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicLong;
import java.util.regex.Pattern; import java.util.regex.Pattern;
import static org.junit.jupiter.api.Assertions.assertDoesNotThrow;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse; import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertThrows; import static org.junit.jupiter.api.Assertions.assertThrows;
@ -45,189 +52,588 @@ import static org.junit.jupiter.api.Assertions.fail;
public class TestMockProcessSession { public class TestMockProcessSession {
private final Processor processor = new TestProcessor();
private final SharedSessionState sharedState = new SharedSessionState(processor, new AtomicLong(0L));
private final MockStateManager stateManager = new MockStateManager(processor);
private final MockProcessSession session = new MockProcessSession(sharedState, processor, stateManager);
private final Processor statefulProcessor = new StatefulTestProcessor();
private final SharedSessionState sharedStateOfStatefulProcessor = new SharedSessionState(statefulProcessor, new AtomicLong(0L));
private final MockStateManager stateManagerOfStatefulProcessor = new MockStateManager(statefulProcessor);
private final MockProcessSession sessionOfStatefulProcessor = new MockProcessSession(sharedStateOfStatefulProcessor, statefulProcessor, stateManagerOfStatefulProcessor);
@Nested
class RegardingActiveReads {
@Test @Test
public void testReadWithoutCloseThrowsExceptionOnCommit() throws IOException { void cannotTransferFlowFileThatIsReadActively() throws IOException {
final Processor processor = new PoorlyBehavedProcessor(); MockFlowFile flowFile = session.createFlowFile("hello, world".getBytes());
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, true, new MockStateManager(processor), true); readWithoutClosingInputStream(session, flowFile);
FlowFile flowFile = session.createFlowFile("hello, world".getBytes());
final InputStream in = session.read(flowFile);
final byte[] buffer = new byte[12];
fillBuffer(in, buffer);
assertEquals("hello, world", new String(buffer)); assertThrows(IllegalStateException.class, () -> {
session.transfer(flowFile, TestProcessor.REL_KNOWN);
try { }, "Was able to transfer FlowFile without closing InputStream");
session.commit();
fail("Was able to commit session without closing InputStream");
} catch (final FlowFileHandlingException | IllegalStateException e) {
System.out.println(e.toString());
}
}
private int fillBuffer(final InputStream source, final byte[] destination) throws IOException {
int bytesRead = 0;
int len;
while (bytesRead < destination.length) {
len = source.read(destination, bytesRead, destination.length - bytesRead);
if (len < 0) {
throw new EOFException("Expected to read " + destination.length + " bytes but encountered EOF after " + bytesRead + " bytes");
}
bytesRead += len;
}
return bytesRead;
} }
@Test @Test
public void testReadWithoutCloseThrowsExceptionOnCommitAsync() throws IOException { void cannotRemoveFlowFileThatIsReadActively() throws IOException {
final Processor processor = new PoorlyBehavedProcessor(); MockFlowFile flowFile = session.createFlowFile("hello, world".getBytes());
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); readWithoutClosingInputStream(session, flowFile);
FlowFile flowFile = session.createFlowFile("hello, world".getBytes());
final InputStream in = session.read(flowFile);
final byte[] buffer = new byte[12];
fillBuffer(in, buffer);
assertEquals("hello, world", new String(buffer)); assertThrows(IllegalStateException.class, () -> {
session.remove(flowFile);
}, "Was able to remove FlowFile without closing InputStream");
}
@Test
void cannotMergeWithFlowFileThatIsReadActively() throws IOException {
MockFlowFile offendingFlowFile = session.createFlowFile("hello, world".getBytes());
readWithoutClosingInputStream(session, offendingFlowFile);
MockFlowFile otherFlowFile = session.createFlowFile("Hola mundo".getBytes());
MockFlowFile destinationFlowFile = session.create();
assertThrows(IllegalStateException.class, () -> {
session.merge(Set.of(offendingFlowFile, otherFlowFile), destinationFlowFile);
}, "Was able to merge FlowFile without closing InputStream");
}
@Test
void cannotMigrateFlowFileThatIsReadActively() throws IOException {
final MockProcessSession targetSession = createMockProcessSession();
MockFlowFile offendingFlowFile = session.createFlowFile("hello, world".getBytes());
readWithoutClosingInputStream(session, offendingFlowFile);
assertThrows(IllegalStateException.class, () -> {
session.migrate(targetSession);
}, "Was able to merge FlowFile without closing InputStream");
}
private static void readWithoutClosingInputStream(MockProcessSession session, MockFlowFile flowFile) throws IOException {
String expectedContent = flowFile.getContent();
@SuppressWarnings("resource") final InputStream in = session.read(flowFile);
final byte[] bytes = in.readAllBytes();
assertEquals(expectedContent, new String(bytes));
}
}
@Nested
class RegardingActiveWrites {
@Test
void cannotTransferFlowFileThatIsWrittenActively() throws IOException {
MockFlowFile flowFile = session.create();
writeWithoutClosingOutputStream(session, flowFile);
assertThrows(IllegalStateException.class, () -> {
session.transfer(flowFile, TestProcessor.REL_KNOWN);
}, "Was able to transfer FlowFile without closing OutputStream");
}
@Test
void cannotRemoveFlowFileThatIsWrittenActively() throws IOException {
MockFlowFile flowFile = session.create();
writeWithoutClosingOutputStream(session, flowFile);
assertThrows(IllegalStateException.class, () -> {
session.remove(flowFile);
}, "Was able to remove FlowFile without closing OutputStream");
}
@Test
void cannotMergeWithFlowFileThatIsWrittenActively() throws IOException {
MockFlowFile offendingFlowFile = session.createFlowFile("hello, world".getBytes());
writeWithoutClosingOutputStream(session, offendingFlowFile);
MockFlowFile otherFlowFile = session.createFlowFile("Hola mundo".getBytes());
MockFlowFile destinationFlowFile = session.create();
assertThrows(IllegalStateException.class, () -> {
session.merge(Set.of(offendingFlowFile, otherFlowFile), destinationFlowFile);
}, "Was able to merge FlowFile without closing OutputStream");
}
@Test
void cannotMigrateFlowFileThatIsWrittenActively() throws IOException {
final MockProcessSession targetSession = createMockProcessSession();
MockFlowFile offendingFlowFile = session.create();
writeWithoutClosingOutputStream(session, offendingFlowFile);
assertThrows(IllegalStateException.class, () -> {
session.migrate(targetSession);
}, "Was able to merge FlowFile without closing OutputStream");
}
private static void writeWithoutClosingOutputStream(MockProcessSession session, MockFlowFile flowFile) throws IOException {
@SuppressWarnings("resource") final OutputStream outputStream = session.write(flowFile);
outputStream.write("some content".getBytes());
}
}
@Nested
class RegardingUnaccountedFlowFiles {
@Test
void cannotCommitWithUnaccountedCreatedFlowFile() {
session.create(); // unaccounted for
assertThrows(FlowFileHandlingException.class, session::commitAsync, "Was able to commit with unaccounted newly created FlowFile");
}
@Test
void cannotCommitWithUnaccountedClonedFlowFile() {
MockFlowFile flowFile = session.create();
session.clone(flowFile); // unaccounted for
session.transfer(flowFile, TestProcessor.REL_KNOWN);
assertThrows(FlowFileHandlingException.class, session::commitAsync, "Was able to commit with unaccounted cloned FlowFile");
}
@Test
void cannotCommitWithUnaccountedMigratedFlowFile() {
MockProcessSession targetSession = createMockProcessSession();
session.create();
session.migrate(targetSession);
assertThrows(FlowFileHandlingException.class, targetSession::commitAsync, "Was able to commit with unaccounted FlowFile that immigrated");
assertDoesNotThrow(() -> session.commitAsync(), "Was not able to commit session that migrated its FlowFiles");
}
}
@Nested
class RegardingTransfer {
@Test
void canTransferSingleFlowFileToKnownRelationship() {
MockFlowFile flowFile = session.create();
assertDoesNotThrow(() -> session.transfer(flowFile, TestProcessor.REL_KNOWN));
try {
session.commitAsync(); session.commitAsync();
fail("Was able to commit session without closing InputStream"); session.assertAllFlowFilesTransferred(TestProcessor.REL_KNOWN, 1);
} catch (final FlowFileHandlingException | IllegalStateException e) {
System.out.println(e.toString());
}
} }
@Test @Test
public void testTransferUnknownRelationship() { void canTransferMultipleFlowFilesToKnownRelationship() {
final Processor processor = new PoorlyBehavedProcessor(); Collection<FlowFile> flowFiles = Set.of(session.create(), session.create(), session.create());
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor));
FlowFile ff1 = session.createFlowFile("hello, world".getBytes());
final Relationship fakeRel = new Relationship.Builder().name("FAKE").build();
try {
session.transfer(ff1, fakeRel);
fail("Should have thrown IllegalArgumentException");
} catch (final IllegalArgumentException ie) {
} assertDoesNotThrow(() -> session.transfer(flowFiles, TestProcessor.REL_KNOWN));
try {
session.transfer(Collections.singleton(ff1), fakeRel);
fail("Should have thrown IllegalArgumentException");
} catch (final IllegalArgumentException ie) {
}
session.commitAsync();
session.assertAllFlowFilesTransferred(TestProcessor.REL_KNOWN, flowFiles.size());
} }
@Test @Test
public void testRejectTransferNewlyCreatedFileToSelf() { void canTransferSingleFlowFileToSelfRelationship() {
final Processor processor = new PoorlyBehavedProcessor(); enqueueFlowFile();
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); MockFlowFile flowFile = session.get();
final FlowFile ff1 = session.createFlowFile("hello, world".getBytes());
// this should throw an exception because we shouldn't allow a newly created flowfile to get routed back to self assertDoesNotThrow(() -> session.transfer(flowFile));
assertThrows(IllegalArgumentException.class, () -> session.transfer(ff1));
session.commitAsync();
session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
assertObjectCountInQueue(session, 1);
} }
@Test @Test
public void testKeepPenalizedStatusAfterPuttingAttribute(){ void canTransferMultipleFlowFilesToSelfRelationship() {
final Processor processor = new PoorlyBehavedProcessor(); enqueueFlowFile();
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); enqueueFlowFile();
FlowFile ff1 = session.createFlowFile("hello, world".getBytes()); enqueueFlowFile();
ff1 = session.penalize(ff1); Collection<FlowFile> flowFiles = session.get(3);
assertTrue(ff1.isPenalized());
ff1 = session.putAttribute(ff1, "hello", "world"); assertDoesNotThrow(() -> session.transfer(flowFiles));
// adding attribute to flow file should not override the original penalized status
assertTrue(ff1.isPenalized()); session.commitAsync();
session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
assertObjectCountInQueue(session, flowFiles.size());
} }
@Test @Test
public void testUnpenalizeFlowFile() { void cannotTransferToUnknownRelationship() {
final Processor processor = new PoorlyBehavedProcessor(); MockFlowFile flowFile = session.create();
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); Collection<FlowFile> flowFiles = Set.of(session.create(), session.create(), session.create());
FlowFile ff1 = session.createFlowFile("hello, world".getBytes()); final Relationship unknownRelationship = new Relationship.Builder().name("unknown").build();
ff1 = session.penalize(ff1);
assertTrue(ff1.isPenalized()); assertThrows(IllegalArgumentException.class, () -> session.transfer(flowFile, unknownRelationship), "Was able to transfer to unknown relationship");
ff1 = session.unpenalize(ff1); assertThrows(IllegalArgumentException.class, () -> session.transfer(flowFiles, unknownRelationship), "Was able to transfer to unknown relationship");
assertFalse(ff1.isPenalized());
} }
@Test @Test
public void testRollbackWithCreatedFlowFile() { void cannotTransferNewlyCreatedFlowFilesToSelfRelationship() {
final Processor processor = new PoorlyBehavedProcessor(); MockFlowFile flowFile = session.create();
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); Collection<FlowFile> flowFiles = Set.of(session.create(), session.create(), session.create());
final FlowFile ff1 = session.createFlowFile("hello, world".getBytes());
session.transfer(ff1, PoorlyBehavedProcessor.REL_FAILURE); assertThrows(IllegalArgumentException.class, () -> session.transfer(flowFile), "Was able to transfer newly created FlowFile to self relationship");
assertThrows(IllegalArgumentException.class, () -> session.transfer(flowFiles), "Was able to transfer newly created FlowFiles to self relationship");
}
@Test
void cannotTransferClonedFlowFilesToSelfRelationship() {
MockFlowFile flowFile = session.create();
MockFlowFile clonedFlowFile = session.clone(flowFile);
Collection<FlowFile> clonedFlowFiles = Set.of(session.clone(flowFile), session.clone(flowFile));
assertThrows(IllegalArgumentException.class, () -> session.transfer(clonedFlowFile), "Was able to transfer cloned FlowFile to self relationship");
assertThrows(IllegalArgumentException.class, () -> session.transfer(clonedFlowFiles), "Was able to transfer cloned FlowFiles to self relationship");
}
}
@Nested
class RegardingPenalizedState {
@Test
void keepsPenalizedStatusAfterAttributeWrite() {
MockFlowFile flowFile = session.create();
flowFile = session.penalize(flowFile);
flowFile = session.putAttribute(flowFile, "Foo", "Bar");
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
assertTrue(() -> getSingleFlowFileInRelationship().isPenalized(), "FlowFile was not penalized");
}
@Test
void keepsPenalizedStatusAfterContentWrite() {
MockFlowFile flowFile = session.create();
flowFile = session.penalize(flowFile);
flowFile = session.write(flowFile, (outputStream) -> outputStream.write("test content".getBytes()));
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
assertTrue(() -> getSingleFlowFileInRelationship().isPenalized(), "FlowFile was not penalized");
}
@Test
void penalizedStatusCanBeReset() {
MockFlowFile flowFile = session.create();
flowFile = session.penalize(flowFile);
flowFile = session.unpenalize(flowFile);
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
assertFalse(() -> getSingleFlowFileInRelationship().isPenalized(), "FlowFile was penalized");
}
}
@Nested
class RegardingAttributes {
private final String UUID_ATTRIBUTE_NAME = CoreAttributes.UUID.key();
@Test
void canWriteAttribute() {
MockFlowFile flowFile = session.create();
flowFile = session.putAttribute(flowFile, "Hello", "world");
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
getSingleFlowFileInRelationship().assertAttributeEquals("Hello", "world");
}
@Test
void canWriteAttributes() {
MockFlowFile flowFile = session.create();
flowFile = session.putAllAttributes(flowFile, Map.of("Hello", "world", "Hola", "mundo"));
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
MockFlowFile resultFlowFile = getSingleFlowFileInRelationship();
resultFlowFile.assertAttributeEquals("Hello", "world");
resultFlowFile.assertAttributeEquals("Hola", "mundo");
}
@Test
void writtenAttributesAreNotAffectedByContentWrite() {
MockFlowFile flowFile = session.create();
flowFile = session.putAttribute(flowFile, "Hello", "world");
flowFile = session.write(flowFile, (outputStream) -> outputStream.write("test content".getBytes()));
flowFile = session.putAttribute(flowFile, "Hola", "mundo");
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
MockFlowFile resultFlowFile = getSingleFlowFileInRelationship();
resultFlowFile.assertAttributeEquals("Hello", "world");
resultFlowFile.assertAttributeEquals("Hola", "mundo");
}
@Test
void cannotModifyUUID() {
MockFlowFile flowFile = session.create();
String expectedUuid = flowFile.getAttribute(UUID_ATTRIBUTE_NAME);
assertThrows(AssertionError.class, () -> session.putAttribute(flowFile, UUID_ATTRIBUTE_NAME, "put single"));
session.putAllAttributes(flowFile, Map.of(UUID_ATTRIBUTE_NAME, "put multiple", "foo", "bar"));
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
assertEquals(expectedUuid, getSingleFlowFileInRelationship().getAttribute(UUID_ATTRIBUTE_NAME));
}
@Test
void cannotRemoveUUID() {
MockFlowFile flowFile = session.create();
String expectedUuid = flowFile.getAttribute(UUID_ATTRIBUTE_NAME);
flowFile = session.removeAttribute(flowFile, UUID_ATTRIBUTE_NAME);
flowFile = session.removeAllAttributes(flowFile, Set.of(UUID_ATTRIBUTE_NAME));
flowFile = session.removeAllAttributes(flowFile, Pattern.compile(Pattern.quote(UUID_ATTRIBUTE_NAME)));
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.commitAsync();
assertEquals(expectedUuid, getSingleFlowFileInRelationship().getAttribute(UUID_ATTRIBUTE_NAME));
}
}
@Nested
class RegardingRollbacks {
@Test
void flowFilesArePutBackToQueueOnRollback() {
enqueueFlowFile();
MockFlowFile flowFile = session.get();
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.rollback(); session.rollback();
session.assertQueueEmpty();
assertObjectCountInQueue(session, 1);
session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
} }
@Test @Test
public void testRollbackWithClonedFlowFile() { void attributeChangesAreResetOnRollback() {
final Processor processor = new PoorlyBehavedProcessor(); enqueueFlowFile();
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); MockFlowFile flowFile = session.get();
final FlowFile ff1 = session.createFlowFile("hello, world".getBytes()); session.putAttribute(flowFile, "attribute", "changed");
session.clone(ff1);
session.transfer(ff1, PoorlyBehavedProcessor.REL_FAILURE);
session.rollback(); session.rollback();
assertObjectCountInQueue(session, 1);
session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
assertObjectCountInQueue(session, 1);
MockFlowFile resultFlowFile = sharedState.getFlowFileQueue().poll();
resultFlowFile.assertAttributeNotExists("attribute");
}
@Test
void contentChangesAreResetOnRollback() {
enqueueFlowFile();
MockFlowFile flowFile = session.get();
String expectedContent = flowFile.getContent();
session.write(flowFile, (outputStream) -> outputStream.write("changed content".getBytes()));
session.rollback();
assertObjectCountInQueue(session, 1);
session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
assertObjectCountInQueue(session, 1);
MockFlowFile resultFlowFile = sharedState.getFlowFileQueue().poll();
resultFlowFile.assertContentEquals(expectedContent);
}
@Disabled("The current implementation of MockProcessSession does not express this behavior defined in the interface")
@Test
void stateChangesAreResetOnRollback() throws IOException {
sessionOfStatefulProcessor.setState(Map.of("attribute", "local"), Scope.LOCAL);
sessionOfStatefulProcessor.setState(Map.of("attribute", "cluster"), Scope.CLUSTER);
sessionOfStatefulProcessor.rollback();
stateManagerOfStatefulProcessor.assertStateNotSet();
stateManagerOfStatefulProcessor.assertStateEquals(Map.of(), Scope.LOCAL);
stateManagerOfStatefulProcessor.assertStateEquals(Map.of(), Scope.CLUSTER);
}
@Test
void newlyCreatedFlowFilesAreRemovedOnRollback() {
MockFlowFile flowFile = session.create();
session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.rollback();
session.assertQueueEmpty(); session.assertQueueEmpty();
session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
} }
@Test @Test
public void testRollbackWithMigratedFlowFile() { void clonedFlowFilesAreRemovedOnRollback() {
final Processor processor = new PoorlyBehavedProcessor(); enqueueFlowFile();
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); MockFlowFile flowFile = session.get();
final MockProcessSession newSession = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); MockFlowFile clonedFlowFile = session.clone(flowFile);
final FlowFile ff1 = session.createFlowFile("hello, world".getBytes()); session.transfer(flowFile, TestProcessor.REL_KNOWN);
session.migrate(newSession); session.transfer(clonedFlowFile, TestProcessor.REL_KNOWN);
newSession.transfer(ff1, PoorlyBehavedProcessor.REL_FAILURE);
newSession.rollback(); session.rollback();
assertObjectCountInQueue(session, 1);
session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
}
@Test
void migrateFlowFileIsPutToQueueOfNewOwnerOnRollback() {
final MockProcessSession targetSession = createMockProcessSession();
enqueueFlowFile();
MockFlowFile flowFile = session.get();
session.migrate(targetSession);
targetSession.transfer(flowFile, TestProcessor.REL_KNOWN);
targetSession.rollback();
session.assertQueueEmpty(); session.assertQueueEmpty();
newSession.assertQueueEmpty(); session.assertTransferCount(TestProcessor.REL_KNOWN, 0);
assertObjectCountInQueue(targetSession, 1);
targetSession.assertTransferCount(TestProcessor.REL_KNOWN, 0);
}
}
@Nested
class RegardingState {
@Test
void cannotClearLocalStateUnlessDeclaredStateful() {
assertThrows(AssertionError.class, () -> {
session.clearState(Scope.LOCAL);
}, "Was able to clear local state without declaring the being stateful");
stateManager.assertStateNotSet();
} }
@Test @Test
public void testAttributePreservedAfterWrite() throws IOException { void cannotClearClusterStateUnlessDeclaredStateful() {
final Processor processor = new PoorlyBehavedProcessor(); assertThrows(AssertionError.class, () -> {
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor)); session.clearState(Scope.CLUSTER);
FlowFile ff1 = session.createFlowFile("hello, world".getBytes()); }, "Was able to clear cluster state without declaring the being stateful");
session.putAttribute(ff1, "key1", "val1"); stateManager.assertStateNotSet();
session.write(ff1).close();
session.transfer(ff1, PoorlyBehavedProcessor.REL_FAILURE);
session.commitAsync();
List<MockFlowFile> output = session.getFlowFilesForRelationship(PoorlyBehavedProcessor.REL_FAILURE);
assertEquals(1, output.size());
output.get(0).assertAttributeEquals("key1", "val1");
} }
@Test @Test
void testAttributeUUIDNotRemovable() { void canClearLocalStateWhenDeclaredStateful() throws IOException {
final Processor processor = new PoorlyBehavedProcessor(); stateManagerOfStatefulProcessor.setState(Map.of("existing", "value"), Scope.LOCAL);
final MockProcessSession session = new MockProcessSession(new SharedSessionState(processor, new AtomicLong(0L)), processor, new MockStateManager(processor));
FlowFile ff1 = session.createFlowFile("removeAttribute(attrName)".getBytes());
FlowFile ff2 = session.createFlowFile("removeAllAttributes(attrNames)".getBytes());
FlowFile ff3 = session.createFlowFile("removeAllAttributes(keyPattern)".getBytes());
String attrName = CoreAttributes.UUID.key(); sessionOfStatefulProcessor.clearState(Scope.LOCAL);
session.removeAttribute(ff1, attrName);
session.removeAllAttributes(ff2, Set.of(attrName));
session.removeAllAttributes(ff3, Pattern.compile(Pattern.quote(attrName)));
session.transfer(List.of(ff1, ff2, ff3), PoorlyBehavedProcessor.REL_FAILURE); sessionOfStatefulProcessor.commitAsync();
session.commitAsync(); stateManagerOfStatefulProcessor.assertStateSet(Scope.LOCAL);
List<MockFlowFile> output = session.getFlowFilesForRelationship(PoorlyBehavedProcessor.REL_FAILURE); stateManagerOfStatefulProcessor.assertStateEquals(Map.of(), Scope.LOCAL);
assertEquals(3, output.size());
output.get(0).assertAttributeEquals(attrName, ff1.getAttribute(attrName));
output.get(1).assertAttributeEquals(attrName, ff2.getAttribute(attrName));
output.get(2).assertAttributeEquals(attrName, ff3.getAttribute(attrName));
} }
protected static class PoorlyBehavedProcessor extends AbstractProcessor { @Test
void canClearClusterStateWhenDeclaredStateful() throws IOException {
stateManagerOfStatefulProcessor.setState(Map.of("existing", "value"), Scope.LOCAL);
private static final Relationship REL_FAILURE = new Relationship.Builder() sessionOfStatefulProcessor.clearState(Scope.CLUSTER);
.name("failure")
.build();
private final Set<Relationship> relationships = Collections.singleton(REL_FAILURE); sessionOfStatefulProcessor.commitAsync();
stateManagerOfStatefulProcessor.assertStateSet(Scope.CLUSTER);
stateManagerOfStatefulProcessor.assertStateEquals(Map.of(), Scope.CLUSTER);
}
@Test
void cannotGetLocalStateUnlessDeclaredStateful() {
assertThrows(AssertionError.class, () -> {
session.getState(Scope.LOCAL);
}, "Was able to get local state without declaring the being stateful");
stateManager.assertStateNotSet();
}
@Test
void cannotGetClusterStateUnlessDeclaredStateful() {
assertThrows(AssertionError.class, () -> {
session.getState(Scope.CLUSTER);
}, "Was able to get cluster state without declaring the being stateful");
stateManager.assertStateNotSet();
}
@Test
void canGetLocalStateWhenDeclaredStateful() throws IOException {
Map<String, String> expectedState = Map.of("key", "value");
stateManagerOfStatefulProcessor.setState(expectedState, Scope.LOCAL);
StateMap result = sessionOfStatefulProcessor.getState(Scope.LOCAL);
assertEquals(expectedState, result.toMap());
}
@Test
void canGetClusterStateWhenDeclaredStateful() throws IOException {
Map<String, String> expectedState = Map.of("key", "value");
stateManagerOfStatefulProcessor.setState(expectedState, Scope.CLUSTER);
StateMap result = sessionOfStatefulProcessor.getState(Scope.CLUSTER);
assertEquals(expectedState, result.toMap());
}
@Test
void cannotSetLocalStateUnlessDeclaredStateful() {
assertThrows(AssertionError.class, () -> {
session.setState(Map.of("key", "value"), Scope.LOCAL);
}, "Was able to set local state without declaring the being stateful");
stateManager.assertStateNotSet();
}
@Test
void cannotSetClusterStateUnlessDeclaredStateful() {
assertThrows(AssertionError.class, () -> {
session.setState(Map.of("key", "value"), Scope.CLUSTER);
}, "Was able to set cluster state without declaring the being stateful");
stateManager.assertStateNotSet();
}
@Test
void canSetLocalStateWhenDeclaredStateful() throws IOException {
Map<String, String> expectedState = Map.of("key", "value");
sessionOfStatefulProcessor.setState(expectedState, Scope.LOCAL);
sessionOfStatefulProcessor.commitAsync();
assertEquals(expectedState, sessionOfStatefulProcessor.getState(Scope.LOCAL).toMap());
}
@Test
void canSetClusterStateWhenDeclaredStateful() throws IOException {
Map<String, String> expectedState = Map.of("key", "value");
sessionOfStatefulProcessor.setState(expectedState, Scope.CLUSTER);
sessionOfStatefulProcessor.commitAsync();
assertEquals(expectedState, sessionOfStatefulProcessor.getState(Scope.CLUSTER).toMap());
}
}
private MockProcessSession createMockProcessSession() {
return createMockProcessSession(new TestProcessor());
}
private static MockProcessSession createMockProcessSession(Processor processor) {
final SharedSessionState sharedState = new SharedSessionState(processor, new AtomicLong(0L));
return new MockProcessSession(sharedState, processor, new MockStateManager(processor));
}
private void enqueueFlowFile() {
MockFlowFile flowFile = new MockFlowFile(sharedState.nextFlowFileId());
flowFile.setData("test content".getBytes());
sharedState.getFlowFileQueue().offer(flowFile);
}
private MockFlowFile getSingleFlowFileInRelationship() {
List<MockFlowFile> flowFiles = session.getFlowFilesForRelationship(TestProcessor.REL_KNOWN);
assertEquals(1, flowFiles.size());
return flowFiles.getFirst();
}
private void assertObjectCountInQueue(MockProcessSession processSession, int expectedObjectCount) {
int actualObjectCount = processSession.getQueueSize().getObjectCount();
assertEquals(expectedObjectCount, actualObjectCount, "Queue had " + actualObjectCount + " FlowFile(s) but expected " + expectedObjectCount);
}
private static class TestProcessor extends AbstractProcessor {
private static final Relationship REL_KNOWN = new Relationship.Builder().name("known").build();
private static final Set<Relationship> relationships = Set.of(REL_KNOWN);
@Override @Override
public Set<Relationship> getRelationships() { public Set<Relationship> getRelationships() {
@ -236,10 +642,10 @@ public class TestMockProcessSession {
@Override @Override
public void onTrigger(final ProcessContext ctx, final ProcessSession session) throws ProcessException { public void onTrigger(final ProcessContext ctx, final ProcessSession session) throws ProcessException {
final FlowFile file = session.create(); fail("onTrigger of TestProcessor is not designed to be invoked");
session.penalize(file); }
session.transfer(file, REL_FAILURE);
} }
} @Stateful(description = "scopes for tests", scopes = {Scope.LOCAL, Scope.CLUSTER})
private static class StatefulTestProcessor extends TestProcessor {}
} }