2018-06-28 08:39:25 +01:00
|
|
|
from datetime import datetime
|
|
|
|
|
|
|
|
|
|
from flask import Blueprint, jsonify, request
|
|
|
|
|
|
2019-08-28 15:24:16 +01:00
|
|
|
from app.dao.date_util import get_financial_year_for_datetime
|
2019-08-22 17:48:24 +01:00
|
|
|
from app.dao.fact_billing_dao import (
|
2021-02-23 18:37:05 +00:00
|
|
|
fetch_billing_details_for_all_services,
|
2022-04-07 17:52:37 +01:00
|
|
|
fetch_daily_sms_provider_volumes_for_platform,
|
2022-03-03 14:47:56 +00:00
|
|
|
fetch_daily_volumes_for_platform,
|
2021-06-11 11:11:47 +01:00
|
|
|
fetch_letter_costs_and_totals_for_all_services,
|
2021-03-10 13:55:06 +00:00
|
|
|
fetch_letter_line_items_for_all_services,
|
|
|
|
|
fetch_sms_billing_for_all_services,
|
2022-03-03 14:47:56 +00:00
|
|
|
fetch_volumes_by_service,
|
2019-08-22 17:48:24 +01:00
|
|
|
)
|
2021-03-10 13:55:06 +00:00
|
|
|
from app.dao.fact_notification_status_dao import (
|
|
|
|
|
fetch_notification_status_totals_for_all_services,
|
|
|
|
|
)
|
|
|
|
|
from app.errors import InvalidRequest, register_errors
|
2020-07-10 17:43:40 +01:00
|
|
|
from app.models import UK_POSTAGE_TYPES
|
2018-06-29 15:54:32 +01:00
|
|
|
from app.platform_stats.platform_stats_schema import platform_stats_request
|
|
|
|
|
from app.schema_validation import validate
|
2021-03-10 13:55:06 +00:00
|
|
|
from app.service.statistics import format_admin_stats
|
2022-11-10 12:33:25 -05:00
|
|
|
from app.utils import get_local_midnight_in_utc
|
2018-06-28 08:39:25 +01:00
|
|
|
|
|
|
|
|
platform_stats_blueprint = Blueprint('platform_stats', __name__)
|
|
|
|
|
|
|
|
|
|
register_errors(platform_stats_blueprint)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@platform_stats_blueprint.route('')
|
2018-06-29 15:54:32 +01:00
|
|
|
def get_platform_stats():
|
|
|
|
|
if request.args:
|
|
|
|
|
validate(request.args, platform_stats_request)
|
|
|
|
|
|
2018-06-28 08:39:25 +01:00
|
|
|
# If start and end date are not set, we are expecting today's stats.
|
|
|
|
|
today = str(datetime.utcnow().date())
|
|
|
|
|
|
|
|
|
|
start_date = datetime.strptime(request.args.get('start_date', today), '%Y-%m-%d').date()
|
|
|
|
|
end_date = datetime.strptime(request.args.get('end_date', today), '%Y-%m-%d').date()
|
2018-12-03 13:59:25 +00:00
|
|
|
data = fetch_notification_status_totals_for_all_services(start_date=start_date, end_date=end_date)
|
2018-06-28 08:39:25 +01:00
|
|
|
stats = format_admin_stats(data)
|
|
|
|
|
|
|
|
|
|
return jsonify(stats)
|
2019-08-22 17:48:24 +01:00
|
|
|
|
|
|
|
|
|
2022-03-03 14:47:56 +00:00
|
|
|
def validate_date_format(date_to_validate):
|
2019-08-22 17:48:24 +01:00
|
|
|
try:
|
2022-03-03 14:47:56 +00:00
|
|
|
validated_date = datetime.strptime(date_to_validate, "%Y-%m-%d").date()
|
2019-08-22 17:48:24 +01:00
|
|
|
except ValueError:
|
|
|
|
|
raise InvalidRequest(message="Input must be a date in the format: YYYY-MM-DD", status_code=400)
|
2022-03-03 14:47:56 +00:00
|
|
|
return validated_date
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def validate_date_range_is_within_a_financial_year(start_date, end_date):
|
|
|
|
|
start_date = validate_date_format(start_date)
|
|
|
|
|
end_date = validate_date_format(end_date)
|
2019-08-22 17:48:24 +01:00
|
|
|
if end_date < start_date:
|
|
|
|
|
raise InvalidRequest(message="Start date must be before end date", status_code=400)
|
2019-08-28 15:24:16 +01:00
|
|
|
|
2022-11-10 12:33:25 -05:00
|
|
|
start_fy = get_financial_year_for_datetime(get_local_midnight_in_utc(start_date))
|
|
|
|
|
end_fy = get_financial_year_for_datetime(get_local_midnight_in_utc(end_date))
|
2019-08-28 15:24:16 +01:00
|
|
|
|
|
|
|
|
if start_fy != end_fy:
|
2019-08-22 17:48:24 +01:00
|
|
|
raise InvalidRequest(message="Date must be in a single financial year.", status_code=400)
|
|
|
|
|
|
2019-08-30 17:16:43 +01:00
|
|
|
return start_date, end_date
|
|
|
|
|
|
2019-08-22 17:48:24 +01:00
|
|
|
|
|
|
|
|
@platform_stats_blueprint.route('usage-for-all-services')
|
2021-03-08 17:53:16 +00:00
|
|
|
@platform_stats_blueprint.route('data-for-billing-report')
|
|
|
|
|
def get_data_for_billing_report():
|
2019-08-22 17:48:24 +01:00
|
|
|
start_date = request.args.get('start_date')
|
|
|
|
|
end_date = request.args.get('end_date')
|
|
|
|
|
|
2019-08-30 17:16:43 +01:00
|
|
|
start_date, end_date = validate_date_range_is_within_a_financial_year(start_date, end_date)
|
2019-08-22 17:48:24 +01:00
|
|
|
|
|
|
|
|
sms_costs = fetch_sms_billing_for_all_services(start_date, end_date)
|
2021-06-11 11:11:47 +01:00
|
|
|
letter_overview = fetch_letter_costs_and_totals_for_all_services(start_date, end_date)
|
2019-08-22 17:48:24 +01:00
|
|
|
letter_breakdown = fetch_letter_line_items_for_all_services(start_date, end_date)
|
|
|
|
|
|
2019-08-23 17:02:44 +01:00
|
|
|
lb_by_service = [
|
2020-07-10 17:43:40 +01:00
|
|
|
(lb.service_id,
|
|
|
|
|
f"{lb.letters_sent} {postage_description(lb.postage)} letters at {format_letter_rate(lb.letter_rate)}")
|
2019-08-23 17:02:44 +01:00
|
|
|
for lb in letter_breakdown
|
|
|
|
|
]
|
2019-08-22 17:48:24 +01:00
|
|
|
combined = {}
|
|
|
|
|
for s in sms_costs:
|
2021-03-10 17:00:26 +00:00
|
|
|
if float(s.sms_cost) > 0:
|
|
|
|
|
entry = {
|
|
|
|
|
"organisation_id": str(s.organisation_id) if s.organisation_id else "",
|
|
|
|
|
"organisation_name": s.organisation_name or "",
|
|
|
|
|
"service_id": str(s.service_id),
|
|
|
|
|
"service_name": s.service_name,
|
|
|
|
|
"sms_cost": float(s.sms_cost),
|
2022-01-18 18:03:13 +00:00
|
|
|
"sms_chargeable_units": s.chargeable_billable_sms,
|
2021-06-11 11:11:47 +01:00
|
|
|
"total_letters": 0,
|
2021-03-10 17:00:26 +00:00
|
|
|
"letter_cost": 0,
|
|
|
|
|
"letter_breakdown": ""
|
|
|
|
|
}
|
|
|
|
|
combined[s.service_id] = entry
|
2019-08-22 17:48:24 +01:00
|
|
|
|
2021-06-11 11:11:47 +01:00
|
|
|
for data in letter_overview:
|
|
|
|
|
if data.service_id in combined:
|
|
|
|
|
combined[data.service_id].update(
|
|
|
|
|
{'total_letters': data.total_letters, 'letter_cost': float(data.letter_cost)}
|
|
|
|
|
)
|
|
|
|
|
|
2019-08-22 17:48:24 +01:00
|
|
|
else:
|
|
|
|
|
letter_entry = {
|
2021-06-11 11:11:47 +01:00
|
|
|
"organisation_id": str(data.organisation_id) if data.organisation_id else "",
|
|
|
|
|
"organisation_name": data.organisation_name or "",
|
|
|
|
|
"service_id": str(data.service_id),
|
|
|
|
|
"service_name": data.service_name,
|
2019-08-22 17:48:24 +01:00
|
|
|
"sms_cost": 0,
|
2022-01-18 18:03:13 +00:00
|
|
|
"sms_chargeable_units": 0,
|
2021-06-11 11:11:47 +01:00
|
|
|
"total_letters": data.total_letters,
|
|
|
|
|
"letter_cost": float(data.letter_cost),
|
2019-08-22 17:48:24 +01:00
|
|
|
"letter_breakdown": ""
|
|
|
|
|
}
|
2021-06-11 11:11:47 +01:00
|
|
|
combined[data.service_id] = letter_entry
|
2019-08-22 17:48:24 +01:00
|
|
|
for service_id, breakdown in lb_by_service:
|
2019-08-28 16:11:35 +01:00
|
|
|
combined[service_id]['letter_breakdown'] += (breakdown + '\n')
|
2019-08-22 17:48:24 +01:00
|
|
|
|
2021-02-23 18:37:05 +00:00
|
|
|
billing_details = fetch_billing_details_for_all_services()
|
|
|
|
|
for service in billing_details:
|
|
|
|
|
if service.service_id in combined:
|
|
|
|
|
combined[service.service_id].update({
|
|
|
|
|
'purchase_order_number': service.purchase_order_number,
|
|
|
|
|
'contact_names': service.billing_contact_names,
|
|
|
|
|
'contact_email_addresses': service.billing_contact_email_addresses,
|
|
|
|
|
'billing_reference': service.billing_reference
|
|
|
|
|
})
|
|
|
|
|
|
2019-08-29 15:45:50 +01:00
|
|
|
# sorting first by name == '' means that blank orgs will be sorted last.
|
2021-02-23 18:37:05 +00:00
|
|
|
|
|
|
|
|
result = sorted(combined.values(), key=lambda x: (
|
2019-08-29 15:45:50 +01:00
|
|
|
x['organisation_name'] == '',
|
|
|
|
|
x['organisation_name'],
|
|
|
|
|
x['service_name']
|
2021-02-23 18:37:05 +00:00
|
|
|
))
|
|
|
|
|
return jsonify(result)
|
2020-07-10 17:43:40 +01:00
|
|
|
|
|
|
|
|
|
2022-03-03 14:47:56 +00:00
|
|
|
@platform_stats_blueprint.route('daily-volumes-report')
|
|
|
|
|
def daily_volumes_report():
|
|
|
|
|
start_date = validate_date_format(request.args.get('start_date'))
|
|
|
|
|
end_date = validate_date_format(request.args.get('end_date'))
|
|
|
|
|
|
|
|
|
|
daily_volumes = fetch_daily_volumes_for_platform(start_date, end_date)
|
|
|
|
|
report = []
|
|
|
|
|
|
|
|
|
|
for row in daily_volumes:
|
|
|
|
|
report.append({
|
2022-11-21 11:49:59 -05:00
|
|
|
"day": row.local_date,
|
2022-03-09 11:55:47 +00:00
|
|
|
"sms_totals": int(row.sms_totals),
|
|
|
|
|
"sms_fragment_totals": int(row.sms_fragment_totals),
|
|
|
|
|
"sms_chargeable_units": int(row.sms_chargeable_units),
|
|
|
|
|
"email_totals": int(row.email_totals),
|
|
|
|
|
"letter_totals": int(row.letter_totals),
|
|
|
|
|
"letter_sheet_totals": int(row.letter_sheet_totals)
|
2022-03-03 14:47:56 +00:00
|
|
|
})
|
|
|
|
|
return jsonify(report)
|
|
|
|
|
|
|
|
|
|
|
2022-04-07 17:52:37 +01:00
|
|
|
@platform_stats_blueprint.route('daily-sms-provider-volumes-report')
|
|
|
|
|
def daily_sms_provider_volumes_report():
|
|
|
|
|
start_date = validate_date_format(request.args.get('start_date'))
|
|
|
|
|
end_date = validate_date_format(request.args.get('end_date'))
|
|
|
|
|
|
|
|
|
|
daily_volumes = fetch_daily_sms_provider_volumes_for_platform(start_date, end_date)
|
|
|
|
|
report = []
|
|
|
|
|
|
|
|
|
|
for row in daily_volumes:
|
|
|
|
|
report.append({
|
2022-11-21 11:49:59 -05:00
|
|
|
'day': row.local_date.isoformat(),
|
2022-04-07 17:52:37 +01:00
|
|
|
'provider': row.provider,
|
|
|
|
|
'sms_totals': int(row.sms_totals),
|
|
|
|
|
'sms_fragment_totals': int(row.sms_fragment_totals),
|
|
|
|
|
'sms_chargeable_units': int(row.sms_chargeable_units),
|
|
|
|
|
# convert from Decimal to float as it's not json serialisable
|
|
|
|
|
'sms_cost': float(row.sms_cost),
|
|
|
|
|
})
|
|
|
|
|
return jsonify(report)
|
|
|
|
|
|
|
|
|
|
|
2022-03-03 14:47:56 +00:00
|
|
|
@platform_stats_blueprint.route('volumes-by-service')
|
|
|
|
|
def volumes_by_service_report():
|
|
|
|
|
start_date = validate_date_format(request.args.get('start_date'))
|
|
|
|
|
end_date = validate_date_format(request.args.get('end_date'))
|
|
|
|
|
|
|
|
|
|
volumes_by_service = fetch_volumes_by_service(start_date, end_date)
|
|
|
|
|
report = []
|
|
|
|
|
|
|
|
|
|
for row in volumes_by_service:
|
|
|
|
|
report.append({
|
|
|
|
|
"service_name": row.service_name,
|
|
|
|
|
"service_id": str(row.service_id),
|
|
|
|
|
"organisation_name": row.organisation_name if row.organisation_name else '',
|
|
|
|
|
"organisation_id": str(row.organisation_id) if row.organisation_id else '',
|
2022-03-09 11:55:47 +00:00
|
|
|
"free_allowance": int(row.free_allowance),
|
|
|
|
|
"sms_notifications": int(row.sms_notifications),
|
|
|
|
|
"sms_chargeable_units": int(row.sms_chargeable_units),
|
|
|
|
|
"email_totals": int(row.email_totals),
|
|
|
|
|
"letter_totals": int(row.letter_totals),
|
|
|
|
|
"letter_sheet_totals": int(row.letter_sheet_totals),
|
2022-03-03 14:47:56 +00:00
|
|
|
"letter_cost": float(row.letter_cost),
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
return jsonify(report)
|
|
|
|
|
|
|
|
|
|
|
2020-07-10 17:43:40 +01:00
|
|
|
def postage_description(postage):
|
|
|
|
|
if postage in UK_POSTAGE_TYPES:
|
|
|
|
|
return f'{postage} class'
|
|
|
|
|
else:
|
|
|
|
|
return 'international'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def format_letter_rate(number):
|
|
|
|
|
if number >= 1:
|
|
|
|
|
return f"£{number:,.2f}"
|
|
|
|
|
|
|
|
|
|
return f"{number * 100:.0f}p"
|