mirror of
https://github.com/GSA/notifications-admin.git
synced 2026-05-06 00:48:46 -04:00
331 lines
13 KiB
Python
331 lines
13 KiB
Python
from unittest.mock import Mock
|
|
|
|
import pytest
|
|
from flask import url_for
|
|
from requests import RequestException
|
|
|
|
from app.utils import normalize_spaces
|
|
from tests.conftest import SERVICE_ONE_ID
|
|
|
|
|
|
def test_get_upload_hub_page(client_request):
|
|
page = client_request.get('main.uploads', service_id=SERVICE_ONE_ID)
|
|
|
|
assert page.find('h1').text == 'Uploads'
|
|
assert page.find('a', text='Upload a letter').attrs['href'] == url_for(
|
|
'main.upload_letter', service_id=SERVICE_ONE_ID
|
|
)
|
|
|
|
|
|
def test_get_upload_letter(client_request):
|
|
page = client_request.get('main.upload_letter', service_id=SERVICE_ONE_ID)
|
|
|
|
assert page.find('h1').text == 'Upload a letter'
|
|
assert page.find('input', class_='file-upload-field')
|
|
assert page.select('button[type=submit]')
|
|
|
|
|
|
def test_post_upload_letter_redirects_for_valid_file(mocker, client_request):
|
|
mocker.patch('uuid.uuid4', return_value='fake-uuid')
|
|
antivirus_mock = mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=True)
|
|
mocker.patch(
|
|
'app.main.views.uploads.sanitise_letter',
|
|
return_value=Mock(content='The sanitised content', json=lambda: {'file': 'VGhlIHNhbml0aXNlZCBjb250ZW50'})
|
|
)
|
|
mock_s3 = mocker.patch('app.main.views.uploads.upload_letter_to_s3')
|
|
mocker.patch('app.main.views.uploads.service_api_client.get_precompiled_template')
|
|
|
|
with open('tests/test_pdf_files/one_page_pdf.pdf', 'rb') as file:
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_follow_redirects=True,
|
|
)
|
|
assert antivirus_mock.called
|
|
|
|
mock_s3.assert_called_once_with(
|
|
b'The sanitised content',
|
|
'service-{}/fake-uuid.pdf'.format(SERVICE_ONE_ID),
|
|
'valid',
|
|
)
|
|
|
|
assert page.find('h1').text == 'tests/test_pdf_files/one_page_pdf.pdf'
|
|
assert not page.find(id='validation-error-message')
|
|
|
|
assert page.find('input', {'type': 'hidden', 'name': 'filename', 'value': 'tests/test_pdf_files/one_page_pdf.pdf'})
|
|
assert page.find('input', {'type': 'hidden', 'name': 'file_id', 'value': 'fake-uuid'})
|
|
assert page.find('button', {'type': 'submit'}).text == 'Send 1 letter'
|
|
|
|
|
|
def test_post_upload_letter_shows_letter_preview_for_valid_file(mocker, client_request):
|
|
letter_template = {'template_type': 'letter',
|
|
'reply_to_text': '',
|
|
'postage': 'second',
|
|
'subject': 'hi',
|
|
'content': 'my letter'}
|
|
|
|
mocker.patch('uuid.uuid4', return_value='fake-uuid')
|
|
mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=True)
|
|
mocker.patch(
|
|
'app.main.views.uploads.sanitise_letter',
|
|
return_value=Mock(content='The sanitised content', json=lambda: {'file': 'VGhlIHNhbml0aXNlZCBjb250ZW50'})
|
|
)
|
|
mocker.patch('app.main.views.uploads.upload_letter_to_s3')
|
|
mocker.patch('app.main.views.uploads.pdf_page_count', return_value=3)
|
|
mocker.patch('app.main.views.uploads.service_api_client.get_precompiled_template', return_value=letter_template)
|
|
|
|
with open('tests/test_pdf_files/one_page_pdf.pdf', 'rb') as file:
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_follow_redirects=True,
|
|
)
|
|
|
|
assert len(page.select('.letter-postage')) == 1
|
|
assert normalize_spaces(page.select_one('.letter-postage').text) == ('Postage: second class')
|
|
assert page.select_one('.letter-postage')['class'] == ['letter-postage', 'letter-postage-second']
|
|
|
|
letter_images = page.select('main img')
|
|
assert len(letter_images) == 3
|
|
|
|
for page_no, img in enumerate(letter_images, start=1):
|
|
assert img['src'] == url_for(
|
|
'.view_letter_upload_as_preview',
|
|
service_id=SERVICE_ONE_ID,
|
|
file_id='fake-uuid',
|
|
page=page_no)
|
|
|
|
|
|
def test_post_upload_letter_shows_error_when_file_is_not_a_pdf(client_request):
|
|
with open('tests/non_spreadsheet_files/actually_a_png.csv', 'rb') as file:
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_expected_status=200
|
|
)
|
|
assert page.find('span', class_='error-message').text.strip() == "Letters must be saved as a PDF"
|
|
|
|
|
|
def test_post_upload_letter_shows_error_when_no_file_uploaded(client_request):
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': ''},
|
|
_expected_status=200
|
|
)
|
|
assert page.find('span', class_='error-message').text.strip() == "You need to upload a file to submit"
|
|
|
|
|
|
def test_post_upload_letter_shows_error_when_file_contains_virus(mocker, client_request):
|
|
mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=False)
|
|
|
|
with open('tests/test_pdf_files/one_page_pdf.pdf', 'rb') as file:
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_expected_status=400
|
|
)
|
|
assert page.find('h1').text == 'Upload a letter'
|
|
assert normalize_spaces(page.select('.banner-dangerous')[0].text) == 'Your file has failed the virus check'
|
|
|
|
|
|
def test_post_choose_upload_file_when_file_is_too_big(mocker, client_request):
|
|
mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=True)
|
|
|
|
with open('tests/test_pdf_files/big.pdf', 'rb') as file:
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_expected_status=400
|
|
)
|
|
assert page.find('h1').text == 'Upload a letter'
|
|
assert normalize_spaces(page.select('.banner-dangerous')[0].text) == 'Your file must be smaller than 2MB'
|
|
|
|
|
|
def test_post_choose_upload_file_when_file_is_malformed(mocker, client_request):
|
|
mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=True)
|
|
|
|
with open('tests/test_pdf_files/no_eof_marker.pdf', 'rb') as file:
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_expected_status=400
|
|
)
|
|
assert page.find('h1').text == 'Upload a letter'
|
|
assert normalize_spaces(page.select('.banner-dangerous')[0].text) == 'Your file must be a valid PDF'
|
|
|
|
|
|
def test_post_upload_letter_with_invalid_file(mocker, client_request):
|
|
mocker.patch('uuid.uuid4', return_value='fake-uuid')
|
|
mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=True)
|
|
mock_s3 = mocker.patch('app.main.views.uploads.upload_letter_to_s3')
|
|
|
|
mock_sanitise_response = Mock()
|
|
mock_sanitise_response.raise_for_status.side_effect = RequestException(response=Mock(status_code=400))
|
|
mocker.patch('app.main.views.uploads.sanitise_letter', return_value=mock_sanitise_response)
|
|
mocker.patch('app.main.views.uploads.service_api_client.get_precompiled_template')
|
|
|
|
with open('tests/test_pdf_files/one_page_pdf.pdf', 'rb') as file:
|
|
file_contents = file.read()
|
|
file.seek(0)
|
|
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_follow_redirects=True
|
|
)
|
|
|
|
mock_s3.assert_called_once_with(
|
|
file_contents,
|
|
'service-{}/fake-uuid.pdf'.format(SERVICE_ONE_ID),
|
|
'invalid',
|
|
)
|
|
|
|
assert page.find('h1').text == 'tests/test_pdf_files/one_page_pdf.pdf'
|
|
assert normalize_spaces(
|
|
page.find(id='validation-error-message').text
|
|
) == 'Validation failed'
|
|
assert not page.find('button', {'type': 'submit'})
|
|
|
|
|
|
def test_post_upload_letter_shows_letter_preview_for_invalid_file(mocker, client_request):
|
|
letter_template = {'template_type': 'letter',
|
|
'reply_to_text': '',
|
|
'postage': 'first',
|
|
'subject': 'hi',
|
|
'content': 'my letter'}
|
|
|
|
mocker.patch('uuid.uuid4', return_value='fake-uuid')
|
|
mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=True)
|
|
mocker.patch('app.main.views.uploads.upload_letter_to_s3')
|
|
mock_sanitise_response = Mock()
|
|
mock_sanitise_response.raise_for_status.side_effect = RequestException(response=Mock(status_code=400))
|
|
mocker.patch('app.main.views.uploads.sanitise_letter', return_value=mock_sanitise_response)
|
|
mocker.patch('app.main.views.uploads.service_api_client.get_precompiled_template', return_value=letter_template)
|
|
|
|
with open('tests/test_pdf_files/one_page_pdf.pdf', 'rb') as file:
|
|
page = client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_follow_redirects=True,
|
|
)
|
|
|
|
assert len(page.select('.letter-postage')) == 1
|
|
assert normalize_spaces(page.select_one('.letter-postage').text) == ('Postage: first class')
|
|
assert page.select_one('.letter-postage')['class'] == ['letter-postage', 'letter-postage-first']
|
|
|
|
letter_images = page.select('main img')
|
|
assert len(letter_images) == 1
|
|
assert letter_images[0]['src'] == url_for(
|
|
'.view_letter_upload_as_preview',
|
|
service_id=SERVICE_ONE_ID,
|
|
file_id='fake-uuid',
|
|
page=1
|
|
)
|
|
|
|
|
|
def test_post_upload_letter_does_not_upload_to_s3_if_template_preview_raises_unknown_error(mocker, client_request):
|
|
mocker.patch('uuid.uuid4', return_value='fake-uuid')
|
|
mocker.patch('app.main.views.uploads.antivirus_client.scan', return_value=True)
|
|
mock_s3 = mocker.patch('app.main.views.uploads.upload_letter_to_s3')
|
|
|
|
mocker.patch('app.main.views.uploads.sanitise_letter', side_effect=RequestException())
|
|
|
|
with pytest.raises(RequestException):
|
|
with open('tests/test_pdf_files/one_page_pdf.pdf', 'rb') as file:
|
|
client_request.post(
|
|
'main.upload_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'file': file},
|
|
_follow_redirects=True
|
|
)
|
|
|
|
assert not mock_s3.called
|
|
|
|
|
|
def test_uploaded_letter_preview(mocker, client_request):
|
|
mocker.patch('app.main.views.uploads.service_api_client')
|
|
|
|
page = client_request.get(
|
|
'main.uploaded_letter_preview',
|
|
service_id=SERVICE_ONE_ID,
|
|
file_id='fake-uuid',
|
|
original_filename='my_letter.pdf',
|
|
page_count=1,
|
|
status='valid',
|
|
)
|
|
|
|
assert page.find('h1').text == 'my_letter.pdf'
|
|
assert page.find('div', class_='letter-sent')
|
|
|
|
|
|
def test_send_uploaded_letter_sends_letter_and_redirects_to_notification_page(mocker, service_one, client_request):
|
|
mocker.patch('app.main.views.uploads.get_letter_pdf_and_metadata', return_value=('file', {'status': 'valid'}))
|
|
mock_send = mocker.patch('app.main.views.uploads.notification_api_client.send_precompiled_letter')
|
|
|
|
service_one['permissions'] = ['letter', 'upload_letters']
|
|
file_id = 'abcd-1234'
|
|
|
|
client_request.post(
|
|
'main.send_uploaded_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'filename': 'my_file.pdf', 'file_id': file_id},
|
|
_expected_redirect=url_for(
|
|
'main.view_notification',
|
|
service_id=SERVICE_ONE_ID,
|
|
notification_id=file_id,
|
|
_external=True
|
|
)
|
|
)
|
|
mock_send.assert_called_once_with(SERVICE_ONE_ID, 'my_file.pdf', file_id)
|
|
|
|
|
|
@pytest.mark.parametrize('permissions', [
|
|
['email'],
|
|
['letter'],
|
|
['upload_letters'],
|
|
])
|
|
def test_send_uploaded_letter_when_service_does_not_have_correct_permissions(
|
|
mocker,
|
|
service_one,
|
|
client_request,
|
|
permissions,
|
|
):
|
|
mocker.patch('app.main.views.uploads.get_letter_pdf_and_metadata', return_value=('file', {'status': 'valid'}))
|
|
mock_send = mocker.patch('app.main.views.uploads.notification_api_client.send_precompiled_letter')
|
|
|
|
service_one['permissions'] = permissions
|
|
file_id = 'abcd-1234'
|
|
|
|
client_request.post(
|
|
'main.send_uploaded_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'filename': 'my_file.pdf', 'file_id': file_id},
|
|
_expected_status=403
|
|
)
|
|
assert not mock_send.called
|
|
|
|
|
|
def test_send_uploaded_letter_when_metadata_states_pdf_is_invalid(mocker, service_one, client_request):
|
|
mocker.patch('app.main.views.uploads.get_letter_pdf_and_metadata', return_value=('file', {'status': 'invalid'}))
|
|
mock_send = mocker.patch('app.main.views.uploads.notification_api_client.send_precompiled_letter')
|
|
|
|
service_one['permissions'] = ['letter', 'upload_letters']
|
|
file_id = 'abcd-1234'
|
|
|
|
client_request.post(
|
|
'main.send_uploaded_letter',
|
|
service_id=SERVICE_ONE_ID,
|
|
_data={'filename': 'my_file.pdf', 'file_id': file_id},
|
|
_expected_status=403
|
|
)
|
|
assert not mock_send.called
|