Commit 93886d8
Fix stale snapshot detection to return 409 instead of 400 (#425)
## Summary
- Fix stale snapshot detection during concurrent modifications to return
HTTP 409 (Conflict) instead of HTTP 400 (Bad Request)
- Reclassify `ValidationException` with stale snapshot message to
`CommitFailedException` (409) to allow client retry
- Ensure other ValidationException instances are handled as HTTP 400 Bad
Request responses (e.g., attempting to delete a non-existent snapshot)
## Problem
When concurrent modifications occur during a transaction commit:
1. Client builds snapshots based on table version N (e.g.,
`lastSequenceNumber = 4`)
2. Client sends commit request with these snapshots in
`SNAPSHOTS_JSON_KEY`
3. Meanwhile, another process commits version N+1 (e.g.,
`lastSequenceNumber = 5`)
4. Server calls `doRefresh()` which updates `current()` to version N+1
5. **Bug:** The snapshots in `SNAPSHOTS_JSON_KEY` are now stale (their
sequence numbers are based on version N)
6. Iceberg's `TableMetadata.addSnapshot()` throws `ValidationException`
→ mapped to 400 Bad Request
7. Should return 409 Conflict so clients know to refresh and retry
## Solution
Let Iceberg's existing validation detect sequence number conflicts, then
catch the `ValidationException` and reclassify it as
`CommitFailedException` for the specific stale snapshot error pattern:
```java
} catch (ValidationException e) {
// Stale snapshot errors are retryable - client should refresh and retry
if (isStaleSnapshotError(e)) {
throw new CommitFailedException(e);
}
throw new BadRequestException(e, e.getMessage());
}
```
This approach is simpler than pre-checking and leverages Iceberg's
existing validation.
## Test Plan
- [x] Unit test `testStaleSnapshotErrorDetection()` verifies error
detection logic
- [x] All existing internalcatalog tests pass
- [ ] Integration testing in staging environment
---------
Co-authored-by: Claude Opus 4.5 <[email protected]>1 parent 1555bf5 commit 93886d8
File tree
4 files changed
+124
-2
lines changed- iceberg/openhouse/internalcatalog/src
- main/java/com/linkedin/openhouse/internal/catalog
- test
- java/com/linkedin/openhouse/internal/catalog
- resources
4 files changed
+124
-2
lines changedLines changed: 18 additions & 1 deletion
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
383 | 383 | | |
384 | 384 | | |
385 | 385 | | |
386 | | - | |
| 386 | + | |
| 387 | + | |
| 388 | + | |
| 389 | + | |
| 390 | + | |
| 391 | + | |
| 392 | + | |
387 | 393 | | |
388 | 394 | | |
389 | 395 | | |
| |||
556 | 562 | | |
557 | 563 | | |
558 | 564 | | |
| 565 | + | |
| 566 | + | |
| 567 | + | |
| 568 | + | |
| 569 | + | |
| 570 | + | |
| 571 | + | |
| 572 | + | |
| 573 | + | |
| 574 | + | |
| 575 | + | |
559 | 576 | | |
560 | 577 | | |
561 | 578 | | |
| |||
Lines changed: 5 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
16 | 16 | | |
17 | 17 | | |
18 | 18 | | |
| 19 | + | |
19 | 20 | | |
20 | 21 | | |
21 | 22 | | |
| |||
31 | 32 | | |
32 | 33 | | |
33 | 34 | | |
| 35 | + | |
| 36 | + | |
| 37 | + | |
| 38 | + | |
34 | 39 | | |
35 | 40 | | |
36 | 41 | | |
| |||
Lines changed: 98 additions & 1 deletion
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
52 | 52 | | |
53 | 53 | | |
54 | 54 | | |
| 55 | + | |
55 | 56 | | |
56 | 57 | | |
57 | 58 | | |
| |||
77 | 78 | | |
78 | 79 | | |
79 | 80 | | |
80 | | - | |
| 81 | + | |
81 | 82 | | |
82 | 83 | | |
83 | 84 | | |
| |||
1718 | 1719 | | |
1719 | 1720 | | |
1720 | 1721 | | |
| 1722 | + | |
| 1723 | + | |
| 1724 | + | |
| 1725 | + | |
| 1726 | + | |
| 1727 | + | |
| 1728 | + | |
| 1729 | + | |
| 1730 | + | |
| 1731 | + | |
| 1732 | + | |
| 1733 | + | |
| 1734 | + | |
| 1735 | + | |
| 1736 | + | |
| 1737 | + | |
| 1738 | + | |
| 1739 | + | |
| 1740 | + | |
| 1741 | + | |
| 1742 | + | |
| 1743 | + | |
| 1744 | + | |
| 1745 | + | |
| 1746 | + | |
| 1747 | + | |
| 1748 | + | |
| 1749 | + | |
| 1750 | + | |
| 1751 | + | |
| 1752 | + | |
| 1753 | + | |
| 1754 | + | |
| 1755 | + | |
| 1756 | + | |
| 1757 | + | |
| 1758 | + | |
| 1759 | + | |
| 1760 | + | |
| 1761 | + | |
| 1762 | + | |
| 1763 | + | |
| 1764 | + | |
| 1765 | + | |
| 1766 | + | |
| 1767 | + | |
| 1768 | + | |
| 1769 | + | |
| 1770 | + | |
| 1771 | + | |
| 1772 | + | |
| 1773 | + | |
| 1774 | + | |
| 1775 | + | |
| 1776 | + | |
| 1777 | + | |
| 1778 | + | |
| 1779 | + | |
| 1780 | + | |
| 1781 | + | |
| 1782 | + | |
| 1783 | + | |
| 1784 | + | |
| 1785 | + | |
| 1786 | + | |
| 1787 | + | |
| 1788 | + | |
| 1789 | + | |
| 1790 | + | |
| 1791 | + | |
| 1792 | + | |
| 1793 | + | |
| 1794 | + | |
| 1795 | + | |
| 1796 | + | |
| 1797 | + | |
| 1798 | + | |
| 1799 | + | |
| 1800 | + | |
| 1801 | + | |
| 1802 | + | |
| 1803 | + | |
| 1804 | + | |
| 1805 | + | |
| 1806 | + | |
| 1807 | + | |
| 1808 | + | |
| 1809 | + | |
| 1810 | + | |
| 1811 | + | |
| 1812 | + | |
| 1813 | + | |
| 1814 | + | |
| 1815 | + | |
| 1816 | + | |
| 1817 | + | |
1721 | 1818 | | |
Lines changed: 3 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
| 1 | + | |
| 2 | + | |
| 3 | + | |
0 commit comments