Skip to content

Commit 0d317c1

Browse files
committed
Merge branch 'main' into dev
2 parents 5db5888 + a1b0aed commit 0d317c1

File tree

7 files changed

+130
-6
lines changed

7 files changed

+130
-6
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
{
2+
"Projects": [
3+
{
4+
"Name": "AWS.Messaging",
5+
"Type": "Patch",
6+
"ChangelogMessages": [
7+
"Fix issue with fifo when a message is failed to process the later messages are not retried"
8+
]
9+
},
10+
{
11+
"Name": "AWS.Messaging.Lambda",
12+
"Type": "Patch",
13+
"ChangelogMessages": [
14+
"Update AWS.Messaging dependency"
15+
]
16+
}
17+
]
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
{
2+
"Projects": [
3+
{
4+
"Name": "AWS.Messaging",
5+
"Type": "Patch",
6+
"ChangelogMessages": [
7+
"Update User-Agent string"
8+
]
9+
},
10+
{
11+
"Name": "AWS.Messaging.Lambda",
12+
"Type": "Patch",
13+
"ChangelogMessages": [
14+
"Update project dependencies"
15+
]
16+
},
17+
{
18+
"Name": "AWS.Messaging.Telemetry.OpenTelemetry",
19+
"Type": "Patch",
20+
"ChangelogMessages": [
21+
"Update project dependencies"
22+
]
23+
}
24+
]
25+
}

.github/ISSUE_TEMPLATE/bug-report.yml

+8
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,14 @@ body:
1212
description: What is the problem? A clear and concise description of the bug.
1313
validations:
1414
required: true
15+
- type: checkboxes
16+
id: regression
17+
attributes:
18+
label: Regression Issue
19+
description: What is a regression? If it worked in a previous version but doesn't in the latest version, it's considered a regression. In this case, please provide specific version number in the report.
20+
options:
21+
- label: Select this option if this issue appears to be a regression.
22+
required: false
1523
- type: textarea
1624
id: expected
1725
attributes:

.github/workflows/BuildandTest.yml

+4
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,10 @@ jobs:
1515
uses: actions/setup-dotnet@4d6c8fcf3c8f7a60068d26b594648e99df24cee3 # pinning V4
1616
with:
1717
dotnet-version: 6.0.x
18+
- name: Setup .NET Core 8.0
19+
uses: actions/setup-dotnet@4d6c8fcf3c8f7a60068d26b594648e99df24cee3 # pinning V4
20+
with:
21+
dotnet-version: 8.0.x
1822
- name: Restore dependencies
1923
run: dotnet restore
2024
- name: Build
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
# Apply potential regression label on issues
2+
name: issue-regression-label
3+
on:
4+
issues:
5+
types: [opened, edited]
6+
jobs:
7+
add-regression-label:
8+
runs-on: ubuntu-latest
9+
permissions:
10+
issues: write
11+
steps:
12+
- name: Fetch template body
13+
id: check_regression
14+
uses: actions/github-script@v7
15+
env:
16+
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
17+
TEMPLATE_BODY: ${{ github.event.issue.body }}
18+
with:
19+
script: |
20+
const regressionPattern = /\[x\] Select this option if this issue appears to be a regression\./i;
21+
const template = `${process.env.TEMPLATE_BODY}`
22+
const match = regressionPattern.test(template);
23+
core.setOutput('is_regression', match);
24+
- name: Manage regression label
25+
env:
26+
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
27+
run: |
28+
if [ "${{ steps.check_regression.outputs.is_regression }}" == "true" ]; then
29+
gh issue edit ${{ github.event.issue.number }} --add-label "potential-regression" -R ${{ github.repository }}
30+
else
31+
gh issue edit ${{ github.event.issue.number }} --remove-label "potential-regression" -R ${{ github.repository }}
32+
fi

src/AWS.Messaging/Services/DefaultMessageManager.cs

+9-2
Original file line numberDiff line numberDiff line change
@@ -133,12 +133,19 @@ public async Task ProcessMessageGroupAsync(List<ConvertToEnvelopeResult> message
133133
var remaining = messageGroup.Count;
134134

135135
// Sequentially process each message within the group
136-
foreach (var message in messageGroup)
136+
for(var i = 0; i < messageGroup.Count; i++)
137137
{
138+
var message = messageGroup[i];
139+
138140
var isSuccessful = await InvokeHandler(message.Envelope, message.Mapping, token);
139141
if (!isSuccessful)
140142
{
141-
// If the handler invocation fails for any message, skip processing subsequent messages in the group.
143+
// If the handler invocation fails for any message, report failure for the rest in the group so they are reported back to Lambda
144+
// as not failure that should be retried. Otherwise Lambda will think the messages were successfully processed and delete the messages.
145+
for (var unprocessedIndex = i + 1; unprocessedIndex < messageGroup.Count; unprocessedIndex++)
146+
{
147+
await _sqsMessageCommunication.ReportMessageFailureAsync(messageGroup[unprocessedIndex].Envelope, token);
148+
}
142149
_logger.LogError("Handler invocation failed for a message belonging to message group '{GroupdId}' having message ID '{MessageID}'. Skipping processing of {Remaining} messages from the same group.", groupId, message.Envelope.Id, remaining);
143150
break;
144151
}

test/AWS.Messaging.UnitTests/LambdaTests.cs

+34-4
Original file line numberDiff line numberDiff line change
@@ -105,15 +105,44 @@ public async Task MessageHandlerReturnsFailedStatusBatchResponse()
105105
Id = "failed-message-1",
106106
ReturnFailedStatus = true
107107
};
108+
var secondMessage = new SimulatedMessage
109+
{
110+
Id = "second-message-1",
111+
ReturnFailedStatus = false
112+
};
108113

109-
var tuple = await ExecuteWithBatchResponse(new SimulatedMessage[] { failedMessage });
114+
var tuple = await ExecuteWithBatchResponse(new SimulatedMessage[] { failedMessage, secondMessage });
110115
_mockSqs!.Verify(
111116
expression: x => x.ChangeMessageVisibilityAsync(It.IsAny<ChangeMessageVisibilityRequest>(), It.IsAny<CancellationToken>()),
112117
times: Times.Never);
113118
Assert.Single(tuple.batchResponse.BatchItemFailures);
114119
Assert.Equal("1", tuple.batchResponse.BatchItemFailures[0].ItemIdentifier);
115120
}
116121

122+
[Fact]
123+
public async Task MessageHandlerReturnsFailedStatusBatchResponseFifo()
124+
{
125+
var failedMessage = new SimulatedMessage
126+
{
127+
Id = "failed-message-1",
128+
ReturnFailedStatus = true,
129+
MessageGroupId = "group1"
130+
};
131+
var secondMessage = new SimulatedMessage
132+
{
133+
Id = "second-message-1",
134+
ReturnFailedStatus = false,
135+
MessageGroupId = "group1"
136+
};
137+
138+
var tuple = await ExecuteWithBatchResponse(new SimulatedMessage[] { failedMessage, secondMessage }, isFifoQueue: true);
139+
_mockSqs!.Verify(
140+
expression: x => x.ChangeMessageVisibilityAsync(It.IsAny<ChangeMessageVisibilityRequest>(), It.IsAny<CancellationToken>()),
141+
times: Times.Never);
142+
Assert.Equal(2, tuple.batchResponse.BatchItemFailures.Count);
143+
Assert.Equal("1", tuple.batchResponse.BatchItemFailures[0].ItemIdentifier);
144+
}
145+
117146
[Fact]
118147
public async Task MessageHandlerResetsVisibilityWhenFailedStatusBatchResponse()
119148
{
@@ -304,13 +333,14 @@ private async Task<string> Execute(SimulatedMessage[] messages, int maxNumberOfC
304333
SimulatedMessage[] messages,
305334
int maxNumberOfConcurrentMessages = 1,
306335
bool deleteMessagesWhenCompleted = false,
307-
int? visibilityTimeoutForBatchFailures = default)
336+
int? visibilityTimeoutForBatchFailures = default,
337+
bool isFifoQueue = false)
308338
{
309339
var provider = CreateServiceProvider(
310340
maxNumberOfConcurrentMessages: maxNumberOfConcurrentMessages,
311341
deleteMessagesWhenCompleted: deleteMessagesWhenCompleted,
312-
visibilityTimeoutForBatchFailures: visibilityTimeoutForBatchFailures);
313-
var sqsEvent = await CreateLambdaEvent(provider, messages);
342+
visibilityTimeoutForBatchFailures: visibilityTimeoutForBatchFailures, isFifoQueue: isFifoQueue);
343+
var sqsEvent = await CreateLambdaEvent(provider, messages, isFifoQueue: isFifoQueue);
314344

315345
var logger = new TestLambdaLogger();
316346
var context = new TestLambdaContext()

0 commit comments

Comments
 (0)