Merge pull request #795 from alphagov/switch-providers-on-error

Auto-switch providers on error
This commit is contained in:
imdadahad
2017-01-24 10:58:50 +00:00
committed by GitHub
6 changed files with 254 additions and 18 deletions

View File

@@ -121,3 +121,30 @@ def test_should_technical_error_and_not_retry_if_invalid_email(sample_notificati
assert provider_tasks.deliver_email.retry.called is False
assert sample_notification.status == 'technical-failure'
def test_send_sms_should_switch_providers_on_provider_failure(sample_notification, mocker):
provider_to_use = mocker.patch('app.delivery.send_to_providers.provider_to_use')
provider_to_use.return_value.send_sms.side_effect = Exception('Error')
switch_provider_mock = mocker.patch('app.delivery.send_to_providers.dao_toggle_sms_provider')
mocker.patch('app.celery.provider_tasks.deliver_sms.retry')
deliver_sms(sample_notification.id)
assert switch_provider_mock.called is True
def test_send_sms_should_not_switch_providers_on_non_provider_failure(
sample_notification,
mocker
):
mocker.patch(
'app.delivery.send_to_providers.send_sms_to_provider',
side_effect=Exception("Non Provider Exception")
)
switch_provider_mock = mocker.patch('app.delivery.send_to_providers.dao_toggle_sms_provider')
mocker.patch('app.celery.provider_tasks.deliver_sms.retry')
deliver_sms(sample_notification.id)
assert switch_provider_mock.called is False

View File

@@ -1,9 +1,10 @@
import requests_mock
import pytest
import uuid
from datetime import (datetime, date, timedelta)
from sqlalchemy import asc
from sqlalchemy.orm.session import make_transient
import requests_mock
import pytest
from flask import current_app
from app import db
@@ -34,7 +35,11 @@ from app.dao.notifications_dao import dao_create_notification
from app.dao.invited_user_dao import save_invited_user
from app.dao.provider_rates_dao import create_provider_rates
from app.clients.sms.firetext import FiretextClient
from app.dao.provider_details_dao import (
dao_update_provider_details,
get_provider_details_by_identifier,
get_alternative_sms_provider
)
from tests.app.db import create_user
@@ -647,6 +652,15 @@ def fake_uuid():
return "6ce466d0-fd6a-11e5-82f5-e0accb9d11a6"
@pytest.fixture(scope='function')
def current_sms_provider():
return ProviderDetails.query.filter_by(
notification_type='sms'
).order_by(
asc(ProviderDetails.priority)
).first()
@pytest.fixture(scope='function')
def ses_provider():
return ProviderDetails.query.filter_by(identifier='ses').one()
@@ -654,7 +668,7 @@ def ses_provider():
@pytest.fixture(scope='function')
def firetext_provider():
return ProviderDetails.query.filter_by(identifier='mmg').one()
return ProviderDetails.query.filter_by(identifier='firetext').one()
@pytest.fixture(scope='function')

View File

@@ -1,3 +1,5 @@
import pytest
from datetime import datetime
from freezegun import freeze_time
@@ -5,12 +7,28 @@ from freezegun import freeze_time
from app.models import ProviderDetails, ProviderDetailsHistory
from app import clients
from app.dao.provider_details_dao import (
get_alternative_sms_provider,
get_current_provider,
get_provider_details,
get_provider_details_by_identifier,
get_provider_details_by_notification_type,
dao_switch_sms_provider_to_provider_with_identifier,
dao_toggle_sms_provider,
dao_update_provider_details
)
def set_primary_sms_provider(identifier='mmg'):
primary_provider = get_provider_details_by_identifier(identifier)
secondary_provider = get_alternative_sms_provider(identifier)
primary_provider.priority = 10
secondary_provider.priority = 20
dao_update_provider_details(primary_provider)
dao_update_provider_details(secondary_provider)
def test_can_get_all_providers(restore_provider_details):
assert len(get_provider_details()) == 4
@@ -21,15 +39,13 @@ def test_can_get_sms_providers(restore_provider_details):
assert all('sms' == prov.notification_type for prov in sms_providers)
def test_can_get_sms_providers_in_order(restore_provider_details):
def test_can_get_sms_providers_in_order_of_priority(restore_provider_details):
providers = get_provider_details_by_notification_type('sms')
assert providers[0].identifier == "mmg"
assert providers[1].identifier == "firetext"
assert providers[2].identifier == "loadtesting"
assert providers[0].priority < providers[1].priority < providers[2].priority
def test_can_get_email_providers_in_order(restore_provider_details):
def test_can_get_email_providers_in_order_of_priority(restore_provider_details):
providers = get_provider_details_by_notification_type('email')
assert providers[0].identifier == "ses"
@@ -78,3 +94,81 @@ def test_update_adds_history(restore_provider_details):
assert not ses_history[1].active
assert ses_history[1].version == 2
assert ses_history[1].updated_at == datetime(2000, 1, 1, 0, 0, 0)
def test_update_sms_provider_to_inactive_sets_inactive(restore_provider_details):
set_primary_sms_provider('mmg')
primary_provider = get_current_provider('sms')
primary_provider.active = False
dao_update_provider_details(primary_provider)
assert not primary_provider.active
def test_get_current_sms_provider_returns_correct_provider(restore_provider_details):
set_primary_sms_provider('mmg')
provider = get_current_provider('sms')
assert provider.identifier == 'mmg'
@pytest.mark.parametrize('provider_identifier', ['firetext', 'mmg'])
def test_get_alternative_sms_provider_returns_expected_provider(notify_db, provider_identifier):
provider = get_alternative_sms_provider(provider_identifier)
assert provider.identifier != provider
def test_switch_sms_provider_to_current_provider_does_not_switch(
restore_provider_details,
current_sms_provider,
mocker
):
dao_switch_sms_provider_to_provider_with_identifier(current_sms_provider.identifier)
new_provider = get_current_provider('sms')
assert current_sms_provider.id == new_provider.id
assert current_sms_provider.identifier == new_provider.identifier
def test_switch_sms_provider_to_inactive_provider_does_not_switch(
restore_provider_details,
current_sms_provider,
mocker
):
alternative_sms_provider = get_alternative_sms_provider(current_sms_provider.identifier)
alternative_sms_provider.active = False
dao_update_provider_details(alternative_sms_provider)
dao_switch_sms_provider_to_provider_with_identifier(alternative_sms_provider.identifier)
new_provider = get_current_provider('sms')
assert new_provider.id == current_sms_provider.id
assert new_provider.identifier == current_sms_provider.identifier
def test_toggle_sms_provider_switches_provider(
restore_provider_details,
current_sms_provider,
mocker
):
dao_toggle_sms_provider(current_sms_provider.identifier)
new_provider = get_current_provider('sms')
assert new_provider.identifier != current_sms_provider.identifier
assert new_provider.priority < current_sms_provider.priority
def test_toggle_sms_provider_switches_when_provider_priorities_are_equal(
restore_provider_details,
current_sms_provider,
mocker
):
new_provider = get_alternative_sms_provider(current_sms_provider.identifier)
current_sms_provider.priority = new_provider.priority
dao_update_provider_details(current_sms_provider)
dao_toggle_sms_provider(current_sms_provider.identifier)
assert new_provider.identifier != current_sms_provider.identifier
assert new_provider.priority < current_sms_provider.priority