Commit af45f26
fix(auth): handle NaN clockDrift to fix token auto-refresh (#14618)
Fix automatic token refresh failing silently when clockDrift is NaN.
This addresses the issue where fetchAuthSession() does not auto-refresh
expired tokens for CUSTOM_WITHOUT_SRP auth flow.
## Root Cause
When clockDrift is NaN, the token expiration check always returns false:
```javascript
// isTokenExpired.ts (before)
return currentTime + clockDrift + tolerance > expiresAt;
// currentTime + NaN + tolerance = NaN
// NaN > expiresAt = false (always)
```
This causes expired tokens to never be detected as expired, preventing
automatic refresh.
## How clockDrift becomes NaN
In TokenStore.loadTokens(), clockDrift is parsed from storage:
```javascript
const clockDriftString = (await storage.getItem(key)) ?? '0';
const clockDrift = Number.parseInt(clockDriftString);
```
The nullish coalescing operator (??) only handles null/undefined, not
empty strings. If clockDrift is stored as "" (empty string):
| Stored Value | ?? '0' result | parseInt() result |
|--------------|---------------|-------------------|
| null | '0' | 0 ✓ |
| '123' | '123' | 123 ✓ |
| '' | '' (falsy!) | NaN ✗ |
This can happen with custom KeyValueStorage implementations or when
certain auth flows don't properly initialize the clockDrift value.
## Fix
Added three-layer defense against NaN clockDrift:
1. **TokenStore.ts**: Sanitize at load time
```javascript
const parsedClockDrift = Number.parseInt(clockDriftString);
const clockDrift = Number.isNaN(parsedClockDrift) ? 0 : parsedClockDrift;
```
2. **TokenOrchestrator.ts**: Use || instead of ?? for fallback
```javascript
clockDrift: tokens.clockDrift || 0 // NaN || 0 = 0
// vs
clockDrift: tokens.clockDrift ?? 0 // NaN ?? 0 = NaN
```
3. **isTokenExpired.ts**: Final safety net
```javascript
const safeClockDrift = Number.isNaN(clockDrift) ? 0 : clockDrift;
```
## Test Coverage
Added comprehensive tests for clockDrift edge cases:
- NaN clockDrift with expired tokens triggers refresh
- NaN clockDrift with valid tokens does not trigger refresh
- undefined clockDrift with expired tokens triggers refresh
- Positive/negative/zero clockDrift handled correctly
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>1 parent df5b60e commit af45f26
File tree
5 files changed
+171
-4
lines changed- packages
- auth
- __tests__/providers/cognito/tokenProvider
- src/providers/cognito/tokenProvider
- core
- __tests__/utils
- src/utils
5 files changed
+171
-4
lines changedLines changed: 110 additions & 0 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
450 | 450 | | |
451 | 451 | | |
452 | 452 | | |
| 453 | + | |
| 454 | + | |
| 455 | + | |
| 456 | + | |
| 457 | + | |
| 458 | + | |
| 459 | + | |
| 460 | + | |
| 461 | + | |
| 462 | + | |
| 463 | + | |
| 464 | + | |
| 465 | + | |
| 466 | + | |
| 467 | + | |
| 468 | + | |
| 469 | + | |
| 470 | + | |
| 471 | + | |
| 472 | + | |
| 473 | + | |
| 474 | + | |
| 475 | + | |
| 476 | + | |
| 477 | + | |
| 478 | + | |
| 479 | + | |
| 480 | + | |
| 481 | + | |
| 482 | + | |
| 483 | + | |
| 484 | + | |
| 485 | + | |
| 486 | + | |
| 487 | + | |
| 488 | + | |
| 489 | + | |
| 490 | + | |
| 491 | + | |
| 492 | + | |
| 493 | + | |
| 494 | + | |
| 495 | + | |
| 496 | + | |
| 497 | + | |
| 498 | + | |
| 499 | + | |
| 500 | + | |
| 501 | + | |
| 502 | + | |
| 503 | + | |
| 504 | + | |
| 505 | + | |
| 506 | + | |
| 507 | + | |
| 508 | + | |
| 509 | + | |
| 510 | + | |
| 511 | + | |
| 512 | + | |
| 513 | + | |
| 514 | + | |
| 515 | + | |
| 516 | + | |
| 517 | + | |
| 518 | + | |
| 519 | + | |
| 520 | + | |
| 521 | + | |
| 522 | + | |
| 523 | + | |
| 524 | + | |
| 525 | + | |
| 526 | + | |
| 527 | + | |
| 528 | + | |
| 529 | + | |
| 530 | + | |
| 531 | + | |
| 532 | + | |
| 533 | + | |
| 534 | + | |
| 535 | + | |
| 536 | + | |
| 537 | + | |
| 538 | + | |
| 539 | + | |
| 540 | + | |
| 541 | + | |
| 542 | + | |
| 543 | + | |
| 544 | + | |
| 545 | + | |
| 546 | + | |
| 547 | + | |
| 548 | + | |
| 549 | + | |
| 550 | + | |
| 551 | + | |
| 552 | + | |
| 553 | + | |
| 554 | + | |
| 555 | + | |
| 556 | + | |
| 557 | + | |
| 558 | + | |
| 559 | + | |
| 560 | + | |
| 561 | + | |
| 562 | + | |
453 | 563 | | |
454 | 564 | | |
Lines changed: 2 additions & 2 deletions
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
126 | 126 | | |
127 | 127 | | |
128 | 128 | | |
129 | | - | |
| 129 | + | |
130 | 130 | | |
131 | 131 | | |
132 | 132 | | |
133 | | - | |
| 133 | + | |
134 | 134 | | |
135 | 135 | | |
136 | 136 | | |
| |||
Lines changed: 2 additions & 1 deletion
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
70 | 70 | | |
71 | 71 | | |
72 | 72 | | |
73 | | - | |
| 73 | + | |
| 74 | + | |
74 | 75 | | |
75 | 76 | | |
76 | 77 | | |
| |||
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
53 | 53 | | |
54 | 54 | | |
55 | 55 | | |
| 56 | + | |
| 57 | + | |
| 58 | + | |
| 59 | + | |
| 60 | + | |
| 61 | + | |
| 62 | + | |
| 63 | + | |
| 64 | + | |
| 65 | + | |
| 66 | + | |
| 67 | + | |
| 68 | + | |
| 69 | + | |
| 70 | + | |
| 71 | + | |
| 72 | + | |
| 73 | + | |
| 74 | + | |
| 75 | + | |
| 76 | + | |
| 77 | + | |
| 78 | + | |
| 79 | + | |
| 80 | + | |
| 81 | + | |
| 82 | + | |
| 83 | + | |
| 84 | + | |
| 85 | + | |
| 86 | + | |
| 87 | + | |
| 88 | + | |
| 89 | + | |
| 90 | + | |
| 91 | + | |
| 92 | + | |
| 93 | + | |
| 94 | + | |
| 95 | + | |
| 96 | + | |
| 97 | + | |
| 98 | + | |
| 99 | + | |
| 100 | + | |
| 101 | + | |
| 102 | + | |
| 103 | + | |
| 104 | + | |
| 105 | + | |
| 106 | + | |
| 107 | + | |
| 108 | + | |
| 109 | + | |
56 | 110 | | |
| Original file line number | Diff line number | Diff line change | |
|---|---|---|---|
| |||
11 | 11 | | |
12 | 12 | | |
13 | 13 | | |
| 14 | + | |
| 15 | + | |
14 | 16 | | |
15 | | - | |
| 17 | + | |
16 | 18 | | |
0 commit comments