Merge pull request #4205 from alphagov/admin-3-providers-181681739

Fix provider UI to cope with 1 or 3+ SMS providers
This commit is contained in:
Ben Thorner
2022-04-11 14:53:36 +01:00
committed by GitHub
9 changed files with 294 additions and 637 deletions

View File

@@ -29,67 +29,4 @@ $item-top-padding: govuk-spacing(3);
display: block;
color: $secondary-text-colour;
}
&-percentage {
$axis-thickness: 2px;
border-bottom: $axis-thickness solid $black;
position: relative;
margin-bottom: 35px;
height: 35px;
&:before,
&:after {
content: "";
position: absolute;
bottom: -6px;
left: 0;
height: 10px;
width: $axis-thickness;
background: $black;
}
&:after {
left: auto;
right: 0;
}
&-without-border {
position: relative;
padding-top: 35px;
}
&-marker {
$size: 20px;
$border-thickness: 7px;
$text-width: 100px;
display: block;
width: $size;
height: $size;
position: absolute;
top: 35px - ($size / 2) - ($border-thickness - ($axis-thickness / 2));
margin-left: 0 - ($size / 2) - $border-thickness;
background: $black;
border-radius: $size;
border: $border-thickness solid $white;
}
&-left-label,
&-right-label {
width: 100%;
position: absolute;
top: 0;
}
&-left-label {
left: 0;
}
&-right-label {
right: 0;
text-align: right;
}
}
}

View File

@@ -1740,34 +1740,49 @@ class EstimateUsageForm(StripWhitespaceForm):
return super().validate(*args, **kwargs)
class AdminProviderForm(StripWhitespaceForm):
priority = GovukIntegerField(
'Priority', [validators.NumberRange(min=1, max=100, message="Must be between 1 and 100")]
)
class AdminProviderRatioForm(Form):
def __init__(self, providers):
self._providers = providers
# hack: https://github.com/wtforms/wtforms/issues/736
self._unbound_fields = [
(
provider['identifier'],
GovukIntegerField(
f"{provider['display_name']} (%)",
validators=[validators.NumberRange(
min=0, max=100, message="Must be between 0 and 100"
)],
param_extensions={
'classes': "govuk-input--width-3",
}
)
) for provider in providers
]
class AdminProviderRatioForm(StripWhitespaceForm):
ratio = GovukRadiosField(choices=[
(str(value), '{}% / {}%'.format(value, 100 - value))
for value in range(100, -10, -10)
],
param_extensions={
"classes": "govuk-radios--inline",
"fieldset": {
"legend": {
"classes": "govuk-visually-hidden"
}
}
super().__init__(data={
provider['identifier']: provider['priority']
for provider in providers
})
@property
def percentage_left(self):
return int(self.ratio.data)
def validate(self):
if not super().validate():
return False
@property
def percentage_right(self):
return 100 - self.percentage_left
total = sum(
getattr(self, provider['identifier']).data
for provider in self._providers
)
if total == 100:
return True
for provider in self._providers:
getattr(self, provider['identifier']).errors += [
'Must add up to 100%'
]
return False
class ServiceContactDetailsForm(StripWhitespaceForm):

View File

@@ -1,13 +1,11 @@
from collections import defaultdict
from datetime import datetime
from operator import itemgetter
from flask import abort, render_template, url_for
from flask import render_template, url_for
from werkzeug.utils import redirect
from app import format_date_numeric, provider_client
from app import provider_client
from app.main import main
from app.main.forms import AdminProviderForm, AdminProviderRatioForm
from app.main.forms import AdminProviderRatioForm
from app.utils.user import user_is_platform_admin
PROVIDER_PRIORITY_MEANING_SWITCHOVER = datetime(2019, 11, 29, 11, 0).isoformat()
@@ -44,79 +42,30 @@ def add_monthly_traffic(domestic_sms_providers):
provider['monthly_traffic'] = round(percentage)
@main.route("/provider/<uuid:provider_id>/edit", methods=['GET', 'POST'])
@user_is_platform_admin
def edit_provider(provider_id):
provider = provider_client.get_provider_by_id(provider_id)['provider_details']
form = AdminProviderForm(active=provider['active'], priority=provider['priority'])
if form.validate_on_submit():
provider_client.update_provider(provider_id, form.priority.data)
return redirect(url_for('.view_providers'))
return render_template('views/providers/edit-provider.html', form=form, provider=provider)
@main.route("/provider/edit-sms-provider-ratio", methods=['GET', 'POST'])
@user_is_platform_admin
def edit_sms_provider_ratio():
providers = sorted([
providers = [
provider
for provider in provider_client.get_all_providers()['provider_details']
if provider['notification_type'] == 'sms'
and provider['active']
], key=itemgetter('identifier'))
if provider['notification_type'] == 'sms' and provider['active']
]
form = AdminProviderRatioForm(ratio=providers[0]['priority'])
if len(providers) < 2:
abort(400)
first_provider, second_provider = providers[0:2]
form = AdminProviderRatioForm(providers)
if form.validate_on_submit():
provider_client.update_provider(first_provider['id'], form.percentage_left)
provider_client.update_provider(second_provider['id'], form.percentage_right)
return redirect(url_for('.edit_sms_provider_ratio'))
for provider in providers:
field = getattr(form, provider['identifier'])
provider_client.update_provider(provider['id'], field.data)
return redirect(url_for('.view_providers'))
return render_template(
'views/providers/edit-sms-provider-ratio.html',
versions=_chunk_versions_by_day(_get_versions_since_switchover(first_provider['id'])),
form=form,
first_provider=providers[0]['display_name'],
second_provider=providers[1]['display_name'],
providers=providers
)
def _get_versions_since_switchover(provider_id):
for version in sorted( # noqa: B020
provider_client.get_provider_versions(provider_id)['data'],
key=lambda version: version['updated_at'] or ''
):
if not version['updated_at']:
continue
if version['updated_at'] < PROVIDER_PRIORITY_MEANING_SWITCHOVER:
continue
yield version
def _chunk_versions_by_day(versions):
days = defaultdict(list)
for version in sorted(versions, key=lambda version: version['updated_at'] or '', reverse=True): # noqa: B020
days[
format_date_numeric(version['updated_at'])
].append(version)
return sorted(days.items(), reverse=True)
@main.route("/provider/<uuid:provider_id>")
@user_is_platform_admin
def view_provider(provider_id):

View File

@@ -1,37 +0,0 @@
{% extends "withoutnav_template.html" %}
{% from "components/page-header.html" import page_header %}
{% from "components/page-footer.html" import page_footer %}
{% from "components/form.html" import form_wrapper %}
{% from "components/back-link/macro.njk" import govukBackLink %}
{% block per_page_title %}
{{provider.display_name}}
{% endblock %}
{% block backLink %}
{{ govukBackLink({ "href": url_for('.view_providers') }) }}
{% endblock %}
{% block maincolumn_content %}
<div class="govuk-grid-row">
<div class="govuk-grid-column-three-quarters">
{{ page_header(provider.display_name) }}
<p class="govuk-body">Update provider:</p>
<ul class="list-bullet">
<li>We only send from the highest priority provider</li>
</ul>
{% call form_wrapper() %}
{{ form.priority }}
{{ page_footer('Save') }}
{% endcall %}
</div>
</div>
{% endblock %}

View File

@@ -4,87 +4,22 @@
{% from "components/form.html" import form_wrapper %}
{% block per_page_title %}
Text message providers
Edit SMS provider priorities
{% endblock %}
{% block platform_admin_content %}
{{ page_header('Text message providers') }}
{{ page_header('Edit SMS provider priorities') }}
<p class="govuk-body">
If you change one priority you will need to change the others so they add up to 100%.
</p>
<h2 class="heading-small top-gutter">
Now
</h2>
<div class="bottom-gutter">
<div class="history-list-item">
<div class="govuk-grid-row">
<div class="govuk-grid-column-one-third">&nbsp;</div>
<div class="govuk-grid-column-two-thirds">
<div class="history-list-percentage-without-border">
<div class="history-list-percentage-left-label">
{{ first_provider }}
</div>
<div class="history-list-percentage-right-label">
{{ second_provider }}
</div>
{% call form_wrapper() %}
<div class="radio-slider" data-module="radio-slider">
{{ form.ratio }}
<div class="radio-slider-left-value">
</div>
<div class="radio-slider-right-value">
</div>
</div>
{{ page_footer('Update', centered_button=True) }}
{% endcall %}
</div>
</div>
</div>
</div>
</div>
{% for day, versions in versions %}
<h2 class="heading-small top-gutter">
{% if day %}
{{ versions[0]['updated_at']|format_date_human|title }}
{% else %}
Start
{% endif %}
</h2>
<ul class="bottom-gutter">
{% for version in versions %}
<li class="history-list-item">
<div class="govuk-grid-row">
<div class="govuk-grid-column-one-third">
<div class="history-list-user">
{{ version.created_by.name or ('&nbsp;'|safe) }}
</div>
{% if version.updated_at %}
<div class="history-list-time">
{{ version.updated_at|format_time }}
</div>
{% endif %}
</div>
<div class="govuk-grid-column-two-thirds">
<div class="history-list-percentage">
<div class="history-list-percentage-left-label">
{{ first_provider }}<br><br>
{{ version.priority|format_thousands }}%
</div>
<div class="history-list-percentage-right-label">
{{ second_provider }}<br><br>
{{ (100 - version.priority)|format_thousands }}%
</div>
<div class="history-list-percentage-marker" style="left: {{ (100 -version.priority)|format_thousands }}%"></div>
</div>
</div>
</div>
</li>
{% endfor %}
</ul>
{% endfor %}
{% call form_wrapper() %}
{% for field in form %}
{{ field }}
{% endfor %}
{{ page_footer('Save') }}
{% endcall %}
{% endblock %}

View File

@@ -18,6 +18,11 @@
<div class="govuk-grid-column-two-thirds">
{{ page_header(provider_versions[0].display_name) }}
<p class="govuk-body">
Only showing the latest 100 versions.
</p>
{% call(item, row_number) list_table(
provider_versions,
caption='',

View File

@@ -25,17 +25,16 @@ Providers
{{ link_field(item.display_name, url_for('main.view_provider', provider_id=item.id)) }}
{{ text_field(item.priority) }}
{{ optional_text_field(item.priority if item.active, default='None') }}
{{ text_field(item.monthly_traffic) }}
{{ text_field(item.active) }}
{% if item.updated_at %}
{{ text_field(item.updated_at|format_datetime_short) }}
{% else %}
{{ text_field('None') }}
{% endif %}
{{ optional_text_field(
item.updated_at|format_datetime_short if item.updated_at,
default='None'
) }}
{{ optional_text_field(item.created_by_name, default='None') }}
@@ -48,26 +47,20 @@ Providers
caption="Email providers",
caption_visible=False,
empty_message='No email providers',
field_headings=['Provider', 'Priority', 'Active', 'Last Updated', 'Updated By'],
field_headings=['Provider', 'Active', 'Last Updated', 'Updated By'],
field_headings_visible=True
) %}
{{ link_field(item.display_name, url_for('main.view_provider', provider_id=item.id)) }}
{{ text_field(item.priority) }}
{{ text_field(item.active) }}
{% if item.updated_at %}
{{ text_field(item.updated_at|format_datetime_short) }}
{% else %}
{{ text_field('None') }}
{% endif %}
{{ optional_text_field(
item.updated_at|format_datetime_short if item.updated_at,
default='None'
) }}
{{ optional_text_field(item.created_by_name, default='None') }}
{{ link_field('change', url_for('main.edit_provider', provider_id=item.id)) }}
{% endcall %}
<h2 class="heading-medium">International SMS Providers</h2>
@@ -77,21 +70,18 @@ Providers
caption="International SMS providers",
caption_visible=False,
empty_message='No international sms providers',
field_headings=['Provider', 'Priority', 'Active', 'Last Updated', 'Updated By'],
field_headings=['Provider', 'Active', 'Last Updated', 'Updated By'],
field_headings_visible=True
) %}
{{ link_field(item.display_name, url_for('main.view_provider', provider_id=item.id)) }}
{{ text_field(item.priority) }}
{{ text_field(item.active) }}
{% if item.updated_at %}
{{ text_field(item.updated_at|format_datetime_short) }}
{% else %}
{{ text_field('None') }}
{% endif %}
{{ optional_text_field(
item.updated_at|format_datetime_short if item.updated_at,
default='None'
) }}
{{ optional_text_field(item.created_by_name, default='None') }}

View File

@@ -3,99 +3,108 @@ from unittest.mock import call
import pytest
from flask import url_for
from freezegun import freeze_time
import app
from app.main.views.providers import add_monthly_traffic
from tests.conftest import normalize_spaces
sms_provider_1 = {
'id': '6005e192-4738-4962-beec-ebd982d0b03f',
'active': True,
'priority': 20,
'display_name': 'First Domestic SMS Provider',
'identifier': 'first_sms_domestic',
'notification_type': 'sms',
'updated_at': datetime(2017, 1, 16, 15, 20, 40).isoformat(),
'version': 1,
'created_by_name': 'Test User',
'supports_international': False,
'current_month_billable_sms': 5020,
}
sms_provider_2 = {
'id': '0bd529cd-a0fd-43e5-80ee-b95ef6b0d51f',
'active': True,
'priority': 10,
'display_name': 'Second Domestic SMS Provider',
'identifier': 'second_sms_domestic',
'notification_type': 'sms',
'updated_at': None,
'version': 1,
'created_by': None,
'supports_international': False,
'current_month_billable_sms': 6891,
}
def provider_json(overrides):
provider = {
'id': 'override-me',
'active': True,
'priority': 20,
'display_name': 'Provider',
'identifier': 'override-me',
'notification_type': 'sms',
'updated_at': None,
'version': 1,
'created_by_name': None,
'supports_international': False,
'current_month_billable_sms': 0,
}
email_provider_1 = {
'id': '6005e192-4738-4962-beec-ebd982d0b03a',
'active': True,
'priority': 1,
'display_name': 'first_email_provider',
'identifier': 'first_email',
'notification_type': 'email',
'updated_at': None,
'version': 1,
'created_by': None,
'supports_international': False,
'current_month_billable_sms': 0,
}
email_provider_2 = {
'active': True,
'priority': 2,
'display_name': 'second_email_provider',
'identifier': 'second_email',
'id': '0bd529cd-a0fd-43e5-80ee-b95ef6b0d51b',
'notification_type': 'email',
'updated_at': None,
'version': 1,
'created_by': None,
'supports_international': False,
'current_month_billable_sms': 0,
}
sms_provider_intl_1 = {
'id': '67c770f5-918e-4afa-a5ff-880b9beb161d',
'active': False,
'priority': 10,
'display_name': 'First International SMS Provider',
'identifier': 'first_sms_international',
'notification_type': 'sms',
'updated_at': None,
'version': 1,
'created_by': None,
'supports_international': True,
'current_month_billable_sms': 0,
}
sms_provider_intl_2 = {
'id': '67c770f5-918e-4afa-a5ff-880b9beb161d',
'active': False,
'priority': 10,
'display_name': 'Second International SMS Provider',
'identifier': 'second_sms_international',
'notification_type': 'sms',
'updated_at': None,
'version': 1,
'created_by': None,
'supports_international': True,
'current_month_billable_sms': 0,
}
provider.update(**overrides)
return provider
@pytest.fixture
def stub_providers():
def sms_provider_1():
return provider_json({
'id': 'sms_provider_1-id',
'priority': 20,
'display_name': 'SMS Provider 1',
'identifier': 'sms_provider_1',
'notification_type': 'sms',
'updated_at': datetime(2017, 1, 16, 15, 20, 40).isoformat(),
'created_by_name': 'Test User',
'current_month_billable_sms': 5020,
})
@pytest.fixture
def sms_provider_2():
return provider_json({
'id': 'sms_provider_2-id',
'priority': 10,
'display_name': 'SMS Provider 2',
'identifier': 'sms_provider_2',
'notification_type': 'sms',
'current_month_billable_sms': 6891,
})
@pytest.fixture
def email_provider_1():
return provider_json({
'id': 'email_provider_1-id',
'display_name': 'Email Provider 1',
'identifier': 'email_provider_1',
'notification_type': 'email',
})
@pytest.fixture
def email_provider_2():
return provider_json({
'id': 'email_provider_2-id',
'display_name': 'Email Provider 2',
'identifier': 'email_provider_2',
'notification_type': 'email',
})
@pytest.fixture
def sms_provider_intl_1():
return provider_json({
'id': 'sms_provider_intl_1-id',
'active': False,
'display_name': 'SMS Provider Intl 1',
'identifier': 'sms_provider_intl_1',
'notification_type': 'sms',
'supports_international': True,
})
@pytest.fixture
def sms_provider_intl_2():
return provider_json({
'id': 'sms_provider_intl_2-id',
'active': False,
'display_name': 'SMS Provider Intl 2',
'identifier': 'sms_provider_intl_2',
'notification_type': 'sms',
'supports_international': True,
})
@pytest.fixture
def stub_providers(
sms_provider_1,
sms_provider_2,
email_provider_1,
email_provider_2,
sms_provider_intl_1,
sms_provider_intl_2,
):
return {
'provider_details': [
sms_provider_1,
@@ -108,13 +117,6 @@ def stub_providers():
}
@pytest.fixture
def stub_provider():
return {
'provider_details': sms_provider_1
}
@pytest.fixture
def stub_provider_history():
return {
@@ -151,7 +153,7 @@ def stub_provider_history():
}
def test_should_show_all_providers(
def test_view_providers_shows_all_providers(
client_request,
platform_admin_user,
mocker,
@@ -181,8 +183,8 @@ def test_should_show_all_providers(
domestic_sms_first_row = domestic_sms_table.tbody.find_all('tr')[0]
table_data = domestic_sms_first_row.find_all('td')
assert table_data[0].find_all("a")[0]['href'] == '/provider/6005e192-4738-4962-beec-ebd982d0b03f'
assert table_data[0].text.strip() == "First Domestic SMS Provider"
assert table_data[0].find_all("a")[0]['href'] == '/provider/sms_provider_1-id'
assert table_data[0].text.strip() == "SMS Provider 1"
assert table_data[1].text.strip() == "20"
assert table_data[2].text.strip() == "42"
assert table_data[3].text.strip() == "True"
@@ -192,8 +194,8 @@ def test_should_show_all_providers(
domestic_sms_second_row = domestic_sms_table.tbody.find_all('tr')[1]
table_data = domestic_sms_second_row.find_all('td')
assert table_data[0].find_all("a")[0]['href'] == '/provider/0bd529cd-a0fd-43e5-80ee-b95ef6b0d51f'
assert table_data[0].text.strip() == "Second Domestic SMS Provider"
assert table_data[0].find_all("a")[0]['href'] == '/provider/sms_provider_2-id'
assert table_data[0].text.strip() == "SMS Provider 2"
assert table_data[1].text.strip() == "10"
assert table_data[2].text.strip() == "58"
assert table_data[3].text.strip() == "True"
@@ -203,36 +205,29 @@ def test_should_show_all_providers(
domestic_email_first_row = domestic_email_table.tbody.find_all('tr')[0]
domestic_email_table_data = domestic_email_first_row.find_all('td')
assert domestic_email_table_data[0].find_all("a")[0]['href'] == '/provider/6005e192-4738-4962-beec-ebd982d0b03a'
assert domestic_email_table_data[0].text.strip() == "first_email_provider"
assert domestic_email_table_data[1].text.strip() == "1"
assert domestic_email_table_data[2].text.strip() == "True"
assert domestic_email_table_data[0].find_all("a")[0]['href'] == '/provider/email_provider_1-id'
assert domestic_email_table_data[0].text.strip() == "Email Provider 1"
assert domestic_email_table_data[1].text.strip() == "True"
assert domestic_email_table_data[2].text.strip() == "None"
assert domestic_email_table_data[3].text.strip() == "None"
assert domestic_email_table_data[4].text.strip() == "None"
assert domestic_email_table_data[5].find_all("a")[0]['href'] \
== '/provider/6005e192-4738-4962-beec-ebd982d0b03a/edit'
domestic_email_second_row = domestic_email_table.tbody.find_all('tr')[1]
domestic_email_table_data = domestic_email_second_row.find_all('td')
assert domestic_email_table_data[0].find_all("a")[0]['href'] == '/provider/0bd529cd-a0fd-43e5-80ee-b95ef6b0d51b'
assert domestic_email_table_data[0].text.strip() == "second_email_provider"
assert domestic_email_table_data[1].text.strip() == "2"
assert domestic_email_table_data[2].text.strip() == "True"
assert domestic_email_table_data[0].find_all("a")[0]['href'] == '/provider/email_provider_2-id'
assert domestic_email_table_data[0].text.strip() == "Email Provider 2"
assert domestic_email_table_data[1].text.strip() == "True"
assert domestic_email_table_data[2].text.strip() == "None"
assert domestic_email_table_data[3].text.strip() == "None"
assert domestic_email_table_data[4].text.strip() == "None"
assert domestic_email_table_data[5].find_all("a")[0]['href'] \
== '/provider/0bd529cd-a0fd-43e5-80ee-b95ef6b0d51b/edit'
international_sms_first_row = international_sms_table.tbody.find_all('tr')[0]
table_data = international_sms_first_row.find_all('td')
assert table_data[0].find_all("a")[0]['href'] == '/provider/67c770f5-918e-4afa-a5ff-880b9beb161d'
assert table_data[0].text.strip() == "First International SMS Provider"
assert table_data[1].text.strip() == "10"
assert table_data[2].text.strip() == "False"
assert table_data[0].find_all("a")[0]['href'] == '/provider/sms_provider_intl_1-id'
assert table_data[0].text.strip() == "SMS Provider Intl 1"
assert table_data[1].text.strip() == "False"
assert table_data[2].text.strip() == "None"
assert table_data[3].text.strip() == "None"
assert table_data[4].text.strip() == "None"
def test_add_monthly_traffic():
@@ -264,134 +259,7 @@ def test_add_monthly_traffic():
}]
def test_should_show_edit_provider_form(
client_request,
platform_admin_user,
mocker,
fake_uuid,
stub_provider
):
mocker.patch('app.provider_client.get_provider_by_id', return_value=stub_provider)
client_request.login(platform_admin_user)
page = client_request.get('main.edit_provider', provider_id=fake_uuid)
h1 = [header.text.strip() for header in page.find_all('h1')]
assert 'First Domestic SMS Provider' in h1
form = [form for form in page.find_all('form')]
form_elements = [element for element in form[0].find_all('input')]
assert form_elements[0]['value'] == '20'
assert form_elements[0]['name'] == 'priority'
def test_should_show_error_on_bad_provider_priority(
client_request,
platform_admin_user,
mocker,
stub_provider,
):
mocker.patch('app.provider_client.get_provider_by_id', return_value=stub_provider)
client_request.login(platform_admin_user)
page = client_request.post(
'main.edit_provider',
provider_id=stub_provider['provider_details']['id'],
_data={'priority': "not valid"},
_expected_status=200,
)
assert normalize_spaces(
page.select_one('.govuk-error-message').text
) == "Error: Not a valid integer value."
def test_should_show_error_on_negative_provider_priority(
client_request,
platform_admin_user,
mocker,
stub_provider,
):
mocker.patch('app.provider_client.get_provider_by_id', return_value=stub_provider)
client_request.login(platform_admin_user)
page = client_request.post(
'main.edit_provider',
provider_id=stub_provider['provider_details']['id'],
_data={'priority': -1},
_expected_status=200,
)
assert normalize_spaces(
page.select_one('.govuk-error-message').text
) == "Error: Must be between 1 and 100"
def test_should_show_error_on_too_big_provider_priority(
client_request,
platform_admin_user,
mocker,
stub_provider,
):
mocker.patch('app.provider_client.get_provider_by_id', return_value=stub_provider)
client_request.login(platform_admin_user)
page = client_request.post(
'main.edit_provider',
provider_id=stub_provider['provider_details']['id'],
_data={'priority': 101},
_expected_status=200,
)
assert normalize_spaces(
page.select_one('.govuk-error-message').text
) == "Error: Must be between 1 and 100"
def test_should_show_error_on_too_little_provider_priority(
client_request,
platform_admin_user,
mocker,
stub_provider,
):
mocker.patch('app.provider_client.get_provider_by_id', return_value=stub_provider)
client_request.login(platform_admin_user)
page = client_request.post(
'main.edit_provider',
provider_id=stub_provider['provider_details']['id'],
_data={'priority': 0},
_expected_status=200,
)
assert normalize_spaces(
page.select_one('.govuk-error-message').text
) == "Error: Must be between 1 and 100"
def test_should_update_provider_priority(
client_request,
platform_admin_user,
mocker,
stub_provider,
):
mocker.patch('app.provider_client.get_provider_by_id', return_value=stub_provider)
mocker.patch('app.provider_client.update_provider', return_value=stub_provider)
client_request.login(platform_admin_user)
client_request.post(
'main.edit_provider',
provider_id=stub_provider['provider_details']['id'],
_data={'priority': 2},
_expected_redirect='http://localhost/providers',
)
app.provider_client.update_provider.assert_called_with(stub_provider['provider_details']['id'], 2)
def test_should_show_provider_version_history(
def test_view_provider_shows_version_history(
client_request,
platform_admin_user,
mocker,
@@ -432,121 +300,80 @@ def test_should_show_provider_version_history(
assert second_row[4].text.strip() == "True"
@freeze_time('2022-2-22 15:00')
def test_should_show_version_history_for_first_two_sms_providers(
def test_edit_sms_provider_provider_ratio(
client_request,
platform_admin_user,
mocker,
stub_providers,
sms_provider_1
):
mocker.patch(
'app.provider_client.get_all_providers',
return_value=stub_providers
)
# Getting the history for one provider implicitly gives us the
# history of the other one (in a world with only two providers).
# The code picks the first provider in alphabetical order of its
# id i.e. sms_provider_1.
mocker.patch(
'app.provider_client.get_provider_versions',
return_value={'data': [
{
'id': id,
'priority': priority,
'display_name': sms_provider_1['display_name'],
'identifier': sms_provider_1['identifier'],
'updated_at': updated_at,
'created_by': {
'email_address': 'test@foo.bar',
'name': 'Test User',
'id': '7cc1dddb-bcbc-4739-8fc1-61bedde3332a'
},
'supports_international': False,
}
for updated_at, priority in [
(datetime(2022, 2, 22, 14).isoformat(), 100),
(datetime(2020, 1, 1, 5).isoformat(), 80),
(datetime(2020, 1, 1, 3).isoformat(), 10),
# Anything older than 11am on 29 November 2019
# should be ignored because the priority numbers
# didnt mean the same thing before then
(datetime(2019, 11, 29, 10, 59).isoformat(), 123),
(datetime(2000, 1, 1, 0).isoformat(), 1999),
(None, 30),
]
]}
client_request.login(platform_admin_user)
page = client_request.get(
'.edit_sms_provider_ratio',
)
client_request.login(platform_admin_user)
page = client_request.get('main.edit_sms_provider_ratio')
inputs = page.select('.govuk-input[type="text"]')
assert len(inputs) == 2
assert [
radio['value']
for radio in page.select('input[name=ratio]')
] == [
'100', '90', '80', '70', '60', '50', '40', '30', '20', '10', '0',
]
assert [
radio['value']
for radio in page.select('input[checked]')
] == [
str(sms_provider_1['priority'])
]
assert [
normalize_spaces(heading.text)
for heading in page.select('main h2')
] == [
'Now',
'Today',
'1 January 2020',
]
assert [
normalize_spaces(version.text)
for version in page.select('li.history-list-item')
] == [
(
'Test User 2:00pm '
'First Domestic SMS Provider 100% '
'Second Domestic SMS Provider 0%'
),
(
'Test User 5:00am '
'First Domestic SMS Provider 80% '
'Second Domestic SMS Provider 20%'
),
(
'Test User 3:00am '
'First Domestic SMS Provider 10% '
'Second Domestic SMS Provider 90%'
),
]
first_input = page.select_one('.govuk-input[name="sms_provider_1"]')
assert first_input.attrs['value'] == str(sms_provider_1['priority'])
@pytest.mark.parametrize('posted_number, expected_calls', [
(
'10',
[
call(sms_provider_1['id'], 10),
call(sms_provider_2['id'], 90),
],
),
(
'80',
[
call(sms_provider_1['id'], 80),
call(sms_provider_2['id'], 20),
],
),
])
def test_should_update_priority_of_first_two_sms_providers(
def test_edit_sms_provider_provider_ratio_only_shows_active_providers(
client_request,
platform_admin_user,
mocker,
posted_number,
stub_providers,
sms_provider_1,
):
sms_provider_1['active'] = False
mocker.patch(
'app.provider_client.get_all_providers',
return_value=stub_providers,
)
client_request.login(platform_admin_user)
page = client_request.get(
'.edit_sms_provider_ratio',
)
inputs = page.select('.govuk-input[type="text"]')
assert len(inputs) == 1
@pytest.mark.parametrize('post_data, expected_calls', [
(
{
'sms_provider_1': 10,
'sms_provider_2': 90
},
[
call('sms_provider_1-id', 10),
call('sms_provider_2-id', 90),
],
),
(
{
'sms_provider_1': 80,
'sms_provider_2': 20
},
[
call('sms_provider_1-id', 80),
call('sms_provider_2-id', 20),
],
),
])
def test_edit_sms_provider_ratio_submit(
client_request,
platform_admin_user,
mocker,
post_data,
expected_calls,
stub_providers,
):
@@ -554,10 +381,6 @@ def test_should_update_priority_of_first_two_sms_providers(
'app.provider_client.get_all_providers',
return_value=stub_providers
)
mocker.patch(
'app.provider_client.get_provider_versions',
return_value={'data': []}
)
mock_update_provider = mocker.patch(
'app.provider_client.update_provider'
)
@@ -565,13 +388,54 @@ def test_should_update_priority_of_first_two_sms_providers(
client_request.login(platform_admin_user)
client_request.post(
'.edit_sms_provider_ratio',
_data={
'ratio': posted_number,
},
_data=post_data,
_expected_redirect=url_for(
'.edit_sms_provider_ratio',
'.view_providers',
_external=True,
),
)
assert mock_update_provider.call_args_list == expected_calls
@pytest.mark.parametrize('post_data, expected_error', [
(
{
'sms_provider_1': 90,
'sms_provider_2': 20
},
"Must add up to 100%"
),
(
{
'sms_provider_1': 101,
'sms_provider_2': 20
},
"Must be between 0 and 100"
),
])
def test_edit_sms_provider_submit_invalid_percentages(
client_request,
platform_admin_user,
mocker,
post_data,
expected_error,
stub_providers,
):
mocker.patch(
'app.provider_client.get_all_providers',
return_value=stub_providers
)
mock_update_provider = mocker.patch(
'app.provider_client.update_provider'
)
client_request.login(platform_admin_user)
page = client_request.post(
'.edit_sms_provider_ratio',
_data=post_data,
_follow_redirects=True
)
assert expected_error in page.select_one('.govuk-error-message').text
mock_update_provider.assert_not_called()

View File

@@ -102,7 +102,6 @@ EXCLUDED_ENDPOINTS = tuple(map(Navigation.get_endpoint_with_blueprint, {
'edit_organisation_notes',
'edit_organisation_type',
'edit_organisation_user',
'edit_provider',
'edit_service_billing_details',
'edit_service_notes',
'edit_service_template',