From 02c77da97b3d3bee2b72ad68108283d62c64ef03 Mon Sep 17 00:00:00 2001
From: Ken Tsang
Date: Fri, 7 Apr 2017 11:16:39 +0100
Subject: [PATCH 1/4] Add letter-jobs page, update admin menu
---
app/main/__init__.py | 3 +-
app/main/views/letter_jobs.py | 45 ++++++++++++++++++++++++++++
app/templates/admin_template.html | 3 ++
app/templates/views/letter-jobs.html | 43 ++++++++++++++++++++++++++
4 files changed, 93 insertions(+), 1 deletion(-)
create mode 100644 app/main/views/letter_jobs.py
create mode 100644 app/templates/views/letter-jobs.html
diff --git a/app/main/__init__.py b/app/main/__init__.py
index b70ccb942..0bf151fe3 100644
--- a/app/main/__init__.py
+++ b/app/main/__init__.py
@@ -26,5 +26,6 @@ from app.main.views import (
invites,
feedback,
providers,
- platform_admin
+ platform_admin,
+ letter_jobs
)
diff --git a/app/main/views/letter_jobs.py b/app/main/views/letter_jobs.py
new file mode 100644
index 000000000..00cb314db
--- /dev/null
+++ b/app/main/views/letter_jobs.py
@@ -0,0 +1,45 @@
+from flask import (render_template, url_for, redirect, request, abort)
+from app.main import main
+from app import convert_to_boolean
+from flask_login import (login_required, current_user)
+
+
+@main.route("/letter-jobs", methods=['GET', 'POST'])
+@login_required
+def letter_jobs():
+ letter_jobs_list = get_letter_jobs()
+
+ msg = ''
+ if request.method == 'POST':
+ send_letters = request.form.getlist('send_letter')
+ for job_id in send_letters:
+ job = [j for j in letter_jobs_list if job_id == j['job_id']][0]
+ job['send'] = True
+
+ msg = 'sending:{}'.format(send_letters)
+
+ return render_template('views/letter-jobs.html', letter_jobs_list=letter_jobs_list, message=msg)
+
+
+def get_letter_jobs():
+ return [
+ {
+ 'service_name': 'test_name',
+ 'job_id': 'test_id',
+ 'status': 'test_status',
+ 'created_at': '2017-04-01'
+ },
+ {
+ 'service_name': 'test_name 2',
+ 'job_id': 'test_id 2',
+ 'status': 'test_status 2',
+ 'created_at': '2017-04-02'
+
+ },
+ {
+ 'service_name': 'test_name 3',
+ 'job_id': 'test_id 3',
+ 'status': 'test_status 3',
+ 'created_at': '2017-04-03'
+ }
+ ]
diff --git a/app/templates/admin_template.html b/app/templates/admin_template.html
index 4fc273400..5bd9868ad 100644
--- a/app/templates/admin_template.html
+++ b/app/templates/admin_template.html
@@ -52,6 +52,9 @@
Providers
+
+ Letter jobs
+
{% endif %}
Sign out
diff --git a/app/templates/views/letter-jobs.html b/app/templates/views/letter-jobs.html
new file mode 100644
index 000000000..aaa50f549
--- /dev/null
+++ b/app/templates/views/letter-jobs.html
@@ -0,0 +1,43 @@
+{% extends "withoutnav_template.html" %}
+{% from "components/page-footer.html" import page_footer %}
+
+{% block service_page_title %}
+ Show letter jobs
+{% endblock %}
+
+{% block maincolumn_content %}
+
+ Letter jobs
+
+
+ {{ page_footer('Send') }}
+
+ {% if message %}
+
+ {{ message }}
+
+ {% endif %}
+
+
+{% endblock %}
From e5a377edd82dc986346597bcf0c828797120c518 Mon Sep 17 00:00:00 2001
From: Ken Tsang
Date: Tue, 11 Apr 2017 10:59:50 +0100
Subject: [PATCH 2/4] Add letter jobs page, client and test
---
app/__init__.py | 3 +
app/main/views/letter_jobs.py | 56 +++++-------
app/notify_client/letter_jobs_client.py | 21 +++++
app/templates/views/letter-jobs.html | 20 +++--
tests/app/main/views/test_letter_jobs.py | 103 +++++++++++++++++++++++
5 files changed, 161 insertions(+), 42 deletions(-)
create mode 100644 app/notify_client/letter_jobs_client.py
create mode 100644 tests/app/main/views/test_letter_jobs.py
diff --git a/app/__init__.py b/app/__init__.py
index a3e22afbd..23f4f8bd5 100644
--- a/app/__init__.py
+++ b/app/__init__.py
@@ -52,6 +52,7 @@ from app.notify_client.events_api_client import EventsApiClient
from app.notify_client.provider_client import ProviderClient
from app.notify_client.organisations_client import OrganisationsClient
from app.notify_client.models import AnonymousUser
+from app.notify_client.letter_jobs_client import LetterJobsClient
login_manager = LoginManager()
csrf = CsrfProtect()
@@ -69,6 +70,7 @@ provider_client = ProviderClient()
organisations_client = OrganisationsClient()
asset_fingerprinter = AssetFingerprinter()
statsd_client = StatsdClient()
+letter_jobs_client = LetterJobsClient()
# The current service attached to the request stack.
current_service = LocalProxy(partial(_lookup_req_object, 'service'))
@@ -100,6 +102,7 @@ def create_app():
events_api_client.init_app(application)
provider_client.init_app(application)
organisations_client.init_app(application)
+ letter_jobs_client.init_app(application)
login_manager.init_app(application)
login_manager.login_view = 'main.sign_in'
diff --git a/app/main/views/letter_jobs.py b/app/main/views/letter_jobs.py
index 00cb314db..e371f33f2 100644
--- a/app/main/views/letter_jobs.py
+++ b/app/main/views/letter_jobs.py
@@ -1,45 +1,31 @@
-from flask import (render_template, url_for, redirect, request, abort)
+import datetime
+
+from flask import render_template, request
+from flask_login import login_required
+
+from app import letter_jobs_client, format_datetime_24h
from app.main import main
-from app import convert_to_boolean
-from flask_login import (login_required, current_user)
+from app.utils import user_has_permissions
@main.route("/letter-jobs", methods=['GET', 'POST'])
@login_required
+@user_has_permissions(admin_override=True)
def letter_jobs():
- letter_jobs_list = get_letter_jobs()
-
msg = ''
- if request.method == 'POST':
- send_letters = request.form.getlist('send_letter')
- for job_id in send_letters:
- job = [j for j in letter_jobs_list if job_id == j['job_id']][0]
- job['send'] = True
+ letter_jobs_list = letter_jobs_client.get_letter_jobs()
- msg = 'sending:{}'.format(send_letters)
+ if request.method == 'POST':
+ if len(request.form.getlist('job_id')) > 0:
+ job_ids = request.form.getlist('job_id')
+
+ response = letter_jobs_client.send_letter_jobs(job_ids)
+ msg = response['response']
+
+ for job_id in job_ids:
+ job = [j for j in letter_jobs_list if job_id == j['id']][0]
+ job['sending'] = 'sending'
+ else:
+ msg = 'No jobs selected'
return render_template('views/letter-jobs.html', letter_jobs_list=letter_jobs_list, message=msg)
-
-
-def get_letter_jobs():
- return [
- {
- 'service_name': 'test_name',
- 'job_id': 'test_id',
- 'status': 'test_status',
- 'created_at': '2017-04-01'
- },
- {
- 'service_name': 'test_name 2',
- 'job_id': 'test_id 2',
- 'status': 'test_status 2',
- 'created_at': '2017-04-02'
-
- },
- {
- 'service_name': 'test_name 3',
- 'job_id': 'test_id 3',
- 'status': 'test_status 3',
- 'created_at': '2017-04-03'
- }
- ]
diff --git a/app/notify_client/letter_jobs_client.py b/app/notify_client/letter_jobs_client.py
new file mode 100644
index 000000000..a3337917b
--- /dev/null
+++ b/app/notify_client/letter_jobs_client.py
@@ -0,0 +1,21 @@
+from app.notify_client import NotifyAdminAPIClient
+
+
+class LetterJobsClient(NotifyAdminAPIClient):
+
+ def __init__(self):
+ super().__init__("a", "b", "c")
+
+ def init_app(self, app):
+ self.base_url = app.config['API_HOST_NAME']
+ self.service_id = app.config['ADMIN_CLIENT_USER_NAME']
+ self.api_key = app.config['ADMIN_CLIENT_SECRET']
+
+ def get_letter_jobs(self):
+ return self.get(url='/letter-jobs')['data']
+
+ def send_letter_jobs(self, job_ids):
+ return self.post(
+ url='/send-letter-jobs',
+ data={"job_ids": job_ids}
+ )['data']
diff --git a/app/templates/views/letter-jobs.html b/app/templates/views/letter-jobs.html
index aaa50f549..4f039aef4 100644
--- a/app/templates/views/letter-jobs.html
+++ b/app/templates/views/letter-jobs.html
@@ -17,25 +17,31 @@
Service name |
Job ID |
Status |
- Created at |
+ Created at |
+
{% for job in letter_jobs_list %}
- | {{ job.service_name }} |
- {{ job.job_id }} |
- {{ job.status }} |
- {{ job.created_at }} |
- |
+ {{ job.service_name.name }} |
+ {{ job.id }} |
+ {{ job.job_status }} |
+ {{ job.created_at|format_datetime_short }} |
+ |
+ {{ job.sending }} |
{% endfor %}
+
{{ page_footer('Send') }}
{% if message %}
-
+
{{ message }}
+ {% if message != 'No jobs selected' %}
+
Refresh page to see status updates
+ {% endif %}
{% endif %}
diff --git a/tests/app/main/views/test_letter_jobs.py b/tests/app/main/views/test_letter_jobs.py
new file mode 100644
index 000000000..5e7467a23
--- /dev/null
+++ b/tests/app/main/views/test_letter_jobs.py
@@ -0,0 +1,103 @@
+from flask import url_for
+from bs4 import BeautifulSoup
+
+from app import format_datetime_short
+
+valid_letter_jobs = [
+ {
+ 'service_name': {'name': 'test_name'},
+ 'template_version': 1,
+ 'id': 'test_id',
+ 'job_status': 'test_status',
+ 'created_at': '2017-04-01T12:00:00'
+ },
+ {
+ 'service_name': {'name': 'test_name 2'},
+ 'template_version': 2,
+ 'id': 'test_id 2',
+ 'job_status': 'test_status 2',
+ 'created_at': '2017-04-02T13:00:00'
+ },
+ {
+ 'service_name': {'name': 'test_name 3'},
+ 'template_version': 3,
+ 'id': 'test_id 3',
+ 'job_status': 'test_status 3',
+ 'created_at': '2017-04-03T14:00:00'
+ }
+]
+
+send_letter_jobs_response = {"response": "Task created to send files to DVLA"}
+
+
+def test_get_letter_jobs_returns_list_of_all_letter_jobs(logged_in_platform_admin_client, mocker):
+ mock_get_letters = mocker.patch('app.letter_jobs_client.get_letter_jobs', return_value=valid_letter_jobs)
+
+ response = logged_in_platform_admin_client.get(url_for('main.letter_jobs'))
+
+ assert mock_get_letters.called
+ assert response.status_code == 200
+
+ page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
+ assert page.h1.string == 'Letter jobs'
+
+ table = page.find('table')
+ table_body = table.find('tbody')
+ rows = table_body.find_all('tr')
+
+ assert len(rows) == len(valid_letter_jobs)
+
+ for row_pos in range(len(rows)):
+ cols = rows[row_pos].find_all('td')
+ assert valid_letter_jobs[row_pos]['service_name']['name'] == cols[0].text
+ assert valid_letter_jobs[row_pos]['id'] == cols[1].text
+ assert valid_letter_jobs[row_pos]['job_status'] == cols[2].text
+ assert format_datetime_short(valid_letter_jobs[row_pos]['created_at']) == cols[3].text
+
+
+def test_post_letter_jobs_select_1_letter_job_submits_1_job(logged_in_platform_admin_client, mocker):
+ letter_jobs_first_selected = {'job_id': ['test_id']}
+
+ mock_get_letters = mocker.patch('app.letter_jobs_client.get_letter_jobs', return_value=valid_letter_jobs)
+ mock_send_letters = mocker.patch('app.letter_jobs_client.send_letter_jobs', return_value=send_letter_jobs_response)
+
+ response = logged_in_platform_admin_client.post(url_for('main.letter_jobs'), data=letter_jobs_first_selected)
+
+ assert mock_get_letters.called
+ assert mock_send_letters.called
+ assert response.status_code == 200
+
+ page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
+
+ table = page.find('table')
+ table_body = table.find('tbody')
+ rows = table_body.find_all('tr')
+
+ assert len(rows) == len(valid_letter_jobs)
+
+ colr0 = rows[0].find_all('td')
+ colr1 = rows[1].find_all('td')
+ colr2 = rows[2].find_all('td')
+
+ assert colr0[5].text == "sending"
+ assert colr1[5].text == ""
+ assert colr2[5].text == ""
+
+ message = page.find('p', attrs={'id': 'message'}).text
+ assert "Task created to send files to DVLA" in message
+
+
+def test_post_letter_jobs_none_selected_shows_message(logged_in_platform_admin_client, mocker):
+ mock_get_letters = mocker.patch('app.letter_jobs_client.get_letter_jobs', return_value=valid_letter_jobs)
+ mock_send_letters = mocker.patch('app.letter_jobs_client.send_letter_jobs', return_value=send_letter_jobs_response)
+
+ response = logged_in_platform_admin_client.post(url_for('main.letter_jobs'), data={})
+
+ assert mock_get_letters.called
+ assert not mock_send_letters.called
+ assert response.status_code == 200
+
+ page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
+ message = page.find('p', attrs={'id': 'message'}).text
+
+ assert "No jobs selected" in message
From 9ce4ce803141d4709840ddf47fdf486e05cd6335 Mon Sep 17 00:00:00 2001
From: Ken Tsang
Date: Tue, 11 Apr 2017 15:02:20 +0100
Subject: [PATCH 3/4] Add count and disabled checkbox until ready / dvla state
---
app/templates/views/letter-jobs.html | 4 ++-
tests/app/main/views/test_letter_jobs.py | 42 ++++++++++++++++--------
2 files changed, 32 insertions(+), 14 deletions(-)
diff --git a/app/templates/views/letter-jobs.html b/app/templates/views/letter-jobs.html
index 4f039aef4..624358b7f 100644
--- a/app/templates/views/letter-jobs.html
+++ b/app/templates/views/letter-jobs.html
@@ -16,6 +16,7 @@
| Service name |
Job ID |
+ Count |
Status |
Created at |
@@ -25,9 +26,10 @@
| {{ job.service_name.name }} |
{{ job.id }} |
+ {{ job.notification_count }} |
{{ job.job_status }} |
{{ job.created_at|format_datetime_short }} |
- |
+ |
{{ job.sending }} |
{% endfor %}
diff --git a/tests/app/main/views/test_letter_jobs.py b/tests/app/main/views/test_letter_jobs.py
index 5e7467a23..af95b509e 100644
--- a/tests/app/main/views/test_letter_jobs.py
+++ b/tests/app/main/views/test_letter_jobs.py
@@ -1,3 +1,4 @@
+from enum import IntEnum
from flask import url_for
from bs4 import BeautifulSoup
@@ -6,23 +7,23 @@ from app import format_datetime_short
valid_letter_jobs = [
{
'service_name': {'name': 'test_name'},
- 'template_version': 1,
'id': 'test_id',
- 'job_status': 'test_status',
+ 'notification_count': 2,
+ 'job_status': 'ready to send',
'created_at': '2017-04-01T12:00:00'
},
{
'service_name': {'name': 'test_name 2'},
- 'template_version': 2,
'id': 'test_id 2',
- 'job_status': 'test_status 2',
+ 'notification_count': 1,
+ 'job_status': 'sent to dvla',
'created_at': '2017-04-02T13:00:00'
},
{
'service_name': {'name': 'test_name 3'},
- 'template_version': 3,
'id': 'test_id 3',
- 'job_status': 'test_status 3',
+ 'notification_count': 1,
+ 'job_status': 'in progress',
'created_at': '2017-04-03T14:00:00'
}
]
@@ -30,6 +31,16 @@ valid_letter_jobs = [
send_letter_jobs_response = {"response": "Task created to send files to DVLA"}
+class letter_jobs_header(IntEnum):
+ SERVICE_NAME = 0
+ JOB_ID = 1
+ NOTIFICATION_COUNT = 2
+ JOB_STATUS = 3
+ CREATED_AT = 4
+ CHECKBOX = 5
+ TEMP_STATUS = 6
+
+
def test_get_letter_jobs_returns_list_of_all_letter_jobs(logged_in_platform_admin_client, mocker):
mock_get_letters = mocker.patch('app.letter_jobs_client.get_letter_jobs', return_value=valid_letter_jobs)
@@ -49,10 +60,15 @@ def test_get_letter_jobs_returns_list_of_all_letter_jobs(logged_in_platform_admi
for row_pos in range(len(rows)):
cols = rows[row_pos].find_all('td')
- assert valid_letter_jobs[row_pos]['service_name']['name'] == cols[0].text
- assert valid_letter_jobs[row_pos]['id'] == cols[1].text
- assert valid_letter_jobs[row_pos]['job_status'] == cols[2].text
- assert format_datetime_short(valid_letter_jobs[row_pos]['created_at']) == cols[3].text
+ assert valid_letter_jobs[row_pos]['service_name']['name'] == cols[letter_jobs_header.SERVICE_NAME].text
+ assert valid_letter_jobs[row_pos]['id'] == cols[letter_jobs_header.JOB_ID].text
+ assert valid_letter_jobs[row_pos]['notification_count'] == int(cols[letter_jobs_header.NOTIFICATION_COUNT].text)
+ assert valid_letter_jobs[row_pos]['job_status'] == cols[letter_jobs_header.JOB_STATUS].text
+ assert format_datetime_short(
+ valid_letter_jobs[row_pos]['created_at']) == cols[letter_jobs_header.CREATED_AT].text
+ if not (valid_letter_jobs[row_pos]['job_status'] == 'ready to send' or
+ valid_letter_jobs[row_pos]['job_status'] == 'sent to dvla'):
+ assert 'disabled' in str(cols[letter_jobs_header.CHECKBOX])
def test_post_letter_jobs_select_1_letter_job_submits_1_job(logged_in_platform_admin_client, mocker):
@@ -79,9 +95,9 @@ def test_post_letter_jobs_select_1_letter_job_submits_1_job(logged_in_platform_a
colr1 = rows[1].find_all('td')
colr2 = rows[2].find_all('td')
- assert colr0[5].text == "sending"
- assert colr1[5].text == ""
- assert colr2[5].text == ""
+ assert colr0[letter_jobs_header.TEMP_STATUS].text == "sending"
+ assert colr1[letter_jobs_header.TEMP_STATUS].text == ""
+ assert colr2[letter_jobs_header.TEMP_STATUS].text == ""
message = page.find('p', attrs={'id': 'message'}).text
assert "Task created to send files to DVLA" in message
From 0f3131cf21d7e4e810d33f6b50b94280714a4705 Mon Sep 17 00:00:00 2001
From: Ken Tsang
Date: Tue, 11 Apr 2017 15:17:18 +0100
Subject: [PATCH 4/4] Refactor test_letter_jobs
---
tests/app/main/views/test_letter_jobs.py | 8 ++------
1 file changed, 2 insertions(+), 6 deletions(-)
diff --git a/tests/app/main/views/test_letter_jobs.py b/tests/app/main/views/test_letter_jobs.py
index af95b509e..aa63c7bf3 100644
--- a/tests/app/main/views/test_letter_jobs.py
+++ b/tests/app/main/views/test_letter_jobs.py
@@ -52,9 +52,7 @@ def test_get_letter_jobs_returns_list_of_all_letter_jobs(logged_in_platform_admi
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
assert page.h1.string == 'Letter jobs'
- table = page.find('table')
- table_body = table.find('tbody')
- rows = table_body.find_all('tr')
+ rows = page.select('table tbody tr')
assert len(rows) == len(valid_letter_jobs)
@@ -85,9 +83,7 @@ def test_post_letter_jobs_select_1_letter_job_submits_1_job(logged_in_platform_a
page = BeautifulSoup(response.data.decode('utf-8'), 'html.parser')
- table = page.find('table')
- table_body = table.find('tbody')
- rows = table_body.find_all('tr')
+ rows = page.select('table tbody tr')
assert len(rows) == len(valid_letter_jobs)