-
Notifications
You must be signed in to change notification settings - Fork 132
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Tests on the catalist match connector
- Loading branch information
1 parent
ca93835
commit ed2c9c5
Showing
2 changed files
with
198 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,45 @@ | ||
from unittest.mock import MagicMock | ||
from collections.abc import Generator | ||
import pytest | ||
import requests_mock | ||
import re | ||
|
||
|
||
@pytest.fixture(autouse=True) | ||
def mock_requests() -> Generator[MagicMock, None, None]: | ||
"""Replace requests in api_connector with a mock client""" | ||
with requests_mock.Mocker() as mocker: | ||
mocker.post(requests_mock.ANY, json={"test": True}) | ||
mocker.post( | ||
"https://auth.catalist.us/oauth/token", | ||
json={"access_token": "tokenexample", "expires_in": 99999, "test": True}, | ||
) | ||
mocker.get(requests_mock.ANY, json=[{"test": True}]) | ||
mocker.get( | ||
re.compile("/mapi/status/id/"), | ||
json={"process": {"processState": "Finished"}}, | ||
) | ||
|
||
yield mocker | ||
|
||
|
||
@pytest.fixture(autouse=True) | ||
def mock_sftp(mocker) -> Generator[MagicMock, None, None]: | ||
"""Replace sftp client with a mock client""" | ||
magic_mock = MagicMock() | ||
|
||
mocker.patch("parsons.catalist.catalist.SFTP", new=magic_mock) | ||
|
||
yield mocker | ||
|
||
|
||
@pytest.fixture(autouse=True) | ||
def mock_miscellaneous(mocker) -> Generator[MagicMock, None, None]: | ||
"""Replace miscellaneous utilities with mocks to simplify testing""" | ||
magic_mock = MagicMock() | ||
|
||
mocker.patch("parsons.catalist.catalist.ZipFile", new=magic_mock) | ||
mocker.patch("parsons.catalist.catalist.os", new=magic_mock) | ||
mocker.patch("parsons.catalist.catalist.Table", new=magic_mock) | ||
|
||
yield mocker |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,153 @@ | ||
from parsons import Table, CatalistMatch | ||
import time | ||
import pytest | ||
from unittest.mock import MagicMock | ||
|
||
TEST_CLIENT_ID = "some_client_id" | ||
TEST_CLIENT_SECRET = "some_client_secret" | ||
TEST_SFTP_USERNAME = "username" | ||
TEST_SFTP_PASSWORD = "password" | ||
|
||
|
||
def table_for_test(include_last_name: bool = True) -> Table: | ||
"""parsons Table for tests""" | ||
table = Table( | ||
[ | ||
{"first_name": "John", "last_name": "Doe"}, | ||
{"first_name": "Jane", "last_name": "Doe"}, | ||
] | ||
) | ||
if not include_last_name: | ||
table = table.cut("first_name") | ||
return table | ||
|
||
|
||
def match_client() -> CatalistMatch: | ||
result = CatalistMatch( | ||
client_id=TEST_CLIENT_ID, | ||
client_secret=TEST_CLIENT_SECRET, | ||
sftp_username=TEST_SFTP_USERNAME, | ||
sftp_password=TEST_SFTP_PASSWORD, | ||
) | ||
result._token_expired_at = time.time() + 99999 | ||
return result | ||
|
||
|
||
class TestCatalist: | ||
def test_fixtures_active(self) -> None: | ||
"""Test to ensure fixtures are active and relevant clients are mocked.""" | ||
match = match_client() | ||
assert isinstance(match.sftp, MagicMock) | ||
assert match.connection.request("url", "get").json()[0]["test"] | ||
|
||
def test_validate_table(self) -> None: | ||
"""Check that table validation method works as expected.""" | ||
match = match_client() | ||
table = table_for_test() | ||
match.validate_table(table) | ||
|
||
# first_name and last_name are required | ||
# We expect an exception raised if last_name is missing | ||
table_to_fail = table_for_test(include_last_name=False) | ||
with pytest.raises(ValueError): | ||
match.validate_table(table_to_fail) | ||
|
||
def test_load_table_to_sftp(self) -> None: | ||
"""Check that table load to SFTP executes as expected.""" | ||
match = match_client() | ||
source_table = table_for_test() | ||
response = match.load_table_to_sftp(source_table) | ||
|
||
assert response.startswith("file://") | ||
assert "myUploads" not in response | ||
assert response.endswith(".csv.gz") | ||
|
||
# We expect one call to the SFTP client to put the file | ||
assert len(match.sftp.mock_calls) == 1 | ||
mocked_call = match.sftp.mock_calls[0] | ||
called_method = str(mocked_call).split("(")[0].split(".")[1] | ||
assert called_method == "put_file" | ||
temp_local_file = mocked_call.args[0] | ||
remote_path = mocked_call.args[1] | ||
|
||
# Expect local temp file CSV is the same as the source table CSV | ||
table_to_load = Table.from_csv(temp_local_file) | ||
for row_index in range(table_to_load.num_rows): | ||
assert source_table[row_index] == table_to_load[row_index] | ||
|
||
# Expect the remote path is structured as expected | ||
assert remote_path.startswith("myUploads/") | ||
assert remote_path.endswith(".csv.gz") | ||
|
||
def test_upload(self, mock_requests) -> None: | ||
"""Mock use of upload() method, check API calls are structured as expected.""" | ||
match = match_client() | ||
source_table = table_for_test() | ||
|
||
# Execute upload | ||
match.upload(source_table) | ||
|
||
requested_endpoint = mock_requests._adapter.request_history[1].path | ||
requested_queries = mock_requests._adapter.request_history[1].qs | ||
requested_base_url = mock_requests._adapter.request_history[1]._url_parts.netloc | ||
|
||
assert requested_base_url == "api.catalist.us" | ||
assert set(requested_queries.keys()) == set(["token"]) | ||
assert requested_queries["token"] == ["tokenexample"] | ||
assert requested_endpoint.startswith( | ||
"/mapi/upload/template/48827/action/publish/url/" | ||
) | ||
|
||
def test_upload_with_options(self, mock_requests) -> None: | ||
"""Mock use of upload() method with options, check API calls.""" | ||
match = match_client() | ||
source_table = table_for_test() | ||
|
||
# Execute upload | ||
match.upload( | ||
source_table, | ||
copy_to_sandbox=True, | ||
static_values={"phone": 123456789}, | ||
) | ||
|
||
requested_queries = mock_requests._adapter.request_history[1].qs | ||
|
||
assert set(requested_queries.keys()) == set(["token", "copytosandbox", "phone"]) | ||
assert requested_queries["copytosandbox"] == ["true"] | ||
assert requested_queries["phone"] == ["123456789"] | ||
|
||
def test_status(self, mock_requests) -> None: | ||
"""Mock use of status() method, check API calls are structured as expected.""" | ||
match = match_client() | ||
|
||
# Check status | ||
match.status("12345") | ||
|
||
requested_endpoint = mock_requests._adapter.request_history[1].path | ||
requested_queries = mock_requests._adapter.request_history[1].qs | ||
requested_base_url = mock_requests._adapter.request_history[1]._url_parts.netloc | ||
|
||
assert requested_base_url == "api.catalist.us" | ||
assert set(requested_queries.keys()) == set(["token"]) | ||
assert requested_queries["token"] == ["tokenexample"] | ||
assert requested_endpoint == "/mapi/status/id/12345" | ||
|
||
def test_load_matches(self) -> None: | ||
"""Check that table download method from SFTP executes as expected.""" | ||
match = match_client() | ||
|
||
# Execute download | ||
match.sftp.list_directory = MagicMock(return_value=["example_12345"]) | ||
match.load_matches("12345") | ||
|
||
# We expect two calls to the SFTP client to list the directory and get the file | ||
assert len(match.sftp.mock_calls) == 2 | ||
first_mocked_call = match.sftp.mock_calls[0] | ||
first_called_method = str(first_mocked_call).split("(")[0].split(".")[1] | ||
assert first_called_method == "list_directory" | ||
assert set(first_mocked_call.args) == set(["/myDownloads/"]) | ||
|
||
second_mocked_call = match.sftp.mock_calls[1] | ||
second_called_method = str(second_mocked_call).split("(")[0].split(".")[1] | ||
assert second_called_method == "get_file" | ||
assert set(second_mocked_call.args) == set(["/myDownloads/example_12345"]) |