mirror of https://github.com/apache/nifi.git
NIFI-6724 - Check for SQS API call result in case of failures
This closes #3897. Signed-off-by: Peter Turcsanyi <turcsanyi@apache.org>
This commit is contained in:
parent
5bfc86737d
commit
6507b78948
|
@ -33,11 +33,13 @@ 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.ProcessContext;
|
import org.apache.nifi.processor.ProcessContext;
|
||||||
import org.apache.nifi.processor.ProcessSession;
|
import org.apache.nifi.processor.ProcessSession;
|
||||||
|
import org.apache.nifi.processor.exception.ProcessException;
|
||||||
import org.apache.nifi.processor.util.StandardValidators;
|
import org.apache.nifi.processor.util.StandardValidators;
|
||||||
|
|
||||||
import com.amazonaws.services.sqs.AmazonSQSClient;
|
import com.amazonaws.services.sqs.AmazonSQSClient;
|
||||||
import com.amazonaws.services.sqs.model.DeleteMessageBatchRequest;
|
import com.amazonaws.services.sqs.model.DeleteMessageBatchRequest;
|
||||||
import com.amazonaws.services.sqs.model.DeleteMessageBatchRequestEntry;
|
import com.amazonaws.services.sqs.model.DeleteMessageBatchRequestEntry;
|
||||||
|
import com.amazonaws.services.sqs.model.DeleteMessageBatchResult;
|
||||||
|
|
||||||
@SupportsBatching
|
@SupportsBatching
|
||||||
@SeeAlso({GetSQS.class, PutSQS.class})
|
@SeeAlso({GetSQS.class, PutSQS.class})
|
||||||
|
@ -66,42 +68,41 @@ public class DeleteSQS extends AbstractSQSProcessor {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onTrigger(final ProcessContext context, final ProcessSession session) {
|
public void onTrigger(final ProcessContext context, final ProcessSession session) {
|
||||||
List<FlowFile> flowFiles = session.get(1);
|
FlowFile flowFile = session.get();
|
||||||
if (flowFiles.isEmpty()) {
|
if (flowFile == null) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final FlowFile firstFlowFile = flowFiles.get(0);
|
final String queueUrl = context.getProperty(QUEUE_URL).evaluateAttributeExpressions(flowFile).getValue();
|
||||||
final String queueUrl = context.getProperty(QUEUE_URL).evaluateAttributeExpressions(firstFlowFile).getValue();
|
|
||||||
|
|
||||||
final AmazonSQSClient client = getClient();
|
final AmazonSQSClient client = getClient();
|
||||||
final DeleteMessageBatchRequest request = new DeleteMessageBatchRequest();
|
final DeleteMessageBatchRequest request = new DeleteMessageBatchRequest();
|
||||||
request.setQueueUrl(queueUrl);
|
request.setQueueUrl(queueUrl);
|
||||||
|
|
||||||
final List<DeleteMessageBatchRequestEntry> entries = new ArrayList<>(flowFiles.size());
|
final List<DeleteMessageBatchRequestEntry> entries = new ArrayList<>();
|
||||||
|
final DeleteMessageBatchRequestEntry entry = new DeleteMessageBatchRequestEntry();
|
||||||
for (final FlowFile flowFile : flowFiles) {
|
String receiptHandle = context.getProperty(RECEIPT_HANDLE).evaluateAttributeExpressions(flowFile).getValue();
|
||||||
final DeleteMessageBatchRequestEntry entry = new DeleteMessageBatchRequestEntry();
|
entry.setReceiptHandle(receiptHandle);
|
||||||
String receiptHandle = context.getProperty(RECEIPT_HANDLE).evaluateAttributeExpressions(flowFile).getValue();
|
String entryId = flowFile.getAttribute(CoreAttributes.UUID.key());
|
||||||
entry.setReceiptHandle(receiptHandle);
|
entry.setId(entryId);
|
||||||
String entryId = flowFile.getAttribute(CoreAttributes.UUID.key());
|
entries.add(entry);
|
||||||
entry.setId(entryId);
|
|
||||||
entries.add(entry);
|
|
||||||
}
|
|
||||||
|
|
||||||
request.setEntries(entries);
|
request.setEntries(entries);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
client.deleteMessageBatch(request);
|
DeleteMessageBatchResult response = client.deleteMessageBatch(request);
|
||||||
getLogger().info("Successfully deleted {} objects from SQS", new Object[]{flowFiles.size()});
|
|
||||||
session.transfer(flowFiles, REL_SUCCESS);
|
// check for errors
|
||||||
} catch (final Exception e) {
|
if (!response.getFailed().isEmpty()) {
|
||||||
getLogger().error("Failed to delete {} objects from SQS due to {}", new Object[]{flowFiles.size(), e});
|
throw new ProcessException(response.getFailed().get(0).toString());
|
||||||
final List<FlowFile> penalizedFlowFiles = new ArrayList<>();
|
|
||||||
for (final FlowFile flowFile : flowFiles) {
|
|
||||||
penalizedFlowFiles.add(session.penalize(flowFile));
|
|
||||||
}
|
}
|
||||||
session.transfer(penalizedFlowFiles, REL_FAILURE);
|
|
||||||
|
getLogger().info("Successfully deleted message from SQS for {}", new Object[] { flowFile });
|
||||||
|
session.transfer(flowFile, REL_SUCCESS);
|
||||||
|
} catch (final Exception e) {
|
||||||
|
getLogger().error("Failed to delete message from SQS due to {}", new Object[] { e });
|
||||||
|
flowFile = session.penalize(flowFile);
|
||||||
|
session.transfer(flowFile, REL_FAILURE);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -40,12 +40,14 @@ import org.apache.nifi.expression.ExpressionLanguageScope;
|
||||||
import org.apache.nifi.flowfile.FlowFile;
|
import org.apache.nifi.flowfile.FlowFile;
|
||||||
import org.apache.nifi.processor.ProcessContext;
|
import org.apache.nifi.processor.ProcessContext;
|
||||||
import org.apache.nifi.processor.ProcessSession;
|
import org.apache.nifi.processor.ProcessSession;
|
||||||
|
import org.apache.nifi.processor.exception.ProcessException;
|
||||||
import org.apache.nifi.processor.util.StandardValidators;
|
import org.apache.nifi.processor.util.StandardValidators;
|
||||||
|
|
||||||
import com.amazonaws.services.sqs.AmazonSQSClient;
|
import com.amazonaws.services.sqs.AmazonSQSClient;
|
||||||
import com.amazonaws.services.sqs.model.MessageAttributeValue;
|
import com.amazonaws.services.sqs.model.MessageAttributeValue;
|
||||||
import com.amazonaws.services.sqs.model.SendMessageBatchRequest;
|
import com.amazonaws.services.sqs.model.SendMessageBatchRequest;
|
||||||
import com.amazonaws.services.sqs.model.SendMessageBatchRequestEntry;
|
import com.amazonaws.services.sqs.model.SendMessageBatchRequestEntry;
|
||||||
|
import com.amazonaws.services.sqs.model.SendMessageBatchResult;
|
||||||
|
|
||||||
@SupportsBatching
|
@SupportsBatching
|
||||||
@SeeAlso({ GetSQS.class, DeleteSQS.class })
|
@SeeAlso({ GetSQS.class, DeleteSQS.class })
|
||||||
|
@ -135,7 +137,12 @@ public class PutSQS extends AbstractSQSProcessor {
|
||||||
request.setEntries(entries);
|
request.setEntries(entries);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
client.sendMessageBatch(request);
|
SendMessageBatchResult response = client.sendMessageBatch(request);
|
||||||
|
|
||||||
|
// check for errors
|
||||||
|
if (!response.getFailed().isEmpty()) {
|
||||||
|
throw new ProcessException(response.getFailed().get(0).toString());
|
||||||
|
}
|
||||||
} catch (final Exception e) {
|
} catch (final Exception e) {
|
||||||
getLogger().error("Failed to send messages to Amazon SQS due to {}; routing to failure", new Object[]{e});
|
getLogger().error("Failed to send messages to Amazon SQS due to {}; routing to failure", new Object[]{e});
|
||||||
flowFile = session.penalize(flowFile);
|
flowFile = session.penalize(flowFile);
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.nifi.processors.aws.sqs;
|
package org.apache.nifi.processors.aws.sqs;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
|
@ -25,6 +26,7 @@ import org.apache.nifi.util.TestRunners;
|
||||||
import com.amazonaws.services.sqs.AmazonSQSClient;
|
import com.amazonaws.services.sqs.AmazonSQSClient;
|
||||||
import com.amazonaws.services.sqs.model.AmazonSQSException;
|
import com.amazonaws.services.sqs.model.AmazonSQSException;
|
||||||
import com.amazonaws.services.sqs.model.DeleteMessageBatchRequest;
|
import com.amazonaws.services.sqs.model.DeleteMessageBatchRequest;
|
||||||
|
import com.amazonaws.services.sqs.model.DeleteMessageBatchResult;
|
||||||
|
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
@ -38,15 +40,17 @@ public class TestDeleteSQS {
|
||||||
|
|
||||||
private TestRunner runner = null;
|
private TestRunner runner = null;
|
||||||
private DeleteSQS mockDeleteSQS = null;
|
private DeleteSQS mockDeleteSQS = null;
|
||||||
private AmazonSQSClient actualSQSClient = null;
|
|
||||||
private AmazonSQSClient mockSQSClient = null;
|
private AmazonSQSClient mockSQSClient = null;
|
||||||
|
|
||||||
@Before
|
@Before
|
||||||
public void setUp() {
|
public void setUp() {
|
||||||
mockSQSClient = Mockito.mock(AmazonSQSClient.class);
|
mockSQSClient = Mockito.mock(AmazonSQSClient.class);
|
||||||
|
DeleteMessageBatchResult mockResponse = Mockito.mock(DeleteMessageBatchResult.class);
|
||||||
|
Mockito.when(mockSQSClient.deleteMessageBatch(Mockito.any())).thenReturn(mockResponse);
|
||||||
|
Mockito.when(mockResponse.getFailed()).thenReturn(new ArrayList<>());
|
||||||
mockDeleteSQS = new DeleteSQS() {
|
mockDeleteSQS = new DeleteSQS() {
|
||||||
|
@Override
|
||||||
protected AmazonSQSClient getClient() {
|
protected AmazonSQSClient getClient() {
|
||||||
actualSQSClient = client;
|
|
||||||
return mockSQSClient;
|
return mockSQSClient;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
<packaging>pom</packaging>
|
<packaging>pom</packaging>
|
||||||
|
|
||||||
<properties>
|
<properties>
|
||||||
<aws-java-sdk-version>1.11.599</aws-java-sdk-version>
|
<aws-java-sdk-version>1.11.677</aws-java-sdk-version>
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
<dependencyManagement>
|
<dependencyManagement>
|
||||||
|
|
Loading…
Reference in New Issue