mirror of
https://github.com/GSA/notifications-api.git
synced 2025-12-21 07:51:13 -05:00
This adds new endpoints to retrieve billing data from the new table:
1. Create a separate billing blueprint to house these endpoints
2. Return monthly breakdown in same format as we did before
3. Return yearly breakdown but only return {billing units, rate,
notification_type}. Admin only makes use of these.
This commit is contained in:
@@ -100,6 +100,7 @@ def register_blueprint(application):
|
|||||||
from app.notifications.notifications_letter_callback import letter_callback_blueprint
|
from app.notifications.notifications_letter_callback import letter_callback_blueprint
|
||||||
from app.authentication.auth import requires_admin_auth, requires_auth, requires_no_auth, restrict_ip_sms
|
from app.authentication.auth import requires_admin_auth, requires_auth, requires_no_auth, restrict_ip_sms
|
||||||
from app.letters.send_letter_jobs import letter_job
|
from app.letters.send_letter_jobs import letter_job
|
||||||
|
from app.billing.rest import billing_blueprint
|
||||||
|
|
||||||
service_blueprint.before_request(requires_admin_auth)
|
service_blueprint.before_request(requires_admin_auth)
|
||||||
application.register_blueprint(service_blueprint, url_prefix='/service')
|
application.register_blueprint(service_blueprint, url_prefix='/service')
|
||||||
@@ -164,6 +165,9 @@ def register_blueprint(application):
|
|||||||
letter_callback_blueprint.before_request(requires_no_auth)
|
letter_callback_blueprint.before_request(requires_no_auth)
|
||||||
application.register_blueprint(letter_callback_blueprint)
|
application.register_blueprint(letter_callback_blueprint)
|
||||||
|
|
||||||
|
billing_blueprint.before_request(requires_admin_auth)
|
||||||
|
application.register_blueprint(billing_blueprint)
|
||||||
|
|
||||||
|
|
||||||
def register_v2_blueprints(application):
|
def register_v2_blueprints(application):
|
||||||
from app.v2.notifications.post_notifications import v2_notification_blueprint as post_notifications
|
from app.v2.notifications.post_notifications import v2_notification_blueprint as post_notifications
|
||||||
|
|||||||
0
app/billing/__init__.py
Normal file
0
app/billing/__init__.py
Normal file
73
app/billing/rest.py
Normal file
73
app/billing/rest.py
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
import json
|
||||||
|
|
||||||
|
from flask import Blueprint, jsonify, request
|
||||||
|
|
||||||
|
from app.dao.notification_usage_dao import get_billing_data_for_month
|
||||||
|
from app.dao.monthly_billing_dao import get_billing_data_for_financial_year
|
||||||
|
from app.dao.date_util import get_financial_year
|
||||||
|
from app.errors import register_errors
|
||||||
|
from app.models import SMS_TYPE, EMAIL_TYPE
|
||||||
|
|
||||||
|
|
||||||
|
billing_blueprint = Blueprint(
|
||||||
|
'billing',
|
||||||
|
__name__,
|
||||||
|
url_prefix='/service/<uuid:service_id>/billing'
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
register_errors(billing_blueprint)
|
||||||
|
|
||||||
|
|
||||||
|
@billing_blueprint.route('/monthly-usage')
|
||||||
|
def get_yearly_usage_by_month(service_id):
|
||||||
|
try:
|
||||||
|
year = int(request.args.get('year'))
|
||||||
|
start_date, end_date = get_financial_year(year)
|
||||||
|
results = get_billing_data_for_month(service_id, start_date, end_date, SMS_TYPE)
|
||||||
|
json_results = [{
|
||||||
|
"month": datetime.strftime(x[0], "%B"),
|
||||||
|
"billing_units": x[1],
|
||||||
|
"rate_multiplier": x[2],
|
||||||
|
"international": x[3],
|
||||||
|
"notification_type": x[4],
|
||||||
|
"rate": x[5]
|
||||||
|
} for x in results]
|
||||||
|
return json.dumps(json_results)
|
||||||
|
except TypeError:
|
||||||
|
return jsonify(result='error', message='No valid year provided'), 400
|
||||||
|
|
||||||
|
|
||||||
|
@billing_blueprint.route('/yearly-usage-summary')
|
||||||
|
def get_yearly_billing_usage_summary(service_id):
|
||||||
|
try:
|
||||||
|
year = int(request.args.get('year'))
|
||||||
|
billing_data = get_billing_data_for_financial_year(service_id, year)
|
||||||
|
notification_types = [SMS_TYPE, EMAIL_TYPE]
|
||||||
|
response = [
|
||||||
|
_get_total_billable_units_and_rate_for_notification_type(billing_data, notification_type)
|
||||||
|
for notification_type in notification_types
|
||||||
|
]
|
||||||
|
|
||||||
|
return json.dumps(response)
|
||||||
|
|
||||||
|
except TypeError:
|
||||||
|
return jsonify(result='error', message='No valid year provided'), 400
|
||||||
|
|
||||||
|
|
||||||
|
def _get_total_billable_units_and_rate_for_notification_type(billing_data, noti_type):
|
||||||
|
total_sent = 0
|
||||||
|
rate = 0
|
||||||
|
for entry in billing_data:
|
||||||
|
for monthly_total in entry.monthly_totals:
|
||||||
|
if entry.notification_type == noti_type:
|
||||||
|
total_sent += monthly_total['billing_units'] \
|
||||||
|
if noti_type == EMAIL_TYPE else (monthly_total['billing_units'] * monthly_total['rate_multiplier'])
|
||||||
|
rate = monthly_total['rate']
|
||||||
|
|
||||||
|
return {
|
||||||
|
"notification_type": noti_type,
|
||||||
|
"billing_units": total_sent,
|
||||||
|
"rate": rate
|
||||||
|
}
|
||||||
@@ -510,13 +510,14 @@ def get_yearly_billing_usage(service_id):
|
|||||||
try:
|
try:
|
||||||
year = int(request.args.get('year'))
|
year = int(request.args.get('year'))
|
||||||
results = notification_usage_dao.get_yearly_billing_data(service_id, year)
|
results = notification_usage_dao.get_yearly_billing_data(service_id, year)
|
||||||
json_result = [{"credits": x[0],
|
json_result = [{
|
||||||
"billing_units": x[1],
|
"credits": x[0],
|
||||||
"rate_multiplier": x[2],
|
"billing_units": x[1],
|
||||||
"notification_type": x[3],
|
"rate_multiplier": x[2],
|
||||||
"international": x[4],
|
"notification_type": x[3],
|
||||||
"rate": x[5]
|
"international": x[4],
|
||||||
} for x in results]
|
"rate": x[5]
|
||||||
|
} for x in results]
|
||||||
return json.dumps(json_result)
|
return json.dumps(json_result)
|
||||||
|
|
||||||
except TypeError:
|
except TypeError:
|
||||||
|
|||||||
130
tests/app/billing/test_billing.py
Normal file
130
tests/app/billing/test_billing.py
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
from datetime import datetime, timedelta
|
||||||
|
import json
|
||||||
|
|
||||||
|
from app.dao.monthly_billing_dao import create_or_update_monthly_billing
|
||||||
|
from app.models import SMS_TYPE, EMAIL_TYPE
|
||||||
|
from tests.app.db import (
|
||||||
|
create_notification,
|
||||||
|
create_rate
|
||||||
|
)
|
||||||
|
from tests import create_authorization_header
|
||||||
|
|
||||||
|
IN_MAY_2016 = datetime(2016, 5, 10, 23, 00, 00)
|
||||||
|
IN_JUN_2016 = datetime(2016, 6, 3, 23, 00, 00)
|
||||||
|
|
||||||
|
|
||||||
|
def _assert_dict_equals(actual, expected_dict):
|
||||||
|
assert set(actual.keys()) == set(expected_dict.keys())
|
||||||
|
assert set(actual.values()) == set(expected_dict.values())
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_yearly_billing_summary_returns_correct_breakdown(client, sample_template):
|
||||||
|
create_rate(start_date=IN_MAY_2016 - timedelta(days=1), value=0.12, notification_type=SMS_TYPE)
|
||||||
|
create_notification(
|
||||||
|
template=sample_template, created_at=IN_MAY_2016,
|
||||||
|
billable_units=1, rate_multiplier=2, status='delivered'
|
||||||
|
)
|
||||||
|
create_notification(
|
||||||
|
template=sample_template, created_at=IN_JUN_2016,
|
||||||
|
billable_units=2, rate_multiplier=3, status='delivered'
|
||||||
|
)
|
||||||
|
|
||||||
|
create_or_update_monthly_billing(service_id=sample_template.service_id, billing_month=IN_MAY_2016)
|
||||||
|
create_or_update_monthly_billing(service_id=sample_template.service_id, billing_month=IN_JUN_2016)
|
||||||
|
|
||||||
|
response = client.get(
|
||||||
|
'/service/{}/billing/yearly-usage-summary?year=2016'.format(sample_template.service.id),
|
||||||
|
headers=[create_authorization_header()]
|
||||||
|
)
|
||||||
|
assert response.status_code == 200
|
||||||
|
|
||||||
|
resp_json = json.loads(response.get_data(as_text=True))
|
||||||
|
assert len(resp_json) == 2
|
||||||
|
|
||||||
|
_assert_dict_equals(resp_json[0], {
|
||||||
|
'notification_type': SMS_TYPE,
|
||||||
|
'billing_units': 8,
|
||||||
|
'rate': 0.12
|
||||||
|
})
|
||||||
|
|
||||||
|
_assert_dict_equals(resp_json[1], {
|
||||||
|
'notification_type': EMAIL_TYPE,
|
||||||
|
'billing_units': 0,
|
||||||
|
'rate': 0
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_yearly_billing_usage_breakdown_returns_400_if_missing_year(client, sample_service):
|
||||||
|
response = client.get(
|
||||||
|
'/service/{}/billing/yearly-usage-summary'.format(sample_service.id),
|
||||||
|
headers=[create_authorization_header()]
|
||||||
|
)
|
||||||
|
assert response.status_code == 400
|
||||||
|
assert json.loads(response.get_data(as_text=True)) == {
|
||||||
|
'message': 'No valid year provided', 'result': 'error'
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_yearly_usage_by_month_returns_400_if_missing_year(client, sample_service):
|
||||||
|
response = client.get(
|
||||||
|
'/service/{}/billing/monthly-usage'.format(sample_service.id),
|
||||||
|
headers=[create_authorization_header()]
|
||||||
|
)
|
||||||
|
assert response.status_code == 400
|
||||||
|
assert json.loads(response.get_data(as_text=True)) == {
|
||||||
|
'message': 'No valid year provided', 'result': 'error'
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_yearly_usage_by_month_returns_empty_list_if_no_notifications(client, sample_template):
|
||||||
|
create_rate(start_date=IN_MAY_2016 - timedelta(days=1), value=0.12, notification_type=SMS_TYPE)
|
||||||
|
response = client.get(
|
||||||
|
'/service/{}/billing/monthly-usage?year=2016'.format(sample_template.service.id),
|
||||||
|
headers=[create_authorization_header()]
|
||||||
|
)
|
||||||
|
assert response.status_code == 200
|
||||||
|
|
||||||
|
results = json.loads(response.get_data(as_text=True))
|
||||||
|
assert results == []
|
||||||
|
|
||||||
|
|
||||||
|
def test_get_yearly_usage_by_month_returns_correctly(client, sample_template):
|
||||||
|
create_rate(start_date=IN_MAY_2016 - timedelta(days=1), value=0.12, notification_type=SMS_TYPE)
|
||||||
|
create_notification(
|
||||||
|
template=sample_template, created_at=IN_MAY_2016,
|
||||||
|
billable_units=1, rate_multiplier=2, status='delivered'
|
||||||
|
)
|
||||||
|
create_notification(
|
||||||
|
template=sample_template, created_at=IN_JUN_2016,
|
||||||
|
billable_units=2, rate_multiplier=3, status='delivered'
|
||||||
|
)
|
||||||
|
|
||||||
|
create_or_update_monthly_billing(service_id=sample_template.service_id, billing_month=IN_MAY_2016)
|
||||||
|
create_or_update_monthly_billing(service_id=sample_template.service_id, billing_month=IN_JUN_2016)
|
||||||
|
|
||||||
|
response = client.get(
|
||||||
|
'/service/{}/billing/monthly-usage?year=2016'.format(sample_template.service.id),
|
||||||
|
headers=[create_authorization_header()]
|
||||||
|
)
|
||||||
|
|
||||||
|
assert response.status_code == 200
|
||||||
|
|
||||||
|
resp_json = json.loads(response.get_data(as_text=True))
|
||||||
|
|
||||||
|
_assert_dict_equals(resp_json[0], {
|
||||||
|
'billing_units': 1,
|
||||||
|
'international': False,
|
||||||
|
'month': 'May',
|
||||||
|
'notification_type': SMS_TYPE,
|
||||||
|
'rate': 0.12,
|
||||||
|
'rate_multiplier': 2
|
||||||
|
})
|
||||||
|
|
||||||
|
_assert_dict_equals(resp_json[1], {
|
||||||
|
'billing_units': 2,
|
||||||
|
'international': False,
|
||||||
|
'month': 'June',
|
||||||
|
'notification_type': SMS_TYPE,
|
||||||
|
'rate': 0.12,
|
||||||
|
'rate_multiplier': 3
|
||||||
|
})
|
||||||
Reference in New Issue
Block a user