|
import json |
|
import time |
|
import unittest |
|
from unittest.mock import patch |
|
|
|
from awslambdaric.lambda_context import LambdaContext |
|
|
|
from samgis.io import lambda_helpers |
|
from wrappers import lambda_wrapper |
|
from tests.local_tiles_http_server import LocalTilesHttpServer |
|
|
|
|
|
class TestAppFailures(unittest.TestCase): |
|
@patch.object(time, "time") |
|
@patch.object(lambda_wrapper, "samexporter_predict") |
|
@patch.object(lambda_wrapper, "get_parsed_bbox_points") |
|
@patch.object(lambda_wrapper, "get_parsed_request_body") |
|
def test_lambda_handler_500( |
|
self, |
|
get_parsed_request_body_mocked, |
|
get_parsed_bbox_points_mocked, |
|
samexporter_predict_mocked, |
|
time_mocked |
|
): |
|
from wrappers.lambda_wrapper import lambda_handler |
|
|
|
time_mocked.return_value = 0 |
|
get_parsed_request_body_mocked.value = {} |
|
get_parsed_bbox_points_mocked.return_value = {"bbox": "bbox_object", "prompt": "prompt_object", "zoom": 1} |
|
samexporter_predict_mocked.side_effect = ValueError("I raise a value error!") |
|
|
|
event = {"body": {}, "version": 1.0} |
|
lambda_context = LambdaContext( |
|
invoke_id="test_invoke_id", |
|
client_context=None, |
|
cognito_identity=None, |
|
epoch_deadline_time_in_ms=time.time() |
|
) |
|
expected_response_500 = '{"statusCode": 500, "header": {"Content-Type": "application/json"}, ' |
|
expected_response_500 += '"body": "{\\"duration_run\\": 0, \\"message\\": \\"Internal server error\\", ' |
|
expected_response_500 += '\\"request_id\\": \\"test_invoke_id\\"}", "isBase64Encoded": false}' |
|
|
|
assert lambda_handler(event, lambda_context) == expected_response_500 |
|
|
|
@patch.object(time, "time") |
|
@patch.object(lambda_wrapper, "get_parsed_request_body") |
|
def test_lambda_handler_400(self, get_parsed_request_body_mocked, time_mocked): |
|
from wrappers.lambda_wrapper import lambda_handler |
|
|
|
time_mocked.return_value = 0 |
|
get_parsed_request_body_mocked.return_value = {} |
|
|
|
event = {"body": {}, "version": 1.0} |
|
lambda_context = LambdaContext( |
|
invoke_id="test_invoke_id", |
|
client_context=None, |
|
cognito_identity=None, |
|
epoch_deadline_time_in_ms=time.time() |
|
) |
|
|
|
assert lambda_handler(event, lambda_context) == ( |
|
'{"statusCode": 400, "header": {"Content-Type": "application/json"}, ' |
|
'"body": "{\\"duration_run\\": 0, \\"message\\": \\"Bad Request\\", ' |
|
'\\"request_id\\": \\"test_invoke_id\\"}", "isBase64Encoded": false}') |
|
|
|
@patch.object(time, "time") |
|
def test_lambda_handler_422(self, time_mocked): |
|
from wrappers.lambda_wrapper import lambda_handler |
|
|
|
time_mocked.return_value = 0 |
|
event = {"body": {}, "version": 1.0} |
|
lambda_context = LambdaContext( |
|
invoke_id="test_invoke_id", |
|
client_context=None, |
|
cognito_identity=None, |
|
epoch_deadline_time_in_ms=time.time() |
|
) |
|
|
|
response_422 = lambda_handler(event, lambda_context) |
|
expected_response_422 = '{"statusCode": 422, "header": {"Content-Type": "application/json"}, ' |
|
expected_response_422 += '"body": "{\\"duration_run\\": 0, \\"message\\": \\"Missing required parameter\\", ' |
|
expected_response_422 += '\\"request_id\\": \\"test_invoke_id\\"}", "isBase64Encoded": false}' |
|
|
|
assert response_422 == expected_response_422 |
|
|
|
@patch.object(time, "time") |
|
@patch.object(lambda_wrapper, "samexporter_predict") |
|
@patch.object(lambda_wrapper, "get_response") |
|
@patch.object(lambda_wrapper, "get_parsed_bbox_points") |
|
@patch.object(lambda_wrapper, "get_parsed_request_body") |
|
def test_lambda_handler_200_mocked( |
|
self, |
|
get_parsed_request_body_mocked, |
|
get_parsed_bbox_points_mocked, |
|
get_response_mocked, |
|
samexporter_predict_mocked, |
|
time_mocked |
|
): |
|
from wrappers.lambda_wrapper import lambda_handler |
|
from tests import TEST_EVENTS_FOLDER |
|
|
|
time_mocked.return_value = 0 |
|
get_parsed_request_body_mocked.value = {} |
|
get_parsed_bbox_points_mocked.return_value = {"bbox": "bbox_object", "prompt": "prompt_object", "zoom": 1} |
|
|
|
response_type = "200" |
|
with open(TEST_EVENTS_FOLDER / "get_response.json") as tst_json_get_response: |
|
get_response_io = json.load(tst_json_get_response) |
|
|
|
input_200 = { |
|
"bbox": { |
|
"ne": {"lat": 38.03932961278458, "lng": 15.36808069832851}, |
|
"sw": {"lat": 37.455509218936974, "lng": 14.632807441554068} |
|
}, |
|
"prompt": [{ |
|
"type": "point", |
|
"data": {"lat": 37.0, "lng": 15.0}, |
|
"label": 0 |
|
}], |
|
"zoom": 10, |
|
"source_type": "OpenStreetMap.Mapnik", |
|
"debug": True |
|
} |
|
|
|
samexporter_predict_output = get_response_io[response_type]["input"] |
|
samexporter_predict_mocked.return_value = samexporter_predict_output |
|
samexporter_predict_mocked.side_effect = None |
|
get_response_mocked.return_value = get_response_io[response_type]["output"] |
|
|
|
event = {"body": input_200, "version": 1.0} |
|
|
|
lambda_context = LambdaContext( |
|
invoke_id="test_invoke_id", |
|
client_context=None, |
|
cognito_identity=None, |
|
epoch_deadline_time_in_ms=time.time() |
|
) |
|
|
|
response_200 = lambda_handler(event, lambda_context) |
|
expected_response_200 = get_response_io[response_type]["output"] |
|
print(f"types: response_200:{type(response_200)}, expected:{type(expected_response_200)}.") |
|
assert response_200 == expected_response_200 |
|
|
|
@patch.object(lambda_helpers, "get_url_tile") |
|
def test_lambda_handler_200_real_single_multi_point(self, get_url_tile_mocked): |
|
import xyzservices |
|
import shapely |
|
|
|
from wrappers.lambda_wrapper import lambda_handler |
|
from tests import LOCAL_URL_TILE, TEST_EVENTS_FOLDER |
|
|
|
local_tile_provider = xyzservices.TileProvider(name="local_tile_provider", url=LOCAL_URL_TILE, attribution="") |
|
get_url_tile_mocked.return_value = local_tile_provider |
|
fn_name = "lambda_handler" |
|
invoke_id = "test_invoke_id" |
|
|
|
for json_filename in [ |
|
"single_point", |
|
"multi_prompt", |
|
"single_rectangle" |
|
]: |
|
with open(TEST_EVENTS_FOLDER / f"{fn_name}_{json_filename}.json") as tst_json: |
|
inputs_outputs = json.load(tst_json) |
|
lambda_context = LambdaContext( |
|
invoke_id=invoke_id, |
|
client_context=None, |
|
cognito_identity=None, |
|
epoch_deadline_time_in_ms=time.time() |
|
) |
|
expected_response_dict = inputs_outputs["output"] |
|
listen_port = 8000 |
|
expected_response_body = json.loads(expected_response_dict["body"]) |
|
|
|
with LocalTilesHttpServer.http_server("localhost", listen_port, directory=TEST_EVENTS_FOLDER): |
|
input_event = inputs_outputs["input"] |
|
input_event_body = json.loads(input_event["body"]) |
|
input_event["body"] = json.dumps(input_event_body) |
|
response = lambda_handler(event=input_event, context=lambda_context) |
|
|
|
response_dict = json.loads(response) |
|
assert response_dict["statusCode"] == 200 |
|
body_dict = json.loads(response_dict["body"]) |
|
assert body_dict["n_predictions"] == 1 |
|
assert body_dict["request_id"] == invoke_id |
|
assert body_dict["message"] == "ok" |
|
assert body_dict["n_shapes_geojson"] == expected_response_body["n_shapes_geojson"] |
|
|
|
output_geojson = shapely.from_geojson(body_dict["geojson"]) |
|
print("output_geojson::", type(output_geojson)) |
|
assert isinstance(output_geojson, shapely.GeometryCollection) |
|
assert len(output_geojson.geoms) == expected_response_body["n_shapes_geojson"] |
|
|
|
def test_debug(self): |
|
from wrappers.lambda_wrapper import lambda_handler |
|
|
|
input_event = { |
|
'bbox': { |
|
'ne': {'lat': 46.302592089330524, 'lng': 9.49493408203125}, |
|
'sw': {'lat': 46.14011755129237, 'lng': 9.143371582031252}}, |
|
'prompt': [ |
|
{'id': 166, 'type': 'point', 'data': {'lat': 46.18244521829928, 'lng': 9.418544769287111}, 'label': 1} |
|
], |
|
'zoom': 12, 'source_type': 'OpenStreetMap' |
|
} |
|
lambda_context = LambdaContext( |
|
invoke_id="test_invoke_id", |
|
client_context=None, |
|
cognito_identity=None, |
|
epoch_deadline_time_in_ms=time.time() |
|
) |
|
response = lambda_handler(event=input_event, context=lambda_context) |
|
print(response) |
|
|