@@ -183,6 +183,8 @@ contract TrimmingTest is Test {
183
183
assertEq (trimmedAmount.amount, amountRoundTrip.amount);
184
184
}
185
185
186
+ // FUZZ TESTS
187
+
186
188
// invariant: forall (TrimmedAmount a, TrimmedAmount b) a.saturatingAdd(b).amount <= type(uint64).max
187
189
function testFuzz_saturatingAddDoesNotOverflow (
188
190
TrimmedAmount memory a ,
@@ -207,4 +209,61 @@ contract TrimmingTest is Test {
207
209
// amount should never underflow
208
210
assertGe (c.amount, 0 );
209
211
}
212
+
213
+ // NOTE: above the TRIMMED_DECIMALS threshold will always get trimmed to TRIMMED_DECIMALS
214
+ // or trimmed to the number of decimals on the recipient chain.
215
+ // this tests for inputs with decimals > TRIMMED_DECIMALS
216
+ function testFuzz_SubOperatorZeroAboveThreshold (uint256 amt , uint8 decimals ) public {
217
+ uint8 decimals = uint8 (bound (decimals, 8 , 18 ));
218
+ uint256 maxAmt = (type (uint64 ).max) / (10 ** decimals);
219
+ vm.assume (amt < maxAmt);
220
+
221
+ uint256 amount = amt * (10 ** decimals);
222
+ uint256 amountOther = 0 ;
223
+ TrimmedAmount memory trimmedAmount = amount.trim (decimals, 8 );
224
+ TrimmedAmount memory trimmedAmountOther = amountOther.trim (decimals, 8 );
225
+
226
+ TrimmedAmount memory trimmedSub = trimmedAmount.sub (trimmedAmountOther);
227
+
228
+ TrimmedAmount memory expectedTrimmedSub = TrimmedAmount (uint64 (amt * (10 ** 8 )), 8 );
229
+ assert (expectedTrimmedSub.eq (trimmedSub));
230
+ }
231
+
232
+ function testFuzz_SubOperatorWillOverflow (
233
+ uint8 decimals ,
234
+ uint256 amtLeft ,
235
+ uint256 amtRight
236
+ ) public {
237
+ uint8 decimals = uint8 (bound (decimals, 8 , 18 ));
238
+ uint256 maxAmt = (type (uint64 ).max) / (10 ** decimals);
239
+ vm.assume (amtRight < maxAmt);
240
+ vm.assume (amtLeft < amtRight);
241
+
242
+ uint256 amountLeft = amtLeft * (10 ** decimals);
243
+ uint256 amountRight = amtRight * (10 ** decimals);
244
+ TrimmedAmount memory trimmedAmount = amountLeft.trim (decimals, 8 );
245
+ TrimmedAmount memory trimmedAmountOther = amountRight.trim (decimals, 8 );
246
+
247
+ vm.expectRevert ();
248
+ trimmedAmount.sub (trimmedAmountOther);
249
+ }
250
+
251
+ // NOTE: above the TRIMMED_DECIMALS threshold will always get trimmed to TRIMMED_DECIMALS
252
+ // or trimmed to the number of decimals on the recipient chain.
253
+ // this tests for inputs with decimals > TRIMMED_DECIMALS
254
+ function testFuzz_AddOperatorZeroAboveThreshold (uint256 amt , uint8 decimals ) public {
255
+ uint8 decimals = uint8 (bound (decimals, 8 , 18 ));
256
+ uint256 maxAmt = (type (uint64 ).max) / (10 ** decimals);
257
+ vm.assume (amt < maxAmt);
258
+
259
+ uint256 amount = amt * (10 ** decimals);
260
+ uint256 amountOther = 0 ;
261
+ TrimmedAmount memory trimmedAmount = amount.trim (decimals, 8 );
262
+ TrimmedAmount memory trimmedAmountOther = amountOther.trim (decimals, 8 );
263
+
264
+ TrimmedAmount memory trimmedSum = trimmedAmount.add (trimmedAmountOther);
265
+
266
+ TrimmedAmount memory expectedTrimmedSum = TrimmedAmount (uint64 (amt * (10 ** 8 )), 8 );
267
+ assert (expectedTrimmedSum.eq (trimmedSum));
268
+ }
210
269
}
0 commit comments