mirror of https://github.com/apache/nifi.git
NIFI-12982 Extend test suite of MockProcessSession
This closes #8589 Signed-off-by: David Handermann <exceptionfactory@apache.org>
This commit is contained in:
parent
98c4061cfe
commit
6939ffc08d
|
@ -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 {}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue