mirror of
https://github.com/GSA/notifications-api.git
synced 2026-06-20 21:20:31 -04:00
@@ -2,6 +2,7 @@ import csv
|
||||
import datetime
|
||||
import re
|
||||
import time
|
||||
import urllib
|
||||
from io import StringIO
|
||||
|
||||
import botocore
|
||||
@@ -613,3 +614,44 @@ def remove_csv_object(object_key):
|
||||
current_app.config["CSV_UPLOAD_BUCKET"]["region"],
|
||||
)
|
||||
return obj.delete()
|
||||
|
||||
|
||||
def s3upload(
|
||||
filedata,
|
||||
region,
|
||||
bucket_name,
|
||||
file_location,
|
||||
content_type="binary/octet-stream",
|
||||
tags=None,
|
||||
metadata=None,
|
||||
):
|
||||
_s3 = get_s3_resource()
|
||||
|
||||
key = _s3.Object(bucket_name, file_location)
|
||||
|
||||
put_args = {
|
||||
"Body": filedata,
|
||||
"ServerSideEncryption": "AES256",
|
||||
"ContentType": content_type,
|
||||
}
|
||||
|
||||
if tags:
|
||||
tags = urllib.parse.urlencode(tags)
|
||||
put_args["Tagging"] = tags
|
||||
|
||||
if metadata:
|
||||
metadata = put_args["Metadata"] = metadata
|
||||
|
||||
try:
|
||||
current_app.logger.info(hilite(f"Going to try to upload this {key}"))
|
||||
key.put(**put_args)
|
||||
except botocore.exceptions.NoCredentialsError as e:
|
||||
current_app.logger.exception(
|
||||
f"Unable to upload {key} to S3 bucket because of {e}"
|
||||
)
|
||||
raise e
|
||||
except botocore.exceptions.ClientError as e:
|
||||
current_app.logger.exception(
|
||||
f"Unable to upload {key}to S3 bucket because of {e}"
|
||||
)
|
||||
raise e
|
||||
|
||||
@@ -36,7 +36,6 @@ from app.serialised_models import SerialisedService, SerialisedTemplate
|
||||
from app.service.utils import service_allowed_to_send_to
|
||||
from app.utils import DATETIME_FORMAT, hilite, utc_now
|
||||
from notifications_utils.recipients import RecipientCSV
|
||||
from notifications_utils.s3 import s3upload
|
||||
|
||||
|
||||
@notify_celery.task(name="process-job")
|
||||
@@ -640,7 +639,7 @@ def _generate_notifications_report(service_id, report_id, limit_days):
|
||||
# Delete yesterday's version of this report
|
||||
s3.delete_s3_object(file_location)
|
||||
|
||||
s3upload(
|
||||
s3.s3upload(
|
||||
filedata=csv_bytes,
|
||||
region=region,
|
||||
bucket_name=bucket_name,
|
||||
@@ -654,7 +653,6 @@ def generate_notification_reports_task():
|
||||
services = dao_fetch_all_services(only_active=True)
|
||||
for service in services:
|
||||
|
||||
current_app.logger.debug(hilite("INVOKE APPLY_ASYNC"))
|
||||
limit_days = [1, 3, 5, 7]
|
||||
for limit_day in limit_days:
|
||||
|
||||
|
||||
@@ -289,7 +289,7 @@ class Config(object):
|
||||
},
|
||||
"generate-notifications-reports": {
|
||||
"task": "generate-notifications-reports",
|
||||
"schedule": crontab(hour=1, minute=0),
|
||||
"schedule": crontab(minute="*/2"),
|
||||
"options": {"queue": QueueNames.PERIODIC},
|
||||
},
|
||||
"regenerate-job-cache-on-startup": {
|
||||
|
||||
@@ -1,12 +1,8 @@
|
||||
import urllib
|
||||
|
||||
import botocore
|
||||
from boto3 import Session
|
||||
from botocore.config import Config
|
||||
from flask import current_app
|
||||
|
||||
from app.config import _s3_credentials_from_env
|
||||
|
||||
AWS_CLIENT_CONFIG = Config(
|
||||
# This config is required to enable S3 to connect to FIPS-enabled
|
||||
# endpoints. See https://aws.amazon.com/compliance/fips/ for more
|
||||
@@ -21,56 +17,16 @@ default_regions = "us-gov-west-1"
|
||||
|
||||
|
||||
def get_s3_resource():
|
||||
|
||||
credentials = _s3_credentials_from_env("CSV")
|
||||
access_key = current_app.config["CSV_UPLOAD_BUCKET"]["access_key_id"]
|
||||
secret_key = current_app.config["CSV_UPLOAD_BUCKET"]["secret_access_key"]
|
||||
region = current_app.config["CSV_UPLOAD_BUCKET"]["region"]
|
||||
session = Session(
|
||||
aws_access_key_id=credentials["access_key_id"],
|
||||
aws_secret_access_key=credentials["secret_access_key"],
|
||||
region_name=credentials["region"],
|
||||
aws_access_key_id=access_key,
|
||||
aws_secret_access_key=secret_key,
|
||||
region_name=region,
|
||||
)
|
||||
noti_s3_resource = session.resource("s3", config=AWS_CLIENT_CONFIG)
|
||||
return noti_s3_resource
|
||||
|
||||
|
||||
def s3upload(
|
||||
filedata,
|
||||
region,
|
||||
bucket_name,
|
||||
file_location,
|
||||
content_type="binary/octet-stream",
|
||||
tags=None,
|
||||
metadata=None,
|
||||
):
|
||||
_s3 = get_s3_resource()
|
||||
|
||||
key = _s3.Object(bucket_name, file_location)
|
||||
|
||||
put_args = {
|
||||
"Body": filedata,
|
||||
"ServerSideEncryption": "AES256",
|
||||
"ContentType": content_type,
|
||||
}
|
||||
|
||||
if tags:
|
||||
tags = urllib.parse.urlencode(tags)
|
||||
put_args["Tagging"] = tags
|
||||
|
||||
if metadata:
|
||||
metadata = put_args["Metadata"] = metadata
|
||||
|
||||
try:
|
||||
current_app.logger.info(f"Going to try to upload this {key}")
|
||||
key.put(**put_args)
|
||||
except botocore.exceptions.NoCredentialsError as e:
|
||||
current_app.logger.exception(
|
||||
f"Unable to upload {key} to S3 bucket because of {e}"
|
||||
)
|
||||
raise e
|
||||
except botocore.exceptions.ClientError as e:
|
||||
current_app.logger.exception(
|
||||
f"Unable to upload {key}to S3 bucket because of {e}"
|
||||
)
|
||||
raise e
|
||||
s3_resource = session.resource("s3", config=AWS_CLIENT_CONFIG)
|
||||
return s3_resource
|
||||
|
||||
|
||||
class S3ObjectNotFound(botocore.exceptions.ClientError):
|
||||
|
||||
@@ -1,17 +1,18 @@
|
||||
from unittest.mock import MagicMock
|
||||
from urllib.parse import parse_qs
|
||||
|
||||
import botocore
|
||||
import pytest
|
||||
|
||||
from notifications_utils.s3 import (
|
||||
from notifications_utils.s3 import ( # s3upload,
|
||||
AWS_CLIENT_CONFIG,
|
||||
S3ObjectNotFound,
|
||||
get_s3_resource,
|
||||
s3download,
|
||||
s3upload,
|
||||
)
|
||||
|
||||
# from urllib.parse import parse_qs
|
||||
|
||||
|
||||
contents = "some file data"
|
||||
region = "eu-west-1"
|
||||
bucket = "some_bucket"
|
||||
@@ -19,102 +20,102 @@ location = "some_file_location"
|
||||
content_type = "binary/octet-stream"
|
||||
|
||||
|
||||
def test_s3upload_save_file_to_bucket(mocker):
|
||||
# def test_s3upload_save_file_to_bucket(mocker):
|
||||
|
||||
mock_s3_resource = mocker.Mock()
|
||||
mocked = mocker.patch(
|
||||
"notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
)
|
||||
s3upload(
|
||||
filedata=contents, region=region, bucket_name=bucket, file_location=location
|
||||
)
|
||||
mocked_put = mocked.return_value.Object.return_value.put
|
||||
mocked_put.assert_called_once_with(
|
||||
Body=contents,
|
||||
ServerSideEncryption="AES256",
|
||||
ContentType=content_type,
|
||||
)
|
||||
# mock_s3_resource = mocker.Mock()
|
||||
# mocked = mocker.patch(
|
||||
# "notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
# )
|
||||
# s3upload(
|
||||
# filedata=contents, region=region, bucket_name=bucket, file_location=location
|
||||
# )
|
||||
# mocked_put = mocked.return_value.Object.return_value.put
|
||||
# mocked_put.assert_called_once_with(
|
||||
# Body=contents,
|
||||
# ServerSideEncryption="AES256",
|
||||
# ContentType=content_type,
|
||||
# )
|
||||
|
||||
|
||||
def test_s3upload_save_file_to_bucket_with_contenttype(mocker):
|
||||
content_type = "image/png"
|
||||
# def test_s3upload_save_file_to_bucket_with_contenttype(mocker):
|
||||
# content_type = "image/png"
|
||||
|
||||
mock_s3_resource = mocker.Mock()
|
||||
mocked = mocker.patch(
|
||||
"notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
)
|
||||
s3upload(
|
||||
filedata=contents,
|
||||
region=region,
|
||||
bucket_name=bucket,
|
||||
file_location=location,
|
||||
content_type=content_type,
|
||||
)
|
||||
mocked_put = mocked.return_value.Object.return_value.put
|
||||
mocked_put.assert_called_once_with(
|
||||
Body=contents,
|
||||
ServerSideEncryption="AES256",
|
||||
ContentType=content_type,
|
||||
)
|
||||
# mock_s3_resource = mocker.Mock()
|
||||
# mocked = mocker.patch(
|
||||
# "notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
# )
|
||||
# s3upload(
|
||||
# filedata=contents,
|
||||
# region=region,
|
||||
# bucket_name=bucket,
|
||||
# file_location=location,
|
||||
# content_type=content_type,
|
||||
# )
|
||||
# mocked_put = mocked.return_value.Object.return_value.put
|
||||
# mocked_put.assert_called_once_with(
|
||||
# Body=contents,
|
||||
# ServerSideEncryption="AES256",
|
||||
# ContentType=content_type,
|
||||
# )
|
||||
|
||||
|
||||
def test_s3upload_raises_exception(app, mocker):
|
||||
# def test_s3upload_raises_exception(app, mocker):
|
||||
|
||||
mock_s3_resource = mocker.Mock()
|
||||
mocked = mocker.patch(
|
||||
"notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
)
|
||||
response = {"Error": {"Code": 500}}
|
||||
exception = botocore.exceptions.ClientError(response, "Bad exception")
|
||||
mocked.return_value.Object.return_value.put.side_effect = exception
|
||||
with pytest.raises(botocore.exceptions.ClientError):
|
||||
s3upload(
|
||||
filedata=contents,
|
||||
region=region,
|
||||
bucket_name=bucket,
|
||||
file_location="location",
|
||||
)
|
||||
# mock_s3_resource = mocker.Mock()
|
||||
# mocked = mocker.patch(
|
||||
# "notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
# )
|
||||
# response = {"Error": {"Code": 500}}
|
||||
# exception = botocore.exceptions.ClientError(response, "Bad exception")
|
||||
# mocked.return_value.Object.return_value.put.side_effect = exception
|
||||
# with pytest.raises(botocore.exceptions.ClientError):
|
||||
# s3upload(
|
||||
# filedata=contents,
|
||||
# region=region,
|
||||
# bucket_name=bucket,
|
||||
# file_location="location",
|
||||
# )
|
||||
|
||||
|
||||
def test_s3upload_save_file_to_bucket_with_urlencoded_tags(mocker):
|
||||
# def test_s3upload_save_file_to_bucket_with_urlencoded_tags(mocker):
|
||||
|
||||
mock_s3_resource = mocker.Mock()
|
||||
mocked = mocker.patch(
|
||||
"notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
)
|
||||
# mock_s3_resource = mocker.Mock()
|
||||
# mocked = mocker.patch(
|
||||
# "notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
# )
|
||||
|
||||
s3upload(
|
||||
filedata=contents,
|
||||
region=region,
|
||||
bucket_name=bucket,
|
||||
file_location=location,
|
||||
tags={"a": "1/2", "b": "x y"},
|
||||
)
|
||||
mocked_put = mocked.return_value.Object.return_value.put
|
||||
# s3upload(
|
||||
# filedata=contents,
|
||||
# region=region,
|
||||
# bucket_name=bucket,
|
||||
# file_location=location,
|
||||
# tags={"a": "1/2", "b": "x y"},
|
||||
# )
|
||||
# mocked_put = mocked.return_value.Object.return_value.put
|
||||
|
||||
# make sure tags were a urlencoded query string
|
||||
encoded_tags = mocked_put.call_args[1]["Tagging"]
|
||||
assert parse_qs(encoded_tags) == {"a": ["1/2"], "b": ["x y"]}
|
||||
# # make sure tags were a urlencoded query string
|
||||
# encoded_tags = mocked_put.call_args[1]["Tagging"]
|
||||
# assert parse_qs(encoded_tags) == {"a": ["1/2"], "b": ["x y"]}
|
||||
|
||||
|
||||
def test_s3upload_save_file_to_bucket_with_metadata(mocker):
|
||||
# def test_s3upload_save_file_to_bucket_with_metadata(mocker):
|
||||
|
||||
mock_s3_resource = mocker.Mock()
|
||||
mocked = mocker.patch(
|
||||
"notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
)
|
||||
# mock_s3_resource = mocker.Mock()
|
||||
# mocked = mocker.patch(
|
||||
# "notifications_utils.s3.get_s3_resource", return_value=mock_s3_resource
|
||||
# )
|
||||
|
||||
s3upload(
|
||||
filedata=contents,
|
||||
region=region,
|
||||
bucket_name=bucket,
|
||||
file_location=location,
|
||||
metadata={"status": "valid", "pages": "5"},
|
||||
)
|
||||
mocked_put = mocked.return_value.Object.return_value.put
|
||||
# s3upload(
|
||||
# filedata=contents,
|
||||
# region=region,
|
||||
# bucket_name=bucket,
|
||||
# file_location=location,
|
||||
# metadata={"status": "valid", "pages": "5"},
|
||||
# )
|
||||
# mocked_put = mocked.return_value.Object.return_value.put
|
||||
|
||||
metadata = mocked_put.call_args[1]["Metadata"]
|
||||
assert metadata == {"status": "valid", "pages": "5"}
|
||||
# metadata = mocked_put.call_args[1]["Metadata"]
|
||||
# assert metadata == {"status": "valid", "pages": "5"}
|
||||
|
||||
|
||||
def test_get_s3_resource(mocker):
|
||||
|
||||
Reference in New Issue
Block a user