mirror of
https://github.com/GSA/notifications-admin.git
synced 2026-02-23 11:51:05 -05:00
878 lines
28 KiB
Python
878 lines
28 KiB
Python
from functools import partial
|
||
import copy
|
||
from unittest.mock import call, ANY
|
||
|
||
from flask import url_for
|
||
import pytest
|
||
from bs4 import BeautifulSoup
|
||
from datetime import datetime, timedelta
|
||
from freezegun import freeze_time
|
||
|
||
from app.main.views.dashboard import (
|
||
get_dashboard_totals,
|
||
format_monthly_stats_to_list,
|
||
get_free_paid_breakdown_for_billable_units,
|
||
aggregate_status_types,
|
||
format_template_stats_to_list,
|
||
get_tuples_of_financial_years,
|
||
get_dashboard_partials
|
||
)
|
||
|
||
from tests import validate_route_permission
|
||
from tests.conftest import (
|
||
SERVICE_ONE_ID,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_inbound_sms_summary_with_no_messages,
|
||
)
|
||
from tests.app.test_utils import normalize_spaces
|
||
|
||
stub_template_stats = [
|
||
{
|
||
'template_type': 'sms',
|
||
'template_name': 'one',
|
||
'template_id': 'id-1',
|
||
'count': 100
|
||
},
|
||
{
|
||
'template_type': 'email',
|
||
'template_name': 'two',
|
||
'template_id': 'id-2',
|
||
'count': 200
|
||
}
|
||
]
|
||
|
||
|
||
def test_get_started(
|
||
logged_in_client,
|
||
mocker,
|
||
mock_get_service_templates_when_no_templates_exist,
|
||
mock_get_jobs,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
mock_template_stats = mocker.patch('app.template_statistics_client.get_template_statistics_for_service',
|
||
return_value=copy.deepcopy(stub_template_stats))
|
||
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
# mock_get_service_templates_when_no_templates_exist.assert_called_once_with(SERVICE_ONE_ID)
|
||
assert response.status_code == 200
|
||
assert 'Get started' in response.get_data(as_text=True)
|
||
|
||
|
||
def test_get_started_is_hidden_once_templates_exist(
|
||
logged_in_client,
|
||
mocker,
|
||
mock_get_service_templates,
|
||
mock_get_jobs,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
mock_template_stats = mocker.patch('app.template_statistics_client.get_template_statistics_for_service',
|
||
return_value=copy.deepcopy(stub_template_stats))
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
# mock_get_service_templates.assert_called_once_with(SERVICE_ONE_ID)
|
||
assert response.status_code == 200
|
||
assert 'Get started' not in response.get_data(as_text=True)
|
||
|
||
|
||
def test_inbound_messages_not_visible_to_service_without_permissions(
|
||
logged_in_client,
|
||
service_one,
|
||
mock_get_service_templates_when_no_templates_exist,
|
||
mock_get_jobs,
|
||
mock_get_detailed_service,
|
||
mock_get_template_statistics,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
|
||
service_one['permissions'] = []
|
||
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
|
||
assert response.status_code == 200
|
||
assert not page.select('.big-number-meta-wrapper')
|
||
assert mock_get_inbound_sms_summary.called is False
|
||
|
||
|
||
@pytest.mark.parametrize('inbound_summary_mock, expected_text', [
|
||
(mock_get_inbound_sms_summary_with_no_messages, '0 text messages received'),
|
||
(mock_get_inbound_sms_summary, '99 text messages received latest message just now'),
|
||
])
|
||
def test_inbound_messages_shows_count_of_messages(
|
||
logged_in_client,
|
||
mocker,
|
||
service_one,
|
||
mock_get_service_templates_when_no_templates_exist,
|
||
mock_get_jobs,
|
||
mock_get_detailed_service,
|
||
mock_get_template_statistics,
|
||
mock_get_usage,
|
||
inbound_summary_mock,
|
||
expected_text,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
|
||
service_one['permissions'] = ['inbound_sms']
|
||
inbound_summary_mock(mocker)
|
||
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
|
||
assert response.status_code == 200
|
||
assert normalize_spaces(page.select('.big-number-meta-wrapper')[0].text) == expected_text
|
||
assert page.select('.big-number-meta-wrapper a')[0]['href'] == url_for(
|
||
'main.inbox', service_id=SERVICE_ONE_ID
|
||
)
|
||
|
||
|
||
@pytest.mark.parametrize('index, expected_row', enumerate([
|
||
'07900 900000 message-1 1 hour ago',
|
||
'07900 900002 message-4 3 hours ago',
|
||
'07900 900004 message-5 5 hours ago',
|
||
'07900 900006 message-6 7 hours ago',
|
||
'07900 900008 message-7 9 hours ago',
|
||
]))
|
||
def test_inbox_showing_inbound_messages(
|
||
logged_in_client,
|
||
service_one,
|
||
mock_get_service_templates_when_no_templates_exist,
|
||
mock_get_jobs,
|
||
mock_get_detailed_service,
|
||
mock_get_template_statistics,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms,
|
||
index,
|
||
expected_row,
|
||
):
|
||
|
||
service_one['permissions'] = ['inbound_sms']
|
||
|
||
response = logged_in_client.get(url_for('main.inbox', service_id=SERVICE_ONE_ID))
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
|
||
assert response.status_code == 200
|
||
rows = page.select('tbody tr')
|
||
assert len(rows) == 5
|
||
assert normalize_spaces(rows[index].text) == expected_row
|
||
assert normalize_spaces(page.select('.table-show-more-link')) == (
|
||
'8 messages from 5 users'
|
||
)
|
||
|
||
|
||
def test_empty_inbox(
|
||
logged_in_client,
|
||
service_one,
|
||
mock_get_service_templates_when_no_templates_exist,
|
||
mock_get_jobs,
|
||
mock_get_detailed_service,
|
||
mock_get_template_statistics,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_with_no_messages,
|
||
):
|
||
|
||
service_one['permissions'] = ['inbound_sms']
|
||
|
||
response = logged_in_client.get(url_for('main.inbox', service_id=SERVICE_ONE_ID))
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
|
||
assert response.status_code == 200
|
||
assert normalize_spaces(page.select('tbody tr')) == (
|
||
'When users text your service’s phone number (GOVUK) you’ll see the messages here'
|
||
)
|
||
|
||
|
||
def test_inbox_not_accessible_to_service_without_permissions(
|
||
logged_in_client,
|
||
service_one,
|
||
):
|
||
service_one['permissions'] = []
|
||
response = logged_in_client.get(url_for('main.inbox', service_id=SERVICE_ONE_ID))
|
||
|
||
assert response.status_code == 403
|
||
|
||
|
||
def test_should_show_recent_templates_on_dashboard(
|
||
logged_in_client,
|
||
mocker,
|
||
mock_get_service_templates,
|
||
mock_get_jobs,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
mock_template_stats = mocker.patch('app.template_statistics_client.get_template_statistics_for_service',
|
||
return_value=copy.deepcopy(stub_template_stats))
|
||
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
assert response.status_code == 200
|
||
response.get_data(as_text=True)
|
||
mock_template_stats.assert_called_once_with(SERVICE_ONE_ID, limit_days=7)
|
||
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
headers = [header.text.strip() for header in page.find_all('h2') + page.find_all('h1')]
|
||
assert 'In the last 7 days' in headers
|
||
|
||
table_rows = page.find_all('tbody')[1].find_all('tr')
|
||
|
||
assert len(table_rows) == 2
|
||
|
||
assert 'two' in table_rows[0].find_all('th')[0].text
|
||
assert 'Email template' in table_rows[0].find_all('th')[0].text
|
||
assert '200' in table_rows[0].find_all('td')[0].text
|
||
|
||
assert 'one' in table_rows[1].find_all('th')[0].text
|
||
assert 'Text message template' in table_rows[1].find_all('th')[0].text
|
||
assert '100' in table_rows[1].find_all('td')[0].text
|
||
|
||
|
||
@freeze_time("2016-07-01 12:00") # 4 months into 2016 financial year
|
||
@pytest.mark.parametrize('partial_url', [
|
||
partial(url_for),
|
||
partial(url_for, year='2016'),
|
||
])
|
||
def test_should_show_monthly_breakdown_of_template_usage(
|
||
logged_in_client,
|
||
mock_get_monthly_template_statistics,
|
||
partial_url,
|
||
):
|
||
response = logged_in_client.get(
|
||
partial_url('main.template_history', service_id=SERVICE_ONE_ID, _external=True)
|
||
)
|
||
|
||
assert response.status_code == 200
|
||
mock_get_monthly_template_statistics.assert_called_once_with(SERVICE_ONE_ID, 2016)
|
||
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
table_rows = page.select('tbody tr')
|
||
|
||
assert ' '.join(table_rows[0].text.split()) == (
|
||
'My first template '
|
||
'Text message template '
|
||
'2'
|
||
)
|
||
|
||
assert len(table_rows) == len(['April'])
|
||
assert len(page.select('.table-no-data')) == len(['May', 'June', 'July'])
|
||
|
||
|
||
@freeze_time("2016-01-01 11:09:00.061258")
|
||
def test_should_show_upcoming_jobs_on_dashboard(
|
||
logged_in_client,
|
||
mock_get_service_templates,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_jobs,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
first_call = mock_get_jobs.call_args_list[0]
|
||
assert first_call[0] == (SERVICE_ONE_ID,)
|
||
assert first_call[1]['statuses'] == ['scheduled']
|
||
|
||
assert response.status_code == 200
|
||
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
|
||
table_rows = page.find_all('tbody')[0].find_all('tr')
|
||
assert len(table_rows) == 2
|
||
|
||
assert 'send_me_later.csv' in table_rows[0].find_all('th')[0].text
|
||
assert 'Sending today at 11:09am' in table_rows[0].find_all('th')[0].text
|
||
assert table_rows[0].find_all('td')[0].text.strip() == '1'
|
||
assert 'even_later.csv' in table_rows[1].find_all('th')[0].text
|
||
assert 'Sending today at 11:09pm' in table_rows[1].find_all('th')[0].text
|
||
assert table_rows[1].find_all('td')[0].text.strip() == '1'
|
||
|
||
|
||
@freeze_time("2016-01-01 11:09:00.061258")
|
||
def test_should_show_recent_jobs_on_dashboard(
|
||
logged_in_client,
|
||
mock_get_service_templates,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_jobs,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
second_call = mock_get_jobs.call_args_list[1]
|
||
assert second_call[0] == (SERVICE_ONE_ID,)
|
||
assert second_call[1]['limit_days'] == 7
|
||
assert 'scheduled' not in second_call[1]['statuses']
|
||
|
||
assert response.status_code == 200
|
||
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
table_rows = page.find_all('tbody')[2].find_all('tr')
|
||
|
||
assert len(table_rows) == 4
|
||
|
||
for index, filename in enumerate((
|
||
"export 1/1/2016.xls",
|
||
"all email addresses.xlsx",
|
||
"applicants.ods",
|
||
"thisisatest.csv",
|
||
)):
|
||
assert filename in table_rows[index].find_all('th')[0].text
|
||
assert 'Sent 1 January at 11:09' in table_rows[index].find_all('th')[0].text
|
||
for column_index, count in enumerate((1, 0, 0)):
|
||
assert table_rows[index].find_all('td')[column_index].text.strip() == str(count)
|
||
|
||
|
||
@freeze_time("2012-03-31 12:12:12")
|
||
def test_usage_page(
|
||
logged_in_client,
|
||
mock_get_usage,
|
||
mock_get_billable_units,
|
||
):
|
||
response = logged_in_client.get(url_for('main.usage', service_id=SERVICE_ONE_ID))
|
||
|
||
assert response.status_code == 200
|
||
|
||
mock_get_billable_units.assert_called_once_with(SERVICE_ONE_ID, 2011)
|
||
mock_get_usage.assert_called_once_with(SERVICE_ONE_ID, 2011)
|
||
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
|
||
cols = page.find_all('div', {'class': 'column-half'})
|
||
nav = page.find('ul', {'class': 'pill', 'role': 'tablist'})
|
||
nav_links = nav.find_all('a')
|
||
|
||
assert normalize_spaces(nav_links[0].text) == '2010 to 2011 financial year'
|
||
assert normalize_spaces(nav.find('li', {'aria-selected': 'true'}).text) == '2011 to 2012 financial year'
|
||
assert normalize_spaces(nav_links[1].text) == '2012 to 2013 financial year'
|
||
assert '252,190' in cols[1].text
|
||
assert 'Text messages' in cols[1].text
|
||
|
||
table = page.find('table').text.strip()
|
||
|
||
assert '249,860 free text messages' in table
|
||
assert '40 free text messages' in table
|
||
assert '960 text messages at 1.65p' in table
|
||
|
||
assert 'April' in table
|
||
assert 'February' in table
|
||
assert 'March' in table
|
||
assert '£15.84' in table
|
||
assert '140 free text messages' in table
|
||
assert '£20.30' in table
|
||
assert '1,230 text messages at 1.65p' in table
|
||
|
||
|
||
def test_usage_page_with_year_argument(
|
||
logged_in_client,
|
||
mock_get_usage,
|
||
mock_get_billable_units
|
||
):
|
||
assert logged_in_client.get(url_for('main.usage', service_id=SERVICE_ONE_ID, year=2000)).status_code == 200
|
||
mock_get_billable_units.assert_called_once_with(SERVICE_ONE_ID, 2000)
|
||
mock_get_usage.assert_called_once_with(SERVICE_ONE_ID, 2000)
|
||
|
||
|
||
def test_usage_page_for_invalid_year(
|
||
logged_in_client,
|
||
):
|
||
assert logged_in_client.get(url_for('main.usage', service_id=SERVICE_ONE_ID, year='abcd')).status_code == 404
|
||
|
||
|
||
@freeze_time("2012-03-31 12:12:12")
|
||
def test_future_usage_page(
|
||
logged_in_client,
|
||
mock_get_future_usage,
|
||
mock_get_future_billable_units,
|
||
):
|
||
assert logged_in_client.get(url_for('main.usage', service_id=SERVICE_ONE_ID, year=2014)).status_code == 200
|
||
|
||
mock_get_future_billable_units.assert_called_once_with(SERVICE_ONE_ID, 2014)
|
||
mock_get_future_usage.assert_called_once_with(SERVICE_ONE_ID, 2014)
|
||
|
||
|
||
def _test_dashboard_menu(mocker, app_, usr, service, permissions):
|
||
with app_.test_request_context():
|
||
with app_.test_client() as client:
|
||
usr._permissions[str(service['id'])] = permissions
|
||
mocker.patch('app.user_api_client.check_verify_code', return_value=(True, ''))
|
||
mocker.patch('app.service_api_client.get_services', return_value={'data': [service]})
|
||
mocker.patch('app.user_api_client.get_user', return_value=usr)
|
||
mocker.patch('app.user_api_client.get_user_by_email', return_value=usr)
|
||
mocker.patch('app.service_api_client.get_service', return_value={'data': service})
|
||
client.login(usr)
|
||
return client.get(url_for('main.service_dashboard', service_id=service['id']))
|
||
|
||
|
||
def test_menu_send_messages(
|
||
mocker,
|
||
app_,
|
||
api_user_active,
|
||
service_one,
|
||
mock_get_service_templates,
|
||
mock_get_jobs,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
with app_.test_request_context():
|
||
resp = _test_dashboard_menu(
|
||
mocker,
|
||
app_,
|
||
api_user_active,
|
||
service_one,
|
||
['view_activity', 'send_texts', 'send_emails', 'send_letters'])
|
||
page = resp.get_data(as_text=True)
|
||
assert url_for(
|
||
'main.choose_template',
|
||
service_id=service_one['id'],
|
||
) in page
|
||
assert url_for('main.manage_users', service_id=service_one['id']) in page
|
||
|
||
assert url_for('main.service_settings', service_id=service_one['id']) not in page
|
||
assert url_for('main.api_keys', service_id=service_one['id']) not in page
|
||
assert url_for('main.view_providers') not in page
|
||
|
||
|
||
def test_menu_manage_service(
|
||
mocker,
|
||
app_,
|
||
api_user_active,
|
||
service_one,
|
||
mock_get_service_templates,
|
||
mock_get_jobs,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
with app_.test_request_context():
|
||
resp = _test_dashboard_menu(
|
||
mocker,
|
||
app_,
|
||
api_user_active,
|
||
service_one,
|
||
['view_activity', 'manage_users', 'manage_templates', 'manage_settings'])
|
||
page = resp.get_data(as_text=True)
|
||
assert url_for(
|
||
'main.choose_template',
|
||
service_id=service_one['id'],
|
||
) in page
|
||
assert url_for('main.manage_users', service_id=service_one['id']) in page
|
||
assert url_for('main.service_settings', service_id=service_one['id']) in page
|
||
|
||
assert url_for('main.api_keys', service_id=service_one['id']) not in page
|
||
|
||
|
||
def test_menu_manage_api_keys(
|
||
mocker,
|
||
app_,
|
||
api_user_active,
|
||
service_one,
|
||
mock_get_service_templates,
|
||
mock_get_jobs,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
with app_.test_request_context():
|
||
resp = _test_dashboard_menu(
|
||
mocker,
|
||
app_,
|
||
api_user_active,
|
||
service_one,
|
||
['view_activity', 'manage_api_keys'])
|
||
page = resp.get_data(as_text=True)
|
||
assert url_for(
|
||
'main.choose_template',
|
||
service_id=service_one['id'],
|
||
) in page
|
||
assert url_for('main.manage_users', service_id=service_one['id']) in page
|
||
assert url_for('main.service_settings', service_id=service_one['id']) not in page
|
||
|
||
assert url_for('main.api_integration', service_id=service_one['id']) in page
|
||
|
||
|
||
def test_menu_all_services_for_platform_admin_user(
|
||
mocker,
|
||
app_,
|
||
platform_admin_user,
|
||
service_one,
|
||
mock_get_service_templates,
|
||
mock_get_jobs,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
with app_.test_request_context():
|
||
resp = _test_dashboard_menu(
|
||
mocker,
|
||
app_,
|
||
platform_admin_user,
|
||
service_one,
|
||
[])
|
||
page = resp.get_data(as_text=True)
|
||
assert url_for('main.choose_template', service_id=service_one['id']) in page
|
||
assert url_for('main.manage_users', service_id=service_one['id']) in page
|
||
assert url_for('main.service_settings', service_id=service_one['id']) in page
|
||
assert url_for('main.view_notifications', service_id=service_one['id'], message_type='email') in page
|
||
assert url_for('main.view_notifications', service_id=service_one['id'], message_type='sms') in page
|
||
assert url_for('main.api_keys', service_id=service_one['id']) not in page
|
||
|
||
|
||
def test_route_for_service_permissions(
|
||
mocker,
|
||
app_,
|
||
api_user_active,
|
||
service_one,
|
||
mock_get_service,
|
||
mock_get_user,
|
||
mock_get_service_templates,
|
||
mock_get_jobs,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
with app_.test_request_context():
|
||
validate_route_permission(
|
||
mocker,
|
||
app_,
|
||
"GET",
|
||
200,
|
||
url_for('main.service_dashboard', service_id=service_one['id']),
|
||
['view_activity'],
|
||
api_user_active,
|
||
service_one)
|
||
|
||
|
||
def test_aggregate_template_stats():
|
||
from app.main.views.dashboard import aggregate_usage
|
||
expected = aggregate_usage(copy.deepcopy(stub_template_stats))
|
||
|
||
assert len(expected) == 2
|
||
assert expected[0]['template_name'] == 'two'
|
||
assert expected[0]['count'] == 200
|
||
assert expected[0]['template_id'] == 'id-2'
|
||
assert expected[0]['template_type'] == 'email'
|
||
assert expected[1]['template_name'] == 'one'
|
||
assert expected[1]['count'] == 100
|
||
assert expected[1]['template_id'] == 'id-1'
|
||
assert expected[1]['template_type'] == 'sms'
|
||
|
||
|
||
def test_service_dashboard_updates_gets_dashboard_totals(
|
||
mocker,
|
||
logged_in_client,
|
||
mock_get_service_templates,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_jobs,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
mocker.patch('app.main.views.dashboard.get_dashboard_totals', return_value={
|
||
'email': {'requested': 123, 'delivered': 0, 'failed': 0},
|
||
'sms': {'requested': 456, 'delivered': 0, 'failed': 0}
|
||
})
|
||
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
assert response.status_code == 200
|
||
|
||
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
|
||
numbers = [number.text.strip() for number in page.find_all('div', class_='big-number-number')]
|
||
assert '123' in numbers
|
||
assert '456' in numbers
|
||
|
||
|
||
def test_get_dashboard_totals_adds_percentages():
|
||
stats = {
|
||
'sms': {
|
||
'requested': 3,
|
||
'delivered': 0,
|
||
'failed': 2
|
||
},
|
||
'email': {
|
||
'requested': 0,
|
||
'delivered': 0,
|
||
'failed': 0
|
||
}
|
||
}
|
||
assert get_dashboard_totals(stats)['sms']['failed_percentage'] == '66.7'
|
||
assert get_dashboard_totals(stats)['email']['failed_percentage'] == '0'
|
||
|
||
|
||
@pytest.mark.parametrize(
|
||
'failures,expected', [
|
||
(2, False),
|
||
(3, False),
|
||
(4, True)
|
||
]
|
||
)
|
||
def test_get_dashboard_totals_adds_warning(failures, expected):
|
||
stats = {
|
||
'sms': {
|
||
'requested': 100,
|
||
'delivered': 0,
|
||
'failed': failures
|
||
}
|
||
}
|
||
assert get_dashboard_totals(stats)['sms']['show_warning'] == expected
|
||
|
||
|
||
def test_format_monthly_stats_empty_case():
|
||
assert format_monthly_stats_to_list({}) == []
|
||
|
||
|
||
def test_format_monthly_stats_labels_month():
|
||
resp = format_monthly_stats_to_list({'2016-07': {}})
|
||
assert resp[0]['name'] == 'July'
|
||
|
||
|
||
def test_format_monthly_stats_has_stats_with_failure_rate():
|
||
resp = format_monthly_stats_to_list({
|
||
'2016-07': {'sms': _stats(3, 1, 2)}
|
||
})
|
||
assert resp[0]['sms_counts'] == {
|
||
'failed': 2,
|
||
'failed_percentage': '66.7',
|
||
'requested': 3,
|
||
'show_warning': True,
|
||
}
|
||
|
||
|
||
def test_format_monthly_stats_works_for_email_letter():
|
||
resp = format_monthly_stats_to_list({
|
||
'2016-07': {
|
||
'sms': {},
|
||
'email': {},
|
||
'letter': {},
|
||
}
|
||
})
|
||
assert isinstance(resp[0]['sms_counts'], dict)
|
||
assert isinstance(resp[0]['email_counts'], dict)
|
||
assert isinstance(resp[0]['letter_counts'], dict)
|
||
|
||
|
||
def _stats(requested, delivered, failed):
|
||
return {'requested': requested, 'delivered': delivered, 'failed': failed}
|
||
|
||
|
||
@pytest.mark.parametrize('dict_in, expected_failed, expected_requested', [
|
||
(
|
||
{},
|
||
0,
|
||
0
|
||
),
|
||
(
|
||
{'temporary-failure': 1, 'permanent-failure': 1, 'technical-failure': 1},
|
||
3,
|
||
3,
|
||
),
|
||
(
|
||
{'created': 1, 'pending': 1, 'sending': 1, 'delivered': 1},
|
||
0,
|
||
4,
|
||
),
|
||
])
|
||
def test_aggregate_status_types(dict_in, expected_failed, expected_requested):
|
||
sms_counts = aggregate_status_types({'sms': dict_in})['sms_counts']
|
||
assert sms_counts['failed'] == expected_failed
|
||
assert sms_counts['requested'] == expected_requested
|
||
|
||
|
||
@pytest.mark.parametrize(
|
||
'now, expected_number_of_months', [
|
||
(freeze_time("2017-12-31 11:09:00.061258"), 12),
|
||
(freeze_time("2017-01-01 11:09:00.061258"), 10)
|
||
]
|
||
)
|
||
def test_get_free_paid_breakdown_for_billable_units(now, expected_number_of_months):
|
||
with now:
|
||
billing_units = get_free_paid_breakdown_for_billable_units(
|
||
2016, [
|
||
{
|
||
'month': 'April', 'international': False, 'rate_multiplier': 1,
|
||
'notification_type': 'sms', 'rate': 1.65, 'billing_units': 100000
|
||
},
|
||
{
|
||
'month': 'May', 'international': False, 'rate_multiplier': 1,
|
||
'notification_type': 'sms', 'rate': 1.65, 'billing_units': 100000
|
||
},
|
||
{
|
||
'month': 'June', 'international': False, 'rate_multiplier': 1,
|
||
'notification_type': 'sms', 'rate': 1.65, 'billing_units': 100000
|
||
},
|
||
{
|
||
'month': 'February', 'international': False, 'rate_multiplier': 1,
|
||
'notification_type': 'sms', 'rate': 1.65, 'billing_units': 2000
|
||
},
|
||
]
|
||
)
|
||
assert list(billing_units) == [
|
||
{'free': 100000, 'name': 'April', 'paid': 0},
|
||
{'free': 100000, 'name': 'May', 'paid': 0},
|
||
{'free': 50000, 'name': 'June', 'paid': 50000},
|
||
{'free': 0, 'name': 'July', 'paid': 0},
|
||
{'free': 0, 'name': 'August', 'paid': 0},
|
||
{'free': 0, 'name': 'September', 'paid': 0},
|
||
{'free': 0, 'name': 'October', 'paid': 0},
|
||
{'free': 0, 'name': 'November', 'paid': 0},
|
||
{'free': 0, 'name': 'December', 'paid': 0},
|
||
{'free': 0, 'name': 'January', 'paid': 0},
|
||
{'free': 0, 'name': 'February', 'paid': 2000},
|
||
{'free': 0, 'name': 'March', 'paid': 0}
|
||
][:expected_number_of_months]
|
||
|
||
|
||
def test_format_template_stats_to_list_with_no_stats():
|
||
assert list(format_template_stats_to_list({})) == []
|
||
|
||
|
||
def test_format_template_stats_to_list():
|
||
counts = {
|
||
'created': 1,
|
||
'pending': 1,
|
||
'delivered': 1,
|
||
'failed': 1,
|
||
'temporary-failure': 1,
|
||
'permanent-failure': 1,
|
||
'technical-failure': 1,
|
||
'do-not-count': 999,
|
||
}
|
||
stats_list = list(format_template_stats_to_list({
|
||
'template_2_id': {
|
||
'counts': {},
|
||
'name': 'bar',
|
||
},
|
||
'template_1_id': {
|
||
'counts': counts,
|
||
'name': 'foo',
|
||
},
|
||
}))
|
||
|
||
# we don’t care about the order of this function’s output
|
||
assert len(stats_list) == 2
|
||
assert {
|
||
'counts': counts,
|
||
'name': 'foo',
|
||
'requested_count': 7,
|
||
'id': 'template_1_id',
|
||
} in stats_list
|
||
assert {
|
||
'counts': {},
|
||
'name': 'bar',
|
||
'requested_count': 0,
|
||
'id': 'template_2_id',
|
||
} in stats_list
|
||
|
||
|
||
def test_get_tuples_of_financial_years():
|
||
assert list(get_tuples_of_financial_years(
|
||
lambda year: 'http://example.com?year={}'.format(year),
|
||
start=2040,
|
||
end=2041,
|
||
)) == [
|
||
('financial year', 2040, 'http://example.com?year=2040', '2040 to 2041'),
|
||
('financial year', 2041, 'http://example.com?year=2041', '2041 to 2042'),
|
||
]
|
||
|
||
|
||
def test_get_tuples_of_financial_years_defaults_to_2015():
|
||
assert 2015 in list(get_tuples_of_financial_years(
|
||
lambda year: 'http://example.com?year={}'.format(year),
|
||
end=2040,
|
||
))[0]
|
||
|
||
|
||
@freeze_time("2016-01-01 11:09:00.061258")
|
||
def test_should_show_all_jobs_with_valid_statuses(
|
||
logged_in_client,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_service_templates_when_no_templates_exist,
|
||
mock_get_jobs,
|
||
mock_get_usage,
|
||
mock_get_inbound_sms_summary,
|
||
mock_get_yearly_sms_unit_count_and_cost
|
||
):
|
||
logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
first_call = mock_get_jobs.call_args_list[0]
|
||
# first call - scheduled jobs only
|
||
assert first_call == call(ANY, statuses=['scheduled'])
|
||
# second call - everything but scheduled and cancelled
|
||
second_call = mock_get_jobs.call_args_list[1]
|
||
assert second_call == call(ANY, limit_days=ANY, statuses={
|
||
'pending',
|
||
'in progress',
|
||
'finished',
|
||
'sending limits exceeded',
|
||
'ready to send',
|
||
'sent to dvla'
|
||
})
|
||
|
||
|
||
def test_should_show_remaining_free_tier_count(
|
||
logged_in_client,
|
||
mock_get_service_templates,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_jobs,
|
||
mock_get_usage,
|
||
mocker
|
||
):
|
||
mocker.patch(
|
||
'app.service_api_client.get_yearly_sms_unit_count_and_cost',
|
||
return_value={"billable_sms_units": 100, "total_cost": 200.0}
|
||
)
|
||
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
assert response.status_code == 200
|
||
assert '249,900' in response.get_data(as_text=True)
|
||
assert 'free text messages left' in response.get_data(as_text=True)
|
||
|
||
|
||
def test_should_show_cost_if_exceeded_free_tier_count(
|
||
logged_in_client,
|
||
mock_get_service_templates,
|
||
mock_get_template_statistics,
|
||
mock_get_detailed_service,
|
||
mock_get_jobs,
|
||
mock_get_usage,
|
||
mocker
|
||
):
|
||
mocker.patch(
|
||
'app.service_api_client.get_yearly_sms_unit_count_and_cost',
|
||
return_value={"billable_sms_units": 300000, "total_cost": 1500.50}
|
||
)
|
||
|
||
response = logged_in_client.get(url_for('main.service_dashboard', service_id=SERVICE_ONE_ID))
|
||
|
||
assert response.status_code == 200
|
||
assert '£1,500.50' in response.get_data(as_text=True)
|
||
assert 'spent on text messages' in response.get_data(as_text=True)
|