|
6 | 6 | import static org.junit.jupiter.api.Assertions.assertNull; |
7 | 7 | import static org.junit.jupiter.api.Assertions.assertTrue; |
8 | 8 |
|
| 9 | +import java.time.OffsetDateTime; |
9 | 10 | import java.util.ArrayList; |
10 | 11 | import java.util.Collections; |
11 | 12 | import java.util.HashMap; |
@@ -418,12 +419,14 @@ public void testListTasksCombinedFilters() { |
418 | 419 | @Test |
419 | 420 | @Transactional |
420 | 421 | public void testListTasksPagination() { |
421 | | - // Create 5 tasks |
| 422 | + // Create 5 tasks with same timestamp to ensure ID-based pagination works |
| 423 | + // (With timestamp DESC sorting, same timestamps allow ID ASC tie-breaking) |
| 424 | + OffsetDateTime sameTimestamp = OffsetDateTime.now(java.time.ZoneOffset.UTC); |
422 | 425 | for (int i = 1; i <= 5; i++) { |
423 | 426 | Task task = new Task.Builder() |
424 | 427 | .id("task-page-" + i) |
425 | 428 | .contextId("context-pagination") |
426 | | - .status(new TaskStatus(TaskState.SUBMITTED)) |
| 429 | + .status(new TaskStatus(TaskState.SUBMITTED, null, sameTimestamp)) |
427 | 430 | .build(); |
428 | 431 | taskStore.save(task); |
429 | 432 | } |
@@ -465,6 +468,122 @@ public void testListTasksPagination() { |
465 | 468 | assertNull(result3.nextPageToken(), "Last page should have no next page token"); |
466 | 469 | } |
467 | 470 |
|
| 471 | + @Test |
| 472 | + @Transactional |
| 473 | + public void testListTasksPaginationWithDifferentTimestamps() { |
| 474 | + // Create tasks with different timestamps to verify keyset pagination |
| 475 | + // with composite sort (timestamp DESC, id ASC) |
| 476 | + OffsetDateTime now = OffsetDateTime.now(java.time.ZoneOffset.UTC); |
| 477 | + |
| 478 | + // Task 1: 10 minutes ago, ID="task-diff-a" |
| 479 | + Task task1 = new Task.Builder() |
| 480 | + .id("task-diff-a") |
| 481 | + .contextId("context-diff-timestamps") |
| 482 | + .status(new TaskStatus(TaskState.WORKING, null, now.minusMinutes(10))) |
| 483 | + .build(); |
| 484 | + taskStore.save(task1); |
| 485 | + |
| 486 | + // Task 2: 5 minutes ago, ID="task-diff-b" |
| 487 | + Task task2 = new Task.Builder() |
| 488 | + .id("task-diff-b") |
| 489 | + .contextId("context-diff-timestamps") |
| 490 | + .status(new TaskStatus(TaskState.WORKING, null, now.minusMinutes(5))) |
| 491 | + .build(); |
| 492 | + taskStore.save(task2); |
| 493 | + |
| 494 | + // Task 3: 5 minutes ago, ID="task-diff-c" (same timestamp as task2, tests ID tie-breaker) |
| 495 | + Task task3 = new Task.Builder() |
| 496 | + .id("task-diff-c") |
| 497 | + .contextId("context-diff-timestamps") |
| 498 | + .status(new TaskStatus(TaskState.WORKING, null, now.minusMinutes(5))) |
| 499 | + .build(); |
| 500 | + taskStore.save(task3); |
| 501 | + |
| 502 | + // Task 4: Now, ID="task-diff-d" |
| 503 | + Task task4 = new Task.Builder() |
| 504 | + .id("task-diff-d") |
| 505 | + .contextId("context-diff-timestamps") |
| 506 | + .status(new TaskStatus(TaskState.WORKING, null, now)) |
| 507 | + .build(); |
| 508 | + taskStore.save(task4); |
| 509 | + |
| 510 | + // Task 5: 1 minute ago, ID="task-diff-e" |
| 511 | + Task task5 = new Task.Builder() |
| 512 | + .id("task-diff-e") |
| 513 | + .contextId("context-diff-timestamps") |
| 514 | + .status(new TaskStatus(TaskState.WORKING, null, now.minusMinutes(1))) |
| 515 | + .build(); |
| 516 | + taskStore.save(task5); |
| 517 | + |
| 518 | + // Expected order (timestamp DESC, id ASC): |
| 519 | + // 1. task-diff-d (now) |
| 520 | + // 2. task-diff-e (1 min ago) |
| 521 | + // 3. task-diff-b (5 min ago, ID 'b') |
| 522 | + // 4. task-diff-c (5 min ago, ID 'c') |
| 523 | + // 5. task-diff-a (10 min ago) |
| 524 | + |
| 525 | + // Page 1: Get first 2 tasks |
| 526 | + ListTasksParams params1 = new ListTasksParams.Builder() |
| 527 | + .contextId("context-diff-timestamps") |
| 528 | + .pageSize(2) |
| 529 | + .build(); |
| 530 | + ListTasksResult result1 = taskStore.list(params1); |
| 531 | + |
| 532 | + assertEquals(5, result1.totalSize()); |
| 533 | + assertEquals(2, result1.pageSize()); |
| 534 | + assertNotNull(result1.nextPageToken(), "Should have next page token"); |
| 535 | + |
| 536 | + // Verify first page order |
| 537 | + assertEquals("task-diff-d", result1.tasks().get(0).getId(), "First task should be most recent"); |
| 538 | + assertEquals("task-diff-e", result1.tasks().get(1).getId(), "Second task should be 1 min ago"); |
| 539 | + |
| 540 | + // Verify pageToken format: "timestamp_millis:taskId" |
| 541 | + assertTrue(result1.nextPageToken().contains(":"), "PageToken should have format timestamp:id"); |
| 542 | + String[] tokenParts = result1.nextPageToken().split(":", 2); |
| 543 | + assertEquals(2, tokenParts.length, "PageToken should have exactly 2 parts"); |
| 544 | + assertEquals("task-diff-e", tokenParts[1], "PageToken should contain last task ID"); |
| 545 | + |
| 546 | + // Page 2: Get next 2 tasks |
| 547 | + ListTasksParams params2 = new ListTasksParams.Builder() |
| 548 | + .contextId("context-diff-timestamps") |
| 549 | + .pageSize(2) |
| 550 | + .pageToken(result1.nextPageToken()) |
| 551 | + .build(); |
| 552 | + ListTasksResult result2 = taskStore.list(params2); |
| 553 | + |
| 554 | + assertEquals(5, result2.totalSize()); |
| 555 | + assertEquals(2, result2.pageSize()); |
| 556 | + assertNotNull(result2.nextPageToken(), "Should have next page token"); |
| 557 | + |
| 558 | + // Verify second page order (tasks with same timestamp, sorted by ID) |
| 559 | + assertEquals("task-diff-b", result2.tasks().get(0).getId(), "Third task should be 5 min ago, ID 'b'"); |
| 560 | + assertEquals("task-diff-c", result2.tasks().get(1).getId(), "Fourth task should be 5 min ago, ID 'c'"); |
| 561 | + |
| 562 | + // Page 3: Get last task |
| 563 | + ListTasksParams params3 = new ListTasksParams.Builder() |
| 564 | + .contextId("context-diff-timestamps") |
| 565 | + .pageSize(2) |
| 566 | + .pageToken(result2.nextPageToken()) |
| 567 | + .build(); |
| 568 | + ListTasksResult result3 = taskStore.list(params3); |
| 569 | + |
| 570 | + assertEquals(5, result3.totalSize()); |
| 571 | + assertEquals(1, result3.pageSize()); |
| 572 | + assertNull(result3.nextPageToken(), "Last page should have no next page token"); |
| 573 | + |
| 574 | + // Verify last task |
| 575 | + assertEquals("task-diff-a", result3.tasks().get(0).getId(), "Last task should be oldest"); |
| 576 | + |
| 577 | + // Verify no duplicates across all pages |
| 578 | + List<String> allTaskIds = new ArrayList<>(); |
| 579 | + allTaskIds.addAll(result1.tasks().stream().map(Task::getId).toList()); |
| 580 | + allTaskIds.addAll(result2.tasks().stream().map(Task::getId).toList()); |
| 581 | + allTaskIds.addAll(result3.tasks().stream().map(Task::getId).toList()); |
| 582 | + |
| 583 | + assertEquals(5, allTaskIds.size(), "Should have exactly 5 tasks across all pages"); |
| 584 | + assertEquals(5, allTaskIds.stream().distinct().count(), "Should have no duplicate tasks"); |
| 585 | + } |
| 586 | + |
468 | 587 | @Test |
469 | 588 | @Transactional |
470 | 589 | public void testListTasksHistoryLimiting() { |
@@ -573,34 +692,80 @@ public void testListTasksDefaultPageSize() { |
573 | 692 | assertNotNull(result.nextPageToken(), "Should have next page"); |
574 | 693 | } |
575 | 694 |
|
| 695 | + @Test |
| 696 | + @Transactional |
| 697 | + public void testListTasksInvalidPageTokenFormat() { |
| 698 | + // Create a task |
| 699 | + Task task = new Task.Builder() |
| 700 | + .id("task-invalid-token") |
| 701 | + .contextId("context-invalid-token") |
| 702 | + .status(new TaskStatus(TaskState.WORKING)) |
| 703 | + .build(); |
| 704 | + taskStore.save(task); |
| 705 | + |
| 706 | + // Test 1: Legacy ID-only pageToken should throw InvalidParamsError |
| 707 | + ListTasksParams params1 = new ListTasksParams.Builder() |
| 708 | + .contextId("context-invalid-token") |
| 709 | + .pageToken("task-invalid-token") // ID-only format (legacy) |
| 710 | + .build(); |
| 711 | + |
| 712 | + try { |
| 713 | + taskStore.list(params1); |
| 714 | + throw new AssertionError("Expected InvalidParamsError for legacy ID-only pageToken"); |
| 715 | + } catch (io.a2a.spec.InvalidParamsError e) { |
| 716 | + // Expected - legacy format not supported |
| 717 | + assertTrue(e.getMessage().contains("Invalid pageToken format"), |
| 718 | + "Error message should mention invalid format"); |
| 719 | + } |
| 720 | + |
| 721 | + // Test 2: Malformed timestamp in pageToken should throw InvalidParamsError |
| 722 | + ListTasksParams params2 = new ListTasksParams.Builder() |
| 723 | + .contextId("context-invalid-token") |
| 724 | + .pageToken("not-a-number:task-id") // Invalid timestamp |
| 725 | + .build(); |
| 726 | + |
| 727 | + try { |
| 728 | + taskStore.list(params2); |
| 729 | + throw new AssertionError("Expected InvalidParamsError for malformed timestamp"); |
| 730 | + } catch (io.a2a.spec.InvalidParamsError e) { |
| 731 | + // Expected - malformed timestamp |
| 732 | + assertTrue(e.getMessage().contains("timestamp must be numeric"), |
| 733 | + "Error message should mention numeric timestamp requirement"); |
| 734 | + } |
| 735 | + } |
| 736 | + |
| 737 | + |
576 | 738 | @Test |
577 | 739 | @Transactional |
578 | 740 | public void testListTasksOrderingById() { |
579 | | - // Create tasks with IDs that will sort in specific order |
| 741 | + // Create tasks with same timestamp to test ID-based tie-breaking |
| 742 | + // (spec requires sorting by timestamp DESC, then ID ASC) |
| 743 | + OffsetDateTime sameTimestamp = OffsetDateTime.now(java.time.ZoneOffset.UTC); |
| 744 | + |
580 | 745 | Task task1 = new Task.Builder() |
581 | 746 | .id("task-order-a") |
582 | 747 | .contextId("context-order") |
583 | | - .status(new TaskStatus(TaskState.SUBMITTED)) |
| 748 | + .status(new TaskStatus(TaskState.SUBMITTED, null, sameTimestamp)) |
584 | 749 | .build(); |
585 | 750 |
|
586 | 751 | Task task2 = new Task.Builder() |
587 | 752 | .id("task-order-b") |
588 | 753 | .contextId("context-order") |
589 | | - .status(new TaskStatus(TaskState.SUBMITTED)) |
| 754 | + .status(new TaskStatus(TaskState.SUBMITTED, null, sameTimestamp)) |
590 | 755 | .build(); |
591 | 756 |
|
592 | 757 | Task task3 = new Task.Builder() |
593 | 758 | .id("task-order-c") |
594 | 759 | .contextId("context-order") |
595 | | - .status(new TaskStatus(TaskState.SUBMITTED)) |
| 760 | + .status(new TaskStatus(TaskState.SUBMITTED, null, sameTimestamp)) |
596 | 761 | .build(); |
597 | 762 |
|
598 | 763 | // Save in reverse order |
599 | 764 | taskStore.save(task3); |
600 | 765 | taskStore.save(task1); |
601 | 766 | taskStore.save(task2); |
602 | 767 |
|
603 | | - // List should return in ID order |
| 768 | + // List should return sorted by timestamp DESC (all same), then by ID ASC |
604 | 769 | ListTasksParams params = new ListTasksParams.Builder() |
605 | 770 | .contextId("context-order") |
606 | 771 | .build(); |
|
0 commit comments