Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

staging -> main #409

Merged
merged 10 commits into from
Dec 3, 2024
2 changes: 1 addition & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ Formerly known as "Help to Heat".
### (Optional) Poetry Setup
If you want to be able to make changes to the dependency list using Poetry, or install the dependencies locally (so you can click into them in your IDE, or use autocomplete) perform these steps:
1. [Install Poetry](https://python-poetry.org/docs/) on your machine
2. [Install Python 3.9.13](https://www.python.org/downloads/release/python-3913/)
2. [Install Python 3.11.9](https://www.python.org/downloads/release/python-3119/)
3. Set up a virtual environment within the project for Poetry to manage your dependencies. A guide for Pycharm can be found [here](https://www.jetbrains.com/help/pycharm/poetry.html).
## Using Docker

Expand Down
46 changes: 43 additions & 3 deletions help_to_heat/frontdoor/interface.py
Original file line number Diff line number Diff line change
@@ -1,11 +1,13 @@
import ast
import logging
import time
from http import HTTPStatus

import marshmallow
import osdatahub
import requests
from django.conf import settings
from requests import RequestException

from help_to_heat import portal
from help_to_heat.utils import Entity, Interface, register_event, with_schema
Expand Down Expand Up @@ -37,6 +39,8 @@
from .os_api import OSApi, ThrottledApiException
from .routing import calculate_journey

logger = logging.getLogger(__name__)


class SaveAnswerSchema(marshmallow.Schema):
session_id = marshmallow.fields.UUID()
Expand Down Expand Up @@ -613,9 +617,45 @@ def get_address_and_epc_lmk(self, building_name_or_number, postcode):

def get_epc(self, lmk):
epc_api = EPCApi()
epc_details = epc_api.get_epc_details(lmk)
epc_recommendations = epc_api.get_epc_recommendations(lmk)
return epc_details, epc_recommendations
epc_details_response = self._get_epc_details(epc_api, lmk)
epc_recommendations = self._get_epc_recommendations(epc_api, lmk)
return (
epc_details_response,
epc_recommendations,
)

def _get_epc_details(self, epc_api, lmk):
return epc_api.get_epc_details(lmk)["rows"][0]

RECOMMENDATION_RETRIES_COUNT = 5

def _get_epc_recommendations(self, epc_api, lmk):
delay = 0.5
for i in range(self.RECOMMENDATION_RETRIES_COUNT):
try:
return epc_api.get_epc_recommendations(lmk)["rows"]
except RequestException as requestException:
match requestException.response.status_code:
case 500:
# if on the last try
if i == self.RECOMMENDATION_RETRIES_COUNT - 1:
# raise this to logs, though let user continue with no recommendations
logger.exception(requestException)
return []
else:
# exponential backoff
time.sleep(delay)
delay = delay * 1.5

# else if not on last try, try again
continue

# 404 is confirmed to mean no recommendations, so this is e.b. and so can return an empty list
case 404:
return []

# re-raise any other status codes
raise requestException


class Feedback(Entity):
Expand Down
38 changes: 38 additions & 0 deletions help_to_heat/frontdoor/mock_epc_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,38 @@ def get_epc_recommendations(self, lmk_key):
raise NotFoundRequestException()


class MockRecommendationsNotFoundEPCApi(MockEPCApi):
def get_epc_recommendations(self, lmk_key):
raise NotFoundRequestException()


class MockRecommendationsInternalServerErrorEPCApi(MockEPCApi):
def get_epc_recommendations(self, lmk_key):
raise InternalServerErrorRequestException()


class MockRecommendationsTransientInternalServerErrorEPCApi(MockEPCApi):
number_of_calls = 0

def get_epc_recommendations(self, lmk_key):
self.number_of_calls += 1
if self.number_of_calls <= 2:
raise InternalServerErrorRequestException()

return super().get_epc_recommendations(lmk_key)


def get_mock_epc_api_expecting_address_and_postcode(check_address, check_postcode):
class MockEpcApiExpectingAddressAndPostcode(MockEPCApi):
def search_epc_details(self, address, postcode):
if address != check_address or postcode != check_postcode:
raise NotFoundRequestException()
else:
return super().search_epc_details(address, postcode)

return MockEpcApiExpectingAddressAndPostcode


class UnauthorizedRequestException(requests.exceptions.RequestException):
def __init__(self):
self.response = requests.Response()
Expand All @@ -66,3 +98,9 @@ class NotFoundRequestException(requests.exceptions.RequestException):
def __init__(self):
self.response = requests.Response()
self.response.status_code = HTTPStatus.NOT_FOUND


class InternalServerErrorRequestException(requests.exceptions.RequestException):
def __init__(self):
self.response = requests.Response()
self.response.status_code = HTTPStatus.INTERNAL_SERVER_ERROR
19 changes: 11 additions & 8 deletions help_to_heat/frontdoor/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -619,8 +619,13 @@ def save_post_data(self, data, session_id, page_name):
reset_epc_details(session_id)
session_data = interface.api.session.get_session(session_id)
country = session_data.get(country_field)
building_name_or_number = data.get(address_building_name_or_number_field)
postcode = data.get(address_postcode_field)
building_name_or_number = data.get(address_building_name_or_number_field).strip()
postcode = data.get(address_postcode_field).strip()

# overwrite the answers with stripped ones
data[address_building_name_or_number_field] = building_name_or_number
data[address_postcode_field] = postcode

try:
if country != country_field_scotland:
address_and_lmk_details = interface.api.epc.get_address_and_epc_lmk(building_name_or_number, postcode)
Expand Down Expand Up @@ -683,9 +688,7 @@ def save_post_data(self, data, session_id, page_name):
return data

try:
epc_details_response, epc_recommendations_response = interface.api.epc.get_epc(lmk)
epc_details = epc_details_response["rows"][0]
recommendations = epc_recommendations_response["rows"]
epc_details, recommendations = interface.api.epc.get_epc(lmk)
except Exception as e: # noqa: B902
logger.exception(f"An error occurred: {e}")
reset_epc_details(session_id)
Expand Down Expand Up @@ -718,9 +721,9 @@ def save_post_data(self, data, session_id, page_name):
@register_page(address_select_page)
class AddressSelectView(PageView):
def build_extra_context(self, request, session_id, page_name, data, is_change_page):
data = interface.api.session.get_answer(session_id, address_page)
building_name_or_number = data[address_building_name_or_number_field]
postcode = data[address_postcode_field]
address_data = interface.api.session.get_answer(session_id, address_page)
building_name_or_number = address_data[address_building_name_or_number_field]
postcode = address_data[address_postcode_field]
addresses = interface.api.address.find_addresses(building_name_or_number, postcode)

uprn_options = tuple(
Expand Down
4 changes: 2 additions & 2 deletions help_to_heat/locale/cy/LC_MESSAGES/django.po
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ msgid ""
msgstr ""
"Project-Id-Version: PACKAGE VERSION\n"
"Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2024-11-25 11:59+0000\n"
"POT-Creation-Date: 2024-11-28 09:50+0000\n"
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
"Language-Team: LANGUAGE <LL@li.org>\n"
Expand Down Expand Up @@ -671,7 +671,7 @@ msgstr "Fflat"
msgid "Maisonette"
msgstr "Fflat deulawr"

#: help_to_heat/frontdoor/views.py:667 help_to_heat/frontdoor/views.py:737
#: help_to_heat/frontdoor/views.py:672 help_to_heat/frontdoor/views.py:740
msgid "I cannot find my address, I want to enter it manually"
msgstr "Dwi'n methu gweld fy nghyfeiriad i. Hoffwn ei roi fy hunan."

Expand Down
2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@ phonenumbers = "^8.13.38"

[tool.black]
line-length = 120
target-version = ['py38']
target-version = ['py311']

[tool.isort]
multi_line_output = 3
Expand Down
138 changes: 138 additions & 0 deletions tests/test_frontdoor.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,13 +8,23 @@
from help_to_heat.frontdoor import interface
from help_to_heat.frontdoor.consts import (
address_all_address_and_lmk_details_field,
address_building_name_or_number_field,
address_page,
address_postcode_field,
country_field,
country_page,
lmk_field,
recommendations_field,
supplier_field,
)
from help_to_heat.frontdoor.mock_epc_api import (
MockEPCApi,
MockEPCApiWithEPCC,
MockNotFoundEPCApi,
MockRecommendationsInternalServerErrorEPCApi,
MockRecommendationsNotFoundEPCApi,
MockRecommendationsTransientInternalServerErrorEPCApi,
get_mock_epc_api_expecting_address_and_postcode,
)
from help_to_heat.frontdoor.mock_os_api import EmptyOSApi, MockOSApi
from help_to_heat.portal import models
Expand Down Expand Up @@ -2157,6 +2167,97 @@ def test_epc_select_only_shows_most_recent_epc_per_uprn():
assert data[address_all_address_and_lmk_details_field][1]["lmk-key"] != "3333333333333333333333333333333333"


@unittest.mock.patch("help_to_heat.frontdoor.interface.EPCApi", MockRecommendationsNotFoundEPCApi)
def test_on_recommendations_not_found_response_recommendations_are_empty_list():
_get_empty_recommendations_journey()


@unittest.mock.patch("help_to_heat.frontdoor.interface.EPCApi", MockRecommendationsInternalServerErrorEPCApi)
def test_on_recommendations_internal_server_error_response_recommendations_are_empty_list():
_get_empty_recommendations_journey()


def _get_empty_recommendations_journey():
client = utils.get_client()
page = client.get("/start")
assert page.status_code == 302
page = page.follow()

assert page.status_code == 200
session_id = page.path.split("/")[1]
assert uuid.UUID(session_id)
_check_page = _make_check_page(session_id)

form = page.get_form()
form[country_field] = "England"
page = form.submit().follow()

form = page.get_form()
form[supplier_field] = "Utilita"
page = form.submit().follow()

assert page.has_text("Do you own the property?")
page = _check_page(page, "own-property", "own_property", "Yes, I own my property and live in it")

assert page.has_text("Do you live in a park home")
page = _check_page(page, "park-home", "park_home", "No")

form = page.get_form()
form[address_building_name_or_number_field] = "22"
form[address_postcode_field] = "FL23 4JA"
page = form.submit().follow()

form = page.get_form()
form[lmk_field] = "222222222222222222222222222222222"
page = form.submit().follow()

data = interface.api.session.get_answer(session_id, page_name="epc-select")
assert data[recommendations_field] == []

assert page.has_one("h1:contains('What is the council tax band of your property?')")


@unittest.mock.patch("help_to_heat.frontdoor.interface.EPCApi", MockRecommendationsTransientInternalServerErrorEPCApi)
def test_on_recommendations_transient_internal_server_error_response_recommendations_are_empty_list():
client = utils.get_client()
page = client.get("/start")
assert page.status_code == 302
page = page.follow()

assert page.status_code == 200
session_id = page.path.split("/")[1]
assert uuid.UUID(session_id)
_check_page = _make_check_page(session_id)

form = page.get_form()
form[country_field] = "England"
page = form.submit().follow()

form = page.get_form()
form[supplier_field] = "Utilita"
page = form.submit().follow()

assert page.has_text("Do you own the property?")
page = _check_page(page, "own-property", "own_property", "Yes, I own my property and live in it")

assert page.has_text("Do you live in a park home")
page = _check_page(page, "park-home", "park_home", "No")

form = page.get_form()
form[address_building_name_or_number_field] = "22"
form[address_postcode_field] = "FL23 4JA"
page = form.submit().follow()

form = page.get_form()
form[lmk_field] = "222222222222222222222222222222222"
page = form.submit().follow()

assert page.has_one("h1:contains('What is the council tax band of your property?')")
page = _check_page(page, "council-tax-band", "council_tax_band", "B")

assert page.has_one("h1:contains('We found an Energy Performance Certificate that might be yours')")


@unittest.mock.patch("help_to_heat.frontdoor.interface.EPCApi", MockEPCApi)
def test_success_page_still_shows_if_journey_cannot_reach_it():
supplier = "Utilita"
Expand All @@ -2171,6 +2272,43 @@ def test_success_page_still_shows_if_journey_cannot_reach_it():
assert page.has_one(f"h1:contains('Your details have been submitted to {supplier}')")


@unittest.mock.patch(
"help_to_heat.frontdoor.interface.EPCApi", get_mock_epc_api_expecting_address_and_postcode("22", "FL23 4JA")
)
def test_epc_api_is_called_with_trimmed_address_and_postcode():
client = utils.get_client()
page = client.get("/start")
assert page.status_code == 302
page = page.follow()

assert page.status_code == 200
session_id = page.path.split("/")[1]
assert uuid.UUID(session_id)
_check_page = _make_check_page(session_id)

form = page.get_form()
form["country"] = "England"
page = form.submit().follow()

form = page.get_form()
form["supplier"] = "Utilita"
page = form.submit().follow()

assert page.has_text("Do you own the property?")
page = _check_page(page, "own-property", "own_property", "Yes, I own my property and live in it")

assert page.has_text("Do you live in a park home")
page = _check_page(page, "park-home", "park_home", "No")

form = page.get_form()
form["building_name_or_number"] = " 22 "
form["postcode"] = " FL23 4JA "
page = form.submit().follow()

assert page.has_one("label:contains('22 Acacia Avenue, Upper Wellgood, Fulchester, FL23 4JA')")
assert page.has_one("label:contains('11 Acacia Avenue, Upper Wellgood, Fulchester, FL23 4JA')")


def _setup_client_and_page():
client = utils.get_client()
page = client.get("/start")
Expand Down
2 changes: 1 addition & 1 deletion tests/test_interface.py
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ def test_get_address():
def test_get_epc():
assert interface.api.epc.get_address_and_epc_lmk("22", "FL23 4JA")
found_epc, _ = interface.api.epc.get_epc("1111111111111111111111111111111111")
assert found_epc["rows"][0].get("current-energy-rating").upper() == "G"
assert found_epc.get("current-energy-rating").upper() == "G"


@unittest.mock.patch("help_to_heat.frontdoor.interface.EPCApi", MockNotFoundEPCApi)
Expand Down
Loading