|
| 1 | +import inspect |
1 | 2 | import json |
2 | 3 | import logging |
3 | 4 | from io import StringIO |
4 | | -from unittest.mock import MagicMock, patch |
| 5 | +from unittest.mock import MagicMock, Mock, call, patch |
5 | 6 |
|
6 | 7 | import pytest |
| 8 | +from requests.exceptions import HTTPError |
7 | 9 |
|
8 | 10 | import fourinsight.api as fapi |
9 | 11 | from fourinsight.api import authenticate |
@@ -523,6 +525,195 @@ def test_request_logger(self, mock_request, mock_fetch, mock_refresh): |
523 | 525 | log_out = stream.read() |
524 | 526 | assert log_out.startswith("request initiated") |
525 | 527 |
|
| 528 | + def test_get_pages(self, mock_fetch, mock_refresh): |
| 529 | + |
| 530 | + JSON_DATA = [ |
| 531 | + { |
| 532 | + "value": {"a": 0, "b": "foo", "c": "bar"}, |
| 533 | + "@odata.nextLink": "first/nextlink", |
| 534 | + }, |
| 535 | + { |
| 536 | + "value": [ |
| 537 | + {"a": 123, "b": "baz", "c": "bar"}, |
| 538 | + {"a": 456, "b": "foo", "c": "bar"}, |
| 539 | + ], |
| 540 | + "@odata.nextLink": "second/nextlink", |
| 541 | + }, |
| 542 | + { |
| 543 | + "value": None, |
| 544 | + "@odata.nextLink": "third/nextlink", |
| 545 | + }, |
| 546 | + { |
| 547 | + "value": {"a": 0, "b": "foo", "c": "bar"}, |
| 548 | + }, |
| 549 | + ] |
| 550 | + |
| 551 | + mock_response = Mock() |
| 552 | + mock_response._n_calls = 0 |
| 553 | + mock_response.json.side_effect = lambda *args, **kwargs: JSON_DATA[ |
| 554 | + mock_response._n_calls - 1 |
| 555 | + ] |
| 556 | + |
| 557 | + def get_side_effect(*args, **kwargs): |
| 558 | + mock_response._n_calls += 1 |
| 559 | + return mock_response |
| 560 | + |
| 561 | + with patch.object( |
| 562 | + authenticate.ClientSession, "get", side_effect=get_side_effect |
| 563 | + ) as mock_get: |
| 564 | + session = authenticate.ClientSession("my_client_id", "my_client_secret") |
| 565 | + |
| 566 | + pages = session.get_pages("foo/bar/baz", baz="foobar") |
| 567 | + |
| 568 | + for i, page_i in enumerate(pages): |
| 569 | + assert page_i.json() == JSON_DATA[i] |
| 570 | + assert page_i is mock_response |
| 571 | + |
| 572 | + mock_get.assert_has_calls( |
| 573 | + [ |
| 574 | + call("foo/bar/baz", baz="foobar"), |
| 575 | + call("first/nextlink", baz="foobar"), |
| 576 | + call("second/nextlink", baz="foobar"), |
| 577 | + call("third/nextlink", baz="foobar"), |
| 578 | + ] |
| 579 | + ) |
| 580 | + |
| 581 | + def test_get_pages_last_nextlink_none(self, mock_fetch, mock_refresh): |
| 582 | + |
| 583 | + JSON_DATA = [ |
| 584 | + { |
| 585 | + "value": {"a": 0, "b": "foo", "c": "bar"}, |
| 586 | + "@odata.nextLink": "first/nextlink", |
| 587 | + }, |
| 588 | + { |
| 589 | + "value": [ |
| 590 | + {"a": 123, "b": "baz", "c": "bar"}, |
| 591 | + {"a": 456, "b": "foo", "c": "bar"}, |
| 592 | + ], |
| 593 | + "@odata.nextLink": "second/nextlink", |
| 594 | + }, |
| 595 | + { |
| 596 | + "value": None, |
| 597 | + "@odata.nextLink": "third/nextlink", |
| 598 | + }, |
| 599 | + { |
| 600 | + "value": {"a": 0, "b": "foo", "c": "bar"}, |
| 601 | + "@odata.nextLink": None, |
| 602 | + }, |
| 603 | + ] |
| 604 | + |
| 605 | + mock_response = Mock() |
| 606 | + mock_response._n_calls = 0 |
| 607 | + mock_response.json.side_effect = lambda *args, **kwargs: JSON_DATA[ |
| 608 | + mock_response._n_calls - 1 |
| 609 | + ] |
| 610 | + |
| 611 | + def get_side_effect(*args, **kwargs): |
| 612 | + mock_response._n_calls += 1 |
| 613 | + return mock_response |
| 614 | + |
| 615 | + with patch.object( |
| 616 | + authenticate.ClientSession, "get", side_effect=get_side_effect |
| 617 | + ) as mock_get: |
| 618 | + session = authenticate.ClientSession("my_client_id", "my_client_secret") |
| 619 | + |
| 620 | + pages = session.get_pages("foo/bar/baz", baz="foobar") |
| 621 | + |
| 622 | + for i, page_i in enumerate(pages): |
| 623 | + assert page_i.json() == JSON_DATA[i] |
| 624 | + assert page_i is mock_response |
| 625 | + |
| 626 | + mock_get.assert_has_calls( |
| 627 | + [ |
| 628 | + call("foo/bar/baz", baz="foobar"), |
| 629 | + call("first/nextlink", baz="foobar"), |
| 630 | + call("second/nextlink", baz="foobar"), |
| 631 | + call("third/nextlink", baz="foobar"), |
| 632 | + ] |
| 633 | + ) |
| 634 | + |
| 635 | + def test_get_pages_raises_json_list(self, mock_fetch, mock_refresh): |
| 636 | + |
| 637 | + JSON_DATA = [ |
| 638 | + { |
| 639 | + "a": "foo", |
| 640 | + "b": "bar", |
| 641 | + "c": "baz", |
| 642 | + }, |
| 643 | + { |
| 644 | + "a": "foobar", |
| 645 | + "b": "baz", |
| 646 | + "c": "test", |
| 647 | + }, |
| 648 | + ] |
| 649 | + |
| 650 | + mock_response = Mock() |
| 651 | + mock_response.json.return_value = JSON_DATA |
| 652 | + |
| 653 | + with patch.object( |
| 654 | + authenticate.ClientSession, "get", return_value=mock_response |
| 655 | + ): |
| 656 | + session = authenticate.ClientSession("my_client_id", "my_client_secret") |
| 657 | + |
| 658 | + pages = session.get_pages("foo/bar/baz", baz="foobar") |
| 659 | + |
| 660 | + with pytest.raises(AttributeError): |
| 661 | + next(pages) |
| 662 | + |
| 663 | + def test_get_pages_raise_for_status(self, mock_fetch, mock_refresh): |
| 664 | + |
| 665 | + JSON_DATA = [ |
| 666 | + { |
| 667 | + "value": {"a": 0, "b": "foo", "c": "bar"}, |
| 668 | + "@odata.nextLink": "first/nextlink", |
| 669 | + }, |
| 670 | + { |
| 671 | + "value": [ |
| 672 | + {"a": 123, "b": "baz", "c": "bar"}, |
| 673 | + {"a": 456, "b": "foo", "c": "bar"}, |
| 674 | + ], |
| 675 | + "@odata.nextLink": "second/nextlink", |
| 676 | + }, |
| 677 | + { |
| 678 | + "value": None, |
| 679 | + "@odata.nextLink": "third/nextlink", |
| 680 | + }, |
| 681 | + { |
| 682 | + "value": {"a": 0, "b": "foo", "c": "bar"}, |
| 683 | + "@odata.nextLink": None, |
| 684 | + }, |
| 685 | + ] |
| 686 | + |
| 687 | + mock_response = Mock() |
| 688 | + mock_response._n_calls = 0 |
| 689 | + mock_response.json.side_effect = lambda *args, **kwargs: JSON_DATA[ |
| 690 | + mock_response._n_calls - 1 |
| 691 | + ] |
| 692 | + |
| 693 | + def raise_httperror(*args, **kwargs): |
| 694 | + raise HTTPError() |
| 695 | + |
| 696 | + mock_response.raise_for_status.side_effect = raise_httperror |
| 697 | + |
| 698 | + def get_side_effect(*args, **kwargs): |
| 699 | + mock_response._n_calls += 1 |
| 700 | + return mock_response |
| 701 | + |
| 702 | + with patch.object( |
| 703 | + authenticate.ClientSession, "get", side_effect=get_side_effect |
| 704 | + ) as mock_get: |
| 705 | + session = authenticate.ClientSession("my_client_id", "my_client_secret") |
| 706 | + |
| 707 | + pages = session.get_pages("foo/bar/baz", baz="foobar") |
| 708 | + |
| 709 | + for i, page_i in enumerate(pages): |
| 710 | + assert page_i.json() == JSON_DATA[i] |
| 711 | + assert page_i is mock_response |
| 712 | + |
| 713 | + mock_get.assert_called_once_with("foo/bar/baz", baz="foobar") |
| 714 | + |
| 715 | + mock_response.raise_for_status.assert_called_once() |
| 716 | + |
526 | 717 |
|
527 | 718 | if __name__ == "__main__": |
528 | 719 | pytest.main() |
0 commit comments