Files
notifications-admin/tests/app/main/views/test_add_service.py
Chris Hill-Scott 07318b2d11 Replace instances of client.login with client_request
We have a `client_request` fixture which does a bunch of useful stuff
like:
- checking the status code of the response
- returning a `BeautifulSoup` object

Lots of our tests still use an older fixture called `client`. This is
not as good because it:
- returns a raw `Response` object
- doesn’t do the additional checks
- means our tests contain a lot of repetetive boilerplate like `page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')`

This commit converts all the tests which had a `client.login(…)`
statement to use `client_request` (which is already logged in by
default).

Subsequent commits will remove uses of `client` in other tests, but
doing it this way means the work can be broken up into more manageable
chunks.
2022-01-10 14:39:45 +00:00

362 lines
11 KiB
Python
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
import pytest
from flask import url_for
from freezegun import freeze_time
from notifications_python_client.errors import HTTPError
from app.utils.user import is_gov_user
from tests import organisation_json
from tests.conftest import normalize_spaces
def test_non_gov_user_cannot_see_add_service_button(
client_request,
mock_login,
mock_get_non_govuser,
api_nongov_user_active,
mock_get_organisations,
mock_get_organisations_and_services_for_user,
):
client_request.login(api_nongov_user_active)
page = client_request.get('main.choose_account')
assert 'Add a new service' not in page.text
@pytest.mark.parametrize('org_json', (
None,
organisation_json(organisation_type=None),
))
def test_get_should_render_add_service_template(
client_request,
mocker,
org_json,
):
mocker.patch(
'app.organisations_client.get_organisation_by_domain',
return_value=org_json,
)
page = client_request.get('main.add_service')
assert page.select_one('h1').text.strip() == 'About your service'
assert page.select_one('input[name=name]').get('value') is None
assert [
label.text.strip() for label in page.select('.govuk-radios__item label')
] == [
'Central government',
'Local government',
'NHS central government agency or public body',
'NHS Trust or Clinical Commissioning Group',
'GP practice',
'Emergency service',
'School or college',
'Other',
]
assert [
radio['value'] for radio in page.select('.govuk-radios__item input')
] == [
'central',
'local',
'nhs_central',
'nhs_local',
'nhs_gp',
'emergency_service',
'school_or_college',
'other',
]
def test_get_should_not_render_radios_if_org_type_known(
client_request,
mocker,
):
mocker.patch(
'app.organisations_client.get_organisation_by_domain',
return_value=organisation_json(organisation_type='central'),
)
page = client_request.get('main.add_service')
assert page.select_one('h1').text.strip() == 'About your service'
assert page.select_one('input[name=name]').get('value') is None
assert not page.select('.multiple-choice')
def test_show_different_page_if_user_org_type_is_local(
client_request,
mocker,
):
mocker.patch(
'app.organisations_client.get_organisation_by_domain',
return_value=organisation_json(organisation_type='local'),
)
page = client_request.get('main.add_service')
assert page.select_one('h1').text.strip() == 'About your service'
assert page.select_one('input[name=name]').get('value') is None
assert page.select_one('main .govuk-body').text.strip() == (
'Give your service a name that tells users what your '
'messages are about, as well as who theyre from. For example:')
@pytest.mark.parametrize('email_address', (
# Users email address doesnt matter when the organisation is known
'test@example.gov.uk',
'test@example.nhs.uk',
))
@pytest.mark.parametrize('inherited, posted, persisted, sms_limit', (
(None, 'central', 'central', 150_000),
(None, 'nhs_central', 'nhs_central', 150_000),
(None, 'nhs_gp', 'nhs_gp', 10_000),
(None, 'nhs_local', 'nhs_local', 25_000),
(None, 'local', 'local', 25_000),
(None, 'emergency_service', 'emergency_service', 25_000),
(None, 'school_or_college', 'school_or_college', 10_000),
(None, 'other', 'other', 10_000),
('central', None, 'central', 150_000),
('nhs_central', None, 'nhs_central', 150_000),
('nhs_local', None, 'nhs_local', 25_000),
('local', None, 'local', 25_000),
('emergency_service', None, 'emergency_service', 25_000),
('school_or_college', None, 'school_or_college', 10_000),
('other', None, 'other', 10_000),
('central', 'local', 'central', 150_000),
))
@freeze_time("2021-01-01")
def test_should_add_service_and_redirect_to_tour_when_no_services(
mocker,
client_request,
mock_create_service,
mock_create_service_template,
mock_get_services_with_no_services,
api_user_active,
mock_get_all_email_branding,
inherited,
email_address,
posted,
persisted,
sms_limit,
):
api_user_active['email_address'] = email_address
client_request.login(api_user_active)
mocker.patch(
'app.organisations_client.get_organisation_by_domain',
return_value=organisation_json(organisation_type=inherited),
)
client_request.post(
'main.add_service',
_data={
'name': 'testing the post',
'organisation_type': posted,
},
_expected_status=302,
_expected_redirect=url_for(
'main.begin_tour',
service_id=101,
template_id="Example%20text%20message%20template",
_external=True,
),
)
assert mock_get_services_with_no_services.called
mock_create_service.assert_called_once_with(
service_name='testing the post',
organisation_type=persisted,
message_limit=50,
restricted=True,
user_id=api_user_active['id'],
email_from='testing.the.post',
)
mock_create_service_template.assert_called_once_with(
'Example text message template',
'sms',
(
'Hey ((name)), Im trying out Notify. Today is '
'((day of week)) and my favourite colour is ((colour)).'
),
101,
)
with client_request.session_transaction() as session:
assert session['service_id'] == 101
def test_add_service_has_to_choose_org_type(
mocker,
client_request,
mock_create_service,
mock_create_service_template,
mock_get_services_with_no_services,
api_user_active,
mock_get_all_email_branding,
):
mocker.patch(
'app.organisations_client.get_organisation_by_domain',
return_value=None,
)
page = client_request.post(
'main.add_service',
_data={
'name': 'testing the post',
},
_expected_status=200,
)
assert normalize_spaces(page.select_one('.govuk-error-message').text) == (
'Error: Select the type of organisation'
)
assert mock_create_service.called is False
assert mock_create_service_template.called is False
@pytest.mark.parametrize('email_address', (
'test@nhs.net',
'test@nhs.uk',
'test@example.NhS.uK',
'test@EXAMPLE.NHS.NET',
))
def test_get_should_only_show_nhs_org_types_radios_if_user_has_nhs_email(
client_request,
mocker,
api_user_active,
email_address,
):
api_user_active['email_address'] = email_address
client_request.login(api_user_active)
mocker.patch(
'app.organisations_client.get_organisation_by_domain',
return_value=None,
)
page = client_request.get('main.add_service')
assert page.select_one('h1').text.strip() == 'About your service'
assert page.select_one('input[name=name]').get('value') is None
assert [
label.text.strip() for label in page.select('.govuk-radios__item label')
] == [
'NHS central government agency or public body',
'NHS Trust or Clinical Commissioning Group',
'GP practice',
]
assert [
radio['value'] for radio in page.select('.govuk-radios__item input')
] == [
'nhs_central',
'nhs_local',
'nhs_gp',
]
@pytest.mark.parametrize('organisation_type, free_allowance', [
('central', 150_000),
('local', 25_000),
('nhs_central', 150_000),
('nhs_local', 25_000),
('nhs_gp', 10_000),
('school_or_college', 10_000),
('emergency_service', 25_000),
('other', 10_000),
])
def test_should_add_service_and_redirect_to_dashboard_when_existing_service(
notify_admin,
mocker,
client_request,
mock_create_service,
mock_create_service_template,
mock_get_services,
mock_get_no_organisation_by_domain,
api_user_active,
organisation_type,
free_allowance,
mock_get_all_email_branding,
):
client_request.post(
'main.add_service',
_data={
'name': 'testing the post',
'organisation_type': organisation_type,
},
_expected_status=302,
_expected_redirect=url_for(
'main.service_dashboard',
service_id=101,
_external=True,
)
)
assert mock_get_services.called
mock_create_service.assert_called_once_with(
service_name='testing the post',
organisation_type=organisation_type,
message_limit=notify_admin.config['DEFAULT_SERVICE_LIMIT'],
restricted=True,
user_id=api_user_active['id'],
email_from='testing.the.post',
)
assert len(mock_create_service_template.call_args_list) == 0
with client_request.session_transaction() as session:
assert session['service_id'] == 101
@pytest.mark.parametrize('name, error_message', [
('', 'Cannot be empty'),
('.', 'Must include at least two alphanumeric characters'),
('a' * 256, 'Service name must be 255 characters or fewer'),
])
def test_add_service_fails_if_service_name_fails_validation(
client_request,
mock_get_organisation_by_domain,
name,
error_message,
):
page = client_request.post(
'main.add_service',
_data={"name": name},
_expected_status=200,
)
assert error_message in page.find("span", {"class": "govuk-error-message"}).text
@freeze_time("2021-01-01")
def test_should_return_form_errors_with_duplicate_service_name_regardless_of_case(
client_request,
mock_get_organisation_by_domain,
mocker,
):
def _create(**_kwargs):
json_mock = mocker.Mock(return_value={'message': {'name': ["Duplicate service name"]}})
resp_mock = mocker.Mock(status_code=400, json=json_mock)
http_error = HTTPError(response=resp_mock, message="Default message")
raise http_error
mocker.patch(
'app.service_api_client.create_service',
side_effect=_create
)
page = client_request.post(
'main.add_service',
_data={
'name': 'SERVICE ONE',
'organisation_type': 'central',
},
_expected_status=200,
)
assert 'This service name is already in use' in page.select_one('.govuk-error-message').text.strip()
def test_non_government_user_cannot_access_create_service_page(
client_request,
mock_get_non_govuser,
api_nongov_user_active,
mock_get_organisations,
):
assert is_gov_user(api_nongov_user_active['email_address']) is False
client_request.get(
'main.add_service',
_expected_status=403,
)
def test_non_government_user_cannot_create_service(
client_request,
mock_get_non_govuser,
api_nongov_user_active,
mock_get_organisations,
):
assert is_gov_user(api_nongov_user_active['email_address']) is False
client_request.post(
'main.add_service',
_data={'name': 'SERVICE TWO'},
_expected_status=403,
)