mirror of
https://github.com/GSA/notifications-admin.git
synced 2026-02-24 04:10:57 -05:00
Handle files that can’t be interpreted as spreadsheets
There shouldn’t be a case where we see a `ValueError` on upload any more. Our file handling should be robust enough to deal with whatever is thrown at it. This commit: - adds test files with bad data (PNG files with their extensions changed to look like spreadsheets) - catches whatever exceptions are raised by trying to parse these files - returns a helpful flash message to the user Anything else should raise a `500`, eg if the file can’t be uploaded to S3.
This commit is contained in:
@@ -4,6 +4,8 @@ import json
|
||||
import uuid
|
||||
import itertools
|
||||
from contextlib import suppress
|
||||
from zipfile import BadZipFile
|
||||
from xlrd.biffh import XLRDError
|
||||
|
||||
from flask import (
|
||||
request,
|
||||
@@ -123,10 +125,10 @@ def send_messages(service_id, template_id):
|
||||
service_id=service_id,
|
||||
upload_id=upload_id,
|
||||
template_type=template.template_type))
|
||||
except ValueError as e:
|
||||
flash('There was a problem uploading: {}'.format(form.file.data.filename))
|
||||
flash(str(e))
|
||||
return redirect(url_for('.send_messages', service_id=service_id, template_id=template_id))
|
||||
except (UnicodeDecodeError, BadZipFile, XLRDError):
|
||||
flash('Couldn’t read {}. Try using a different file format.'.format(
|
||||
form.file.data.filename
|
||||
))
|
||||
|
||||
return render_template(
|
||||
'views/send.html',
|
||||
|
||||
@@ -1,5 +1,6 @@
|
||||
import pytest
|
||||
import re
|
||||
from itertools import repeat
|
||||
from io import BytesIO
|
||||
from os import path
|
||||
from glob import glob
|
||||
@@ -12,15 +13,26 @@ template_types = ['email', 'sms']
|
||||
|
||||
# The * ignores hidden files, eg .DS_Store
|
||||
test_spreadsheet_files = glob(path.join('tests', 'spreadsheet_files', '*'))
|
||||
test_non_spreadsheet_files = glob(path.join('tests', 'non_spreadsheet_files', '*'))
|
||||
|
||||
|
||||
def test_that_test_files_exist():
|
||||
assert len(test_spreadsheet_files) == 8
|
||||
assert len(test_non_spreadsheet_files) == 6
|
||||
|
||||
|
||||
@pytest.mark.parametrize("filename", test_spreadsheet_files)
|
||||
@pytest.mark.parametrize(
|
||||
"filename, acceptable_file",
|
||||
list(zip(
|
||||
test_spreadsheet_files, repeat(True)
|
||||
)) +
|
||||
list(zip(
|
||||
test_non_spreadsheet_files, repeat(False)
|
||||
))
|
||||
)
|
||||
def test_upload_files_in_different_formats(
|
||||
filename,
|
||||
acceptable_file,
|
||||
app_,
|
||||
api_user_active,
|
||||
mocker,
|
||||
@@ -34,18 +46,24 @@ def test_upload_files_in_different_formats(
|
||||
|
||||
with app_.test_request_context(), app_.test_client() as client, open(filename, 'rb') as uploaded:
|
||||
client.login(api_user_active)
|
||||
client.post(
|
||||
response = client.post(
|
||||
url_for('main.send_messages', service_id=fake_uuid, template_id=fake_uuid),
|
||||
data={'file': (BytesIO(uploaded.read()), filename)},
|
||||
content_type='multipart/form-data'
|
||||
)
|
||||
|
||||
assert mock_s3_upload.call_args[0][2]['data'].strip() == (
|
||||
"phone number,name,favourite colour,fruit\r\n"
|
||||
"07739 468 050,Pete,Coral,tomato\r\n"
|
||||
"07527 125 974,Not Pete,Magenta,Avacado\r\n"
|
||||
"07512 058 823,Still Not Pete,Crimson,Pear"
|
||||
)
|
||||
if acceptable_file:
|
||||
assert mock_s3_upload.call_args[0][2]['data'].strip() == (
|
||||
"phone number,name,favourite colour,fruit\r\n"
|
||||
"07739 468 050,Pete,Coral,tomato\r\n"
|
||||
"07527 125 974,Not Pete,Magenta,Avacado\r\n"
|
||||
"07512 058 823,Still Not Pete,Crimson,Pear"
|
||||
)
|
||||
else:
|
||||
assert not mock_s3_upload.called
|
||||
assert (
|
||||
'Couldn’t read {}. Try using a different file format.'.format(filename)
|
||||
) in response.get_data(as_text=True)
|
||||
|
||||
|
||||
def test_upload_csvfile_with_errors_shows_check_page_with_errors(
|
||||
@@ -100,20 +118,18 @@ def test_upload_csv_invalid_extension(app_,
|
||||
mock_get_users_by_service,
|
||||
mock_get_service_statistics,
|
||||
fake_uuid):
|
||||
contents = u'phone number,name\n+44 123,test1\n+44 456,test2'
|
||||
with app_.test_request_context():
|
||||
filename = 'invalid.txt'
|
||||
with app_.test_client() as client:
|
||||
client.login(api_user_active)
|
||||
resp = client.post(
|
||||
url_for('main.send_messages', service_id=fake_uuid, template_id=fake_uuid),
|
||||
data={'file': (BytesIO(contents.encode('utf-8')), filename)},
|
||||
data={'file': (BytesIO('contents'.encode('utf-8')), 'invalid.txt')},
|
||||
content_type='multipart/form-data',
|
||||
follow_redirects=True
|
||||
)
|
||||
|
||||
assert resp.status_code == 200
|
||||
assert "{} isn’t a spreadsheet that Notify can read".format(filename) in resp.get_data(as_text=True)
|
||||
assert "invalid.txt isn’t a spreadsheet that Notify can read" in resp.get_data(as_text=True)
|
||||
|
||||
|
||||
def test_send_test_sms_message(
|
||||
|
||||
BIN
tests/non_spreadsheet_files/actually_a_png.csv
Normal file
BIN
tests/non_spreadsheet_files/actually_a_png.csv
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 43 KiB |
BIN
tests/non_spreadsheet_files/actually_a_png.ods
Normal file
BIN
tests/non_spreadsheet_files/actually_a_png.ods
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 43 KiB |
BIN
tests/non_spreadsheet_files/actually_a_png.tsv
Normal file
BIN
tests/non_spreadsheet_files/actually_a_png.tsv
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 43 KiB |
BIN
tests/non_spreadsheet_files/actually_a_png.xls
Normal file
BIN
tests/non_spreadsheet_files/actually_a_png.xls
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 43 KiB |
BIN
tests/non_spreadsheet_files/actually_a_png.xlsm
Normal file
BIN
tests/non_spreadsheet_files/actually_a_png.xlsm
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 43 KiB |
BIN
tests/non_spreadsheet_files/actually_a_png.xlsx
Normal file
BIN
tests/non_spreadsheet_files/actually_a_png.xlsx
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 43 KiB |
Reference in New Issue
Block a user