fix(configdb): InsertIfNotExistsAsync swallows duplicate-key races + add keyset tiebreaker test (#23)
Two concurrent sessions can both pass the IF NOT EXISTS check and then both attempt the INSERT against UX_AuditLog_EventId; the loser surfaced as SqlException 2601 (or 2627 for PK violations) and aborted the audit write. First-write-wins idempotency is the documented contract, so the race outcome is semantically a no-op — catch the two duplicate-key error numbers and log at Debug, let any other SqlException bubble. Tests: - InsertIfNotExistsAsync_ConcurrentDuplicateInserts_ProduceExactlyOneRow: 50 parallel inserters with the same EventId end with exactly one row and no escaped exceptions. - QueryAsync_Keyset_SameOccurredAtUtc_TiebreaksOnEventId: four rows sharing the same OccurredAtUtc page deterministically through the (OccurredAtUtc, EventId) keyset cursor — exercises the e.OccurredAtUtc == after && e.EventId.CompareTo(afterId) < 0 branch and verifies EF Core 10's Guid.CompareTo translation against SQL Server uniqueidentifier order (deferred Bundle D reviewer recommendation). AuditLogRepository now takes an optional ILogger<AuditLogRepository> (NullLogger default, mirrors InboundApiRepository); DI registration unchanged.
This commit is contained in:
@@ -217,6 +217,98 @@ public class AuditLogRepositoryTests : IClassFixture<MsSqlMigrationFixture>
|
||||
Assert.Equal(t0.AddMinutes(0), page3[0].OccurredAtUtc);
|
||||
}
|
||||
|
||||
[SkippableFact]
|
||||
public async Task InsertIfNotExistsAsync_ConcurrentDuplicateInserts_ProduceExactlyOneRow()
|
||||
{
|
||||
Skip.IfNot(_fixture.Available, _fixture.SkipReason);
|
||||
|
||||
var siteId = NewSiteId();
|
||||
|
||||
// Single event used by every parallel call — same EventId, same payload.
|
||||
// The repository's IF NOT EXISTS … INSERT pattern has a check-then-act
|
||||
// race window between sessions; under concurrent load SQL Server can
|
||||
// raise a unique-index violation (error 2601) on UX_AuditLog_EventId.
|
||||
// Bundle A's hardening swallows 2601/2627 so duplicates collapse silently.
|
||||
var evt = NewEvent(siteId, occurredAtUtc: new DateTime(2026, 5, 20, 12, 0, 0, DateTimeKind.Utc));
|
||||
|
||||
// 50 parallel inserters, each with its own DbContext (DbContext is not
|
||||
// thread-safe). Parallel.ForEachAsync aggregates exceptions, so a single
|
||||
// unhandled 2601 from the repository would fail this test loudly.
|
||||
await Parallel.ForEachAsync(
|
||||
Enumerable.Range(0, 50),
|
||||
new ParallelOptions { MaxDegreeOfParallelism = 50 },
|
||||
async (_, ct) =>
|
||||
{
|
||||
await using var context = CreateContext();
|
||||
var repo = new AuditLogRepository(context);
|
||||
await repo.InsertIfNotExistsAsync(evt, ct);
|
||||
});
|
||||
|
||||
await using var readContext = CreateContext();
|
||||
var count = await readContext.Set<AuditEvent>()
|
||||
.Where(e => e.SourceSiteId == siteId)
|
||||
.CountAsync();
|
||||
|
||||
Assert.Equal(1, count);
|
||||
}
|
||||
|
||||
[SkippableFact]
|
||||
public async Task QueryAsync_Keyset_SameOccurredAtUtc_TiebreaksOnEventId()
|
||||
{
|
||||
Skip.IfNot(_fixture.Available, _fixture.SkipReason);
|
||||
|
||||
var siteId = NewSiteId();
|
||||
await using var context = CreateContext();
|
||||
var repo = new AuditLogRepository(context);
|
||||
|
||||
// Four events all sharing the exact same OccurredAtUtc — the keyset
|
||||
// cursor must lean on the EventId tiebreaker (descending) to page
|
||||
// deterministically. Bundle D's reviewer flagged this as a deferred
|
||||
// verification because it depends on EF Core 10 translating
|
||||
// Guid.CompareTo against SQL Server's uniqueidentifier sort order.
|
||||
var occurredAt = new DateTime(2026, 5, 20, 13, 0, 0, DateTimeKind.Utc);
|
||||
|
||||
// Build four distinct Guids; we don't care about the literal ordering
|
||||
// produced by Guid.CompareTo — only that paging is deterministic and
|
||||
// covers every row exactly once.
|
||||
var events = Enumerable.Range(0, 4)
|
||||
.Select(_ => NewEvent(siteId, occurredAtUtc: occurredAt))
|
||||
.ToList();
|
||||
|
||||
foreach (var e in events)
|
||||
{
|
||||
await repo.InsertIfNotExistsAsync(e);
|
||||
}
|
||||
|
||||
var filter = new AuditLogQueryFilter(SourceSiteId: siteId);
|
||||
|
||||
var page1 = await repo.QueryAsync(filter, new AuditLogPaging(PageSize: 2));
|
||||
Assert.Equal(2, page1.Count);
|
||||
Assert.All(page1, r => Assert.Equal(occurredAt, r.OccurredAtUtc));
|
||||
|
||||
var cursor = page1[^1];
|
||||
var page2 = await repo.QueryAsync(
|
||||
filter,
|
||||
new AuditLogPaging(
|
||||
PageSize: 2,
|
||||
AfterOccurredAtUtc: cursor.OccurredAtUtc,
|
||||
AfterEventId: cursor.EventId));
|
||||
|
||||
Assert.Equal(2, page2.Count);
|
||||
Assert.All(page2, r => Assert.Equal(occurredAt, r.OccurredAtUtc));
|
||||
|
||||
var page1Ids = page1.Select(r => r.EventId).ToHashSet();
|
||||
var page2Ids = page2.Select(r => r.EventId).ToHashSet();
|
||||
|
||||
// No overlap between pages.
|
||||
Assert.Empty(page1Ids.Intersect(page2Ids));
|
||||
|
||||
// Every inserted EventId appears in exactly one of the two pages.
|
||||
var allIds = page1Ids.Union(page2Ids).ToHashSet();
|
||||
Assert.Equal(4, allIds.Count);
|
||||
Assert.True(events.Select(e => e.EventId).ToHashSet().SetEquals(allIds));
|
||||
}
|
||||
|
||||
[SkippableFact]
|
||||
public async Task SwitchOutPartitionAsync_ThrowsNotSupported_ForM1()
|
||||
{
|
||||
|
||||
Reference in New Issue
Block a user