From bd4ac80850c6591f77779c15855ac767362d4d88 Mon Sep 17 00:00:00 2001 From: Doug Lovett Date: Thu, 15 Sep 2022 09:26:28 -0700 Subject: [PATCH] Create staff mhr registration report. (#944) * Create staff mhr registration report. Signed-off-by: Doug Lovett * 13523 solve last page watermark issue. Signed-off-by: Doug Lovett * UXA update to header subtitle format. Signed-off-by: Doug Lovett * UXA updates and add mail out report changes. Signed-off-by: Doug Lovett * More UXA changes. Signed-off-by: Doug Lovett Signed-off-by: Doug Lovett --- mhr_api/report-templates/coverV2.html | 78 + mhr_api/report-templates/registrationV2.html | 64 +- mhr_api/report-templates/searchResultV2.html | 3 +- .../report-templates/static/v2/footer.html | 4 +- .../static/v2/footer_cover.html | 67 + .../static/v2/footer_mail.html | 68 + .../static/v2/header_cover.html | 108 ++ .../static/v2/header_mail.html | 133 ++ .../static/v2/header_registration.html | 127 ++ .../static/v2/header_replace.html | 2 +- .../template-parts/registration/location.html | 130 +- .../template-parts/registration/owners.html | 17 +- .../template-parts/registration/sections.html | 51 +- .../registration/submittingParty.html | 34 + .../template-parts/v2/style.html | 19 +- .../template-parts/v2/styleMail.html | 1356 +++++++++++++++++ .../template-parts/v2/stylePage.html | 12 +- .../template-parts/v2/stylePageDraft.html | 7 + .../template-parts/v2/stylePageMail.html | 16 + .../v2/stylePageRegistration.html | 30 + mhr_api/src/mhr_api/models/db2/manuhome.py | 4 +- .../src/mhr_api/models/mhr_registration.py | 26 + mhr_api/src/mhr_api/reports/v2/report.py | 117 +- .../src/mhr_api/reports/v2/report_utils.py | 233 ++- mhr_api/src/mhr_api/resources/utils.py | 11 + .../v1/registration_report_callback.py | 7 +- .../resources/v1/search_report_callback.py | 6 +- mhr_api/src/mhr_api/version.py | 2 +- .../unit/models/test_mhr_registration.py | 6 + .../data/registration-common-example.json | 18 +- .../data/registration-joint-example.json | 121 ++ .../data/registration-sole-example.json | 28 +- .../reports/test_report_registration_v2.py | 43 + 33 files changed, 2826 insertions(+), 122 deletions(-) create mode 100644 mhr_api/report-templates/coverV2.html mode change 100644 => 100755 mhr_api/report-templates/searchResultV2.html mode change 100644 => 100755 mhr_api/report-templates/static/v2/footer.html create mode 100755 mhr_api/report-templates/static/v2/footer_cover.html create mode 100755 mhr_api/report-templates/static/v2/footer_mail.html create mode 100644 mhr_api/report-templates/static/v2/header_cover.html create mode 100644 mhr_api/report-templates/static/v2/header_mail.html create mode 100644 mhr_api/report-templates/static/v2/header_registration.html mode change 100644 => 100755 mhr_api/report-templates/static/v2/header_replace.html create mode 100644 mhr_api/report-templates/template-parts/registration/submittingParty.html mode change 100644 => 100755 mhr_api/report-templates/template-parts/v2/style.html create mode 100755 mhr_api/report-templates/template-parts/v2/styleMail.html mode change 100644 => 100755 mhr_api/report-templates/template-parts/v2/stylePage.html mode change 100644 => 100755 mhr_api/report-templates/template-parts/v2/stylePageDraft.html create mode 100644 mhr_api/report-templates/template-parts/v2/stylePageMail.html create mode 100644 mhr_api/report-templates/template-parts/v2/stylePageRegistration.html mode change 100644 => 100755 mhr_api/src/mhr_api/models/db2/manuhome.py mode change 100644 => 100755 mhr_api/src/mhr_api/models/mhr_registration.py mode change 100644 => 100755 mhr_api/src/mhr_api/reports/v2/report.py mode change 100644 => 100755 mhr_api/src/mhr_api/reports/v2/report_utils.py mode change 100644 => 100755 mhr_api/src/mhr_api/resources/v1/registration_report_callback.py mode change 100644 => 100755 mhr_api/tests/unit/models/test_mhr_registration.py mode change 100644 => 100755 mhr_api/tests/unit/reports/data/registration-common-example.json create mode 100644 mhr_api/tests/unit/reports/data/registration-joint-example.json mode change 100644 => 100755 mhr_api/tests/unit/reports/data/registration-sole-example.json mode change 100644 => 100755 mhr_api/tests/unit/reports/test_report_registration_v2.py diff --git a/mhr_api/report-templates/coverV2.html b/mhr_api/report-templates/coverV2.html new file mode 100644 index 000000000..6122f1577 --- /dev/null +++ b/mhr_api/report-templates/coverV2.html @@ -0,0 +1,78 @@ +[[macros.html]] + + + + + {{ meta_title }} + + + [[v2/stylePageMail.html]] + [[v2/styleMail.html]] + + +
+
+{# + +#} + {% if cover.line3 %} + + {% else %} + + {% endif %} + + + + +
+
+
+
Attention/Reference Number: + {% if attentionReference is defined and attentionReference != '' %} + {{ attentionReference }} + {% else %} + N/A + {% endif %} +
+
+ Your document has been registered under the Manufactured Home Act, details of which appear + on the following page(s). If there are any errors or omissions please contact our office within thirty (30) days. +
+
Manufactured Home Registration Number: {{ mhrNumber }}
+
Document ID: {{ documentId }}
+
Document Type: + {% if documentDescription is defined and documentDescription != '' %} + {{ documentDescription }} + {% else %} + N/A + {% endif %} +
+
Document Registration Number: + {% if documentRegistrationId is defined and documentRegistrationId != '' %} + {{ documentRegistrationId }} + {% else %} + N/A + {% endif %} +
+
Document Registration Date and Time: {{ createDateTime }}
+
Examiner: N/A
+
Toll-Free Phone: 1-800-526-1526
+
+ + + diff --git a/mhr_api/report-templates/registrationV2.html b/mhr_api/report-templates/registrationV2.html index 1d7f569f4..299c00458 100644 --- a/mhr_api/report-templates/registrationV2.html +++ b/mhr_api/report-templates/registrationV2.html @@ -6,22 +6,25 @@ {{ meta_title }} - {% if environment == '' %} - [[v2/stylePage.html]] + {% if cover is not defined %} + {% if environment == '' %} + [[v2/stylePageRegistration.html]] + {% else %} + [[v2/stylePageDraft.html]] + {% endif %} + [[v2/style.html]] {% else %} - [[v2/stylePageDraft.html]] + [[v2/stylePageMail.html]] + [[v2/styleMail.html]] {% endif %} - [[v2/style.html]] -
- - - - - - - + {% if cover is not defined %} +
+ {% else %} +
+ {% endif %} +
@@ -48,6 +51,7 @@ {% endif %} +{# +#}
-
-
-
CURRENT REGISTRATION INFORMATION
-
(as of {{searchDateTime}})
-
-
-
[[registration/owners.html]] [[registration/location.html]] [[registration/details.html]] [[registration/sections.html]] -
- +
+
+
Rebuilt Status:
+ {% if description.rebuiltRemarks is defined and description.rebuiltRemarks != '' %} +
{{description.rebuiltRemarks|safe}}
+ {% else %} +
N/A
+ {% endif %} +
+ +
+
+
Other Information:
+ {% if description.otherRemarks is defined and description.otherRemarks != '' %} +
{{description.otherRemarks|safe}}
+ {% else %} +
N/A
+ {% endif %} +
+ + [[registration/submittingParty.html]] + + {% if cover is not defined %} +

+ {% endif %} + diff --git a/mhr_api/report-templates/searchResultV2.html b/mhr_api/report-templates/searchResultV2.html old mode 100644 new mode 100755 index 0d1fb1aba..5513820b3 --- a/mhr_api/report-templates/searchResultV2.html +++ b/mhr_api/report-templates/searchResultV2.html @@ -103,6 +103,7 @@ [[v2/search-result/registration.html]] {% endfor %} {% endif %} -

+ +

diff --git a/mhr_api/report-templates/static/v2/footer.html b/mhr_api/report-templates/static/v2/footer.html old mode 100644 new mode 100755 index 73042e3c7..e1e4da280 --- a/mhr_api/report-templates/static/v2/footer.html +++ b/mhr_api/report-templates/static/v2/footer.html @@ -42,8 +42,8 @@ } .footer-column-logo img { - width: 70%; - height: 70%; + width: 51%; + height: 51%; } diff --git a/mhr_api/report-templates/static/v2/footer_cover.html b/mhr_api/report-templates/static/v2/footer_cover.html new file mode 100755 index 000000000..cf56dd78f --- /dev/null +++ b/mhr_api/report-templates/static/v2/footer_cover.html @@ -0,0 +1,67 @@ + + + + + + + + diff --git a/mhr_api/report-templates/static/v2/footer_mail.html b/mhr_api/report-templates/static/v2/footer_mail.html new file mode 100755 index 000000000..8b8545cee --- /dev/null +++ b/mhr_api/report-templates/static/v2/footer_mail.html @@ -0,0 +1,68 @@ + + + + + + + + diff --git a/mhr_api/report-templates/static/v2/header_cover.html b/mhr_api/report-templates/static/v2/header_cover.html new file mode 100644 index 000000000..4f15ce5c8 --- /dev/null +++ b/mhr_api/report-templates/static/v2/header_cover.html @@ -0,0 +1,108 @@ + + + + Reg Report + + + +
+ + + + + +
+ + +
{{TITLE}}
+
{{SUBTITLE}}
+
+
+
+
+
+ + diff --git a/mhr_api/report-templates/static/v2/header_mail.html b/mhr_api/report-templates/static/v2/header_mail.html new file mode 100644 index 000000000..903a43d3d --- /dev/null +++ b/mhr_api/report-templates/static/v2/header_mail.html @@ -0,0 +1,133 @@ + + + + Reg Report + + + +
+ + + + + +
+ + +
{{TITLE}}
+
{{SUBTITLE}}
+
+
+ + + + + +
+
+ + diff --git a/mhr_api/report-templates/static/v2/header_registration.html b/mhr_api/report-templates/static/v2/header_registration.html new file mode 100644 index 000000000..d6d7dd9cb --- /dev/null +++ b/mhr_api/report-templates/static/v2/header_registration.html @@ -0,0 +1,127 @@ + + + + Reg Report + + + +
+ + + + + +
+ + +
{{TITLE}}
+
{{SUBTITLE}}
+
+
+ + + + + +
+
+ + diff --git a/mhr_api/report-templates/static/v2/header_replace.html b/mhr_api/report-templates/static/v2/header_replace.html old mode 100644 new mode 100755 index 01b178be7..926c01dd7 --- a/mhr_api/report-templates/static/v2/header_replace.html +++ b/mhr_api/report-templates/static/v2/header_replace.html @@ -78,7 +78,7 @@
{{TITLE}}
-
BC Registries and Online Services
+
{{SUBTITLE}}
diff --git a/mhr_api/report-templates/template-parts/registration/location.html b/mhr_api/report-templates/template-parts/registration/location.html index 84060aaa3..1eaa4b465 100644 --- a/mhr_api/report-templates/template-parts/registration/location.html +++ b/mhr_api/report-templates/template-parts/registration/location.html @@ -2,23 +2,19 @@
Registered Location
- + + {% if location.dealerName is defined and location.dealerName != '' %} + + + {% elif location.parkName is defined and location.parkName != '' %} + + + + + {% else %} + + + + - + + {% endif %} + + {% if location.hasLTSAInfo or (location.additionalDescription is defined and location.additionalDescription != '') %} + + + + + + + + {% endif %}
{% endif %} diff --git a/mhr_api/report-templates/template-parts/registration/owners.html b/mhr_api/report-templates/template-parts/registration/owners.html index 14b1777fd..2e259d65d 100644 --- a/mhr_api/report-templates/template-parts/registration/owners.html +++ b/mhr_api/report-templates/template-parts/registration/owners.html @@ -2,24 +2,25 @@ {% if ownerGroups is defined %} {% if ownerGroups|length > 1 %}
- Home Tenancy Type:  + Home Tenancy Type: Tenants in Common
+
{% endif %} {% for group in ownerGroups %} {% if ownerGroups|length > 1 %}
Group {{ group.groupId }}:  {% if group.interest is defined and group.interest != '' %} - Interest {{ group.interest }} + Interest {{ group.interest|lower }} {% endif %}
{% endif %} - + {% for party in group.owners %} + {% if not loop.last %} + + {% endif %} {% endfor %} - + {% if not loop.last %}
{% endif %} diff --git a/mhr_api/report-templates/template-parts/registration/sections.html b/mhr_api/report-templates/template-parts/registration/sections.html index e209401dd..430361cdd 100644 --- a/mhr_api/report-templates/template-parts/registration/sections.html +++ b/mhr_api/report-templates/template-parts/registration/sections.html @@ -1,25 +1,30 @@
-{% if description is defined and description.sections is defined %} -
-
Manufactured Home Sections
-
Number of Sections: {{ description.sections|length }}
- - - - - - - - {% for section in description.sections %} - - - - - + {% if description is defined and description.sections is defined %} +
+
Manufactured Home Sections
+
Number of Sections: {{ description.sections|length }}
+ + + + + + - {% endfor %} - -{% else %} -
None
-{% endif %} -
+ {% for section in description.sections %} + +
+ + +
{{loop.index}}.
+
{{section.serialNumber}}
+
{{section.lengthFeet}} feet {{section.lengthInches}} inches
+
{{section.widthFeet}} feet {{section.widthInches}} inches
+ + {% endfor %} + + + {% else %} +
None
+ {% endif %} + + \ No newline at end of file diff --git a/mhr_api/report-templates/template-parts/registration/submittingParty.html b/mhr_api/report-templates/template-parts/registration/submittingParty.html new file mode 100644 index 000000000..dd8bfe082 --- /dev/null +++ b/mhr_api/report-templates/template-parts/registration/submittingParty.html @@ -0,0 +1,34 @@ +{% if submittingParty is defined %} +
+
+
Submitting Party Information
+ + + + + + + +
+{% endif %} diff --git a/mhr_api/report-templates/template-parts/v2/style.html b/mhr_api/report-templates/template-parts/v2/style.html old mode 100644 new mode 100755 index 55689d5c5..3dd0a511b --- a/mhr_api/report-templates/template-parts/v2/style.html +++ b/mhr_api/report-templates/template-parts/v2/style.html @@ -1,4 +1,8 @@ diff --git a/mhr_api/report-templates/template-parts/v2/stylePage.html b/mhr_api/report-templates/template-parts/v2/stylePage.html old mode 100644 new mode 100755 index 898dfdfe5..733936b7e --- a/mhr_api/report-templates/template-parts/v2/stylePage.html +++ b/mhr_api/report-templates/template-parts/v2/stylePage.html @@ -6,12 +6,18 @@ orphans: 4 !important; widows: 2 !important; } - + @page last_page { + @bottom-right { + width: 10px; + height: 1px; + content: none; + } + } @media print { body { - background-image: url(''); + background-image: url(''); background-repeat:repeat-y; - background-position: 175px 0px !important; /* 400*868 previous 400 * 844 */ + background-position: 175px 0px !important; /* 400*863 previous 400 * 868 */ } } diff --git a/mhr_api/report-templates/template-parts/v2/stylePageDraft.html b/mhr_api/report-templates/template-parts/v2/stylePageDraft.html old mode 100644 new mode 100755 index e10dd33fd..70f1c78fd --- a/mhr_api/report-templates/template-parts/v2/stylePageDraft.html +++ b/mhr_api/report-templates/template-parts/v2/stylePageDraft.html @@ -6,6 +6,13 @@ orphans: 4 !important; widows: 2 !important; } + @page last_page { + @bottom-right { + width: 10px; + height: 1px; + content: none; + } + } @media print { body { diff --git a/mhr_api/report-templates/template-parts/v2/stylePageMail.html b/mhr_api/report-templates/template-parts/v2/stylePageMail.html new file mode 100644 index 000000000..048b0f365 --- /dev/null +++ b/mhr_api/report-templates/template-parts/v2/stylePageMail.html @@ -0,0 +1,16 @@ + diff --git a/mhr_api/report-templates/template-parts/v2/stylePageRegistration.html b/mhr_api/report-templates/template-parts/v2/stylePageRegistration.html new file mode 100644 index 000000000..c933ffb33 --- /dev/null +++ b/mhr_api/report-templates/template-parts/v2/stylePageRegistration.html @@ -0,0 +1,30 @@ + diff --git a/mhr_api/src/mhr_api/models/db2/manuhome.py b/mhr_api/src/mhr_api/models/db2/manuhome.py old mode 100644 new mode 100755 index bdca99aed..c0731c43a --- a/mhr_api/src/mhr_api/models/db2/manuhome.py +++ b/mhr_api/src/mhr_api/models/db2/manuhome.py @@ -36,6 +36,7 @@ 'C': 'HISTORICAL' } DOCUMENT_TYPE_REG = '101' +DOCUMENT_TYPE_CONV = 'CONV' class Db2Manuhome(db.Model): @@ -194,6 +195,7 @@ def registration_json(self): for doc in self.reg_documents: if self.reg_document_id and self.reg_document_id == doc.id: doc_json = doc.registration_json + man_home['documentId'] = doc_json.get('documentId', '') man_home['createDateTime'] = doc_json.get('createDateTime', '') man_home['clientReferenceId'] = doc_json.get('clientReferenceId', '') man_home['attentionReference'] = doc_json.get('attentionReference', '') @@ -239,7 +241,7 @@ def new_registration_json(self): if self.reg_documents: for doc in self.reg_documents: if self.reg_document_id and self.reg_document_id == doc.id and \ - doc.document_type.strip() == DOCUMENT_TYPE_REG: + doc.document_type.strip() in (DOCUMENT_TYPE_REG, DOCUMENT_TYPE_CONV): doc_json = doc.registration_json man_home['documentId'] = doc_json.get('documentId', '') man_home['createDateTime'] = doc_json.get('createDateTime', '') diff --git a/mhr_api/src/mhr_api/models/mhr_registration.py b/mhr_api/src/mhr_api/models/mhr_registration.py old mode 100644 new mode 100755 index 53ee48385..c0dc00536 --- a/mhr_api/src/mhr_api/models/mhr_registration.py +++ b/mhr_api/src/mhr_api/models/mhr_registration.py @@ -49,6 +49,11 @@ '103': 'REG_103', '103E': 'REG_103E' } +DOC_TYPE_REPORT_DESCRIPTION = { + '101': 'Register New Unit', + 'CONV': 'Register New Unit', + 'DEFAULT': '' +} class MhrRegistration(db.Model): # pylint: disable=too-many-instance-attributes, too-many-public-methods @@ -83,6 +88,7 @@ class MhrRegistration(db.Model): # pylint: disable=too-many-instance-attributes draft_number: str = None doc_reg_number: str = None manuhome: Db2Manuhome = None + mail_version: bool = False @property def json(self) -> dict: @@ -116,6 +122,26 @@ def registration_json(self) -> dict: return reg_json return self.json + @property + def new_registration_json(self) -> dict: + """Return the new registration version of the registration as a json object.""" + if self.manuhome: + reg_json = self.manuhome.new_registration_json + if self.mail_version and self.manuhome.reg_documents: + reg_doc = None + for doc in self.manuhome.reg_documents: + if self.manuhome.reg_document_id and self.manuhome.reg_document_id == doc.id: + reg_doc = doc + if reg_doc: + reg_json['documentRegistrationId'] = reg_doc.document_reg_id + doc_type = reg_doc.document_type + if doc_type in DOC_TYPE_REPORT_DESCRIPTION: + reg_json['documentDescription'] = DOC_TYPE_REPORT_DESCRIPTION[doc_type] + else: + reg_json['documentDescription'] = '' + return reg_json + return self.json + def __set_payment_json(self, registration): """Add registration payment info json if payment exists.""" if self.pay_invoice_id and self.pay_path: diff --git a/mhr_api/src/mhr_api/reports/v2/report.py b/mhr_api/src/mhr_api/reports/v2/report.py old mode 100644 new mode 100755 index 4862b96d1..4b5dff330 --- a/mhr_api/src/mhr_api/reports/v2/report.py +++ b/mhr_api/src/mhr_api/reports/v2/report.py @@ -13,6 +13,7 @@ from http import HTTPStatus from pathlib import Path +import markupsafe import pycountry import requests from flask import current_app, jsonify @@ -57,13 +58,15 @@ def get_pdf(self, report_type=None): if self._report_key == ReportTypes.SEARCH_DETAIL_REPORT: current_app.logger.debug('Search report generating TOC page numbers as a second report api call.') return self.get_search_pdf() + if self._report_key == ReportTypes.MHR_REGISTRATION_MAIL: + return self.get_registration_mail_pdf() current_app.logger.debug('Account {0} report type {1} setting up report data.' .format(self._account_id, self._report_key)) data = self._setup_report_data() url = current_app.config.get('REPORT_SVC_URL') + SINGLE_URI current_app.logger.debug('Account {0} report type {1} calling report-api {2}.' .format(self._account_id, self._report_key, url)) - meta_data = report_utils.get_report_meta_data() + meta_data = report_utils.get_report_meta_data(self._report_key) files = report_utils.get_report_files(data, self._report_key) headers = {} token = GoogleAuthService.get_report_api_token() @@ -89,7 +92,7 @@ def get_search_pdf(self): url = current_app.config.get('REPORT_SVC_URL') + SINGLE_URI current_app.logger.debug('Account {0} report type {1} calling report-api {2}.' .format(self._account_id, self._report_key, url)) - meta_data = report_utils.get_report_meta_data() + meta_data = report_utils.get_report_meta_data(self._report_key) files = report_utils.get_report_files(data, self._report_key) headers = {} token = GoogleAuthService.get_report_api_token() @@ -120,6 +123,61 @@ def get_search_pdf(self): return jsonify(message=content), response.status_code, None return response.content, response.status_code, {'Content-Type': 'application/pdf'} + def get_registration_mail_pdf(self): + """Render a mail registration report with cover letter.""" + current_app.logger.debug('Account {0} setting up mail reg report data.'.format(self._account_id,)) + create_ts = self._report_data['createDateTime'] + # 1: Generate the cover page report. + self._report_key = ReportTypes.MHR_COVER + data = self._setup_report_data() + url = current_app.config.get('REPORT_SVC_URL') + SINGLE_URI + meta_data = report_utils.get_report_meta_data(self._report_key) + files = report_utils.get_report_files(data, self._report_key, True) + headers = {} + token = GoogleAuthService.get_report_api_token() + if token: + headers['Authorization'] = 'Bearer {}'.format(token) + response_cover = requests.post(url=url, headers=headers, data=meta_data, files=files) + current_app.logger.debug('Account {0} report type {1} response status: {2}.' + .format(self._account_id, self._report_key, response_cover.status_code)) + if response_cover.status_code != HTTPStatus.OK: + content = ResourceErrorCodes.REPORT_ERR + ': ' + response_cover.content.decode('ascii') + current_app.logger.error('Account {0} response status: {1} error: {2}.' + .format(self._account_id, response_cover.status_code, content)) + return jsonify(message=content), response_cover.status_code, None + + # 2: Generate the registration pdf. + self._report_key = ReportTypes.MHR_REGISTRATION + self._report_data['createDateTime'] = create_ts + data = self._setup_report_data() + current_app.logger.debug('Account {0} report type {1} calling report-api {2}.' + .format(self._account_id, self._report_key, url)) + meta_data = report_utils.get_report_meta_data(self._report_key) + files = report_utils.get_report_files(data, self._report_key, True) + response_reg = requests.post(url=url, headers=headers, data=meta_data, files=files) + current_app.logger.debug('Account {0} report type {1} response status: {2}.' + .format(self._account_id, self._report_key, response_reg.status_code)) + if response_reg.status_code != HTTPStatus.OK: + content = ResourceErrorCodes.REPORT_ERR + ': ' + response_reg.content.decode('ascii') + current_app.logger.error('Account {0} response status: {1} error: {2}.' + .format(self._account_id, response_reg.status_code, content)) + return jsonify(message=content), response_reg.status_code, None + # 3: Merge cover leter and registraiton reports. + url = current_app.config.get('REPORT_SVC_URL') + MERGE_URI + files = { + 'pdf1.pdf': response_cover.content, + 'pdf2.pdf': response_reg.content + } + response = requests.post(url=url, headers=headers, files=files) + current_app.logger.debug('Merge cover and registration reports response status: {0}.' + .format(response.status_code)) + if response.status_code != HTTPStatus.OK: + content = ResourceErrorCodes.REPORT_ERR + ': ' + response.content.decode('ascii') + current_app.logger.error('Account {0} merge response status: {1} error: {2}.' + .format(self._account_id, response.status_code, content)) + return jsonify(message=content), response.status_code, None + return response.content, response.status_code, {'Content-Type': 'application/pdf'} + def _setup_report_data(self): """Set up the report service request data.""" # current_app.logger.debug('Setup report data template starting.') @@ -186,8 +244,11 @@ def _substitute_template_parts(template_code): template_path = current_app.config.get('REPORT_TEMPLATE_PATH') template_parts = [ 'v2/style', + 'v2/styleMail', 'v2/stylePage', 'v2/stylePageDraft', + 'v2/stylePageMail', + 'v2/stylePageRegistration', 'stylePageMail', 'logo', 'macros', @@ -197,6 +258,7 @@ def _substitute_template_parts(template_code): 'registration/notes', 'registration/owners', 'registration/sections', + 'registration/submittingParty', 'search-result/details', 'search-result/location', 'search-result/notes', @@ -242,6 +304,9 @@ def _get_template_data(self): self._set_meta_info() if self._report_key == ReportTypes.SEARCH_TOC_REPORT: self._set_selected() + elif self._report_key == ReportTypes.MHR_COVER: + self._report_data['cover'] = report_utils.set_cover(self._report_data) + self._report_data['createDateTime'] = Report._to_report_datetime(self._report_data['createDateTime']) else: self._set_addresses() self._set_date_times() @@ -253,6 +318,8 @@ def _get_template_data(self): elif self._report_key == ReportTypes.SEARCH_BODY_REPORT: # Add PPR search template setup here: self._set_ppr_search() + self._set_location() + self._set_description() return self._report_data def _set_ppr_search(self): # pylint: disable=too-many-branches, too-many-statements @@ -272,6 +339,30 @@ def _set_notes(self): if self._report_key == ReportTypes.SEARCH_DETAIL_REPORT and self._report_data['totalResultsSize'] > 0: self._set_search_notes() + def _set_description(self): + """Set up report description information.""" + if self._report_key == ReportTypes.MHR_REGISTRATION: + description = self._report_data.get('description') + if description and description.get('rebuiltRemarks'): + description['rebuiltRemarks'] = markupsafe.Markup(description['rebuiltRemarks']) + if description and description.get('otherRemarks'): + description['rebuiltRemarks'] = markupsafe.Markup(description['otherRemarks']) + + def _set_location(self): + """Set up report location information.""" + if self._report_key == ReportTypes.MHR_REGISTRATION: + location = self._report_data.get('location') + if location.get('lot') or location.get('parcel') or location.get('block') or location.get('districtLot') or\ + location.get('partOf') or location.get('section') or location.get('township') or \ + location.get('range') or location.get('meridian') or location.get('landDistrict') or \ + location.get('plan'): + location['hasLTSAInfo'] = True + else: + location['hasLTSAInfo'] = False + if location.get('pidNumber'): + pid = location.get('pidNumber') + location['pidNumber'] = pid[0:3] + '-' + pid[3:6] + '-' + pid[6:] + def _set_search_notes(self): """Add search note document type description and dates.""" if self._report_data and self._report_data['details']: @@ -312,6 +403,8 @@ def _set_registration_addresses(self): """Replace registration addresses country code with description.""" if self._report_data: reg = self._report_data + if reg.get('submittingParty'): + Report._format_address(reg['submittingParty']['address']) if reg.get('ownerGroups'): for group in reg['ownerGroups']: for owner in group['owners']: @@ -394,6 +487,7 @@ def _set_meta_info(self): # Get source ??? # Appears in the Description section of the PDF Document Properties as Title. self._report_data['meta_title'] = ReportMeta.reports[self._report_key]['metaTitle'].upper() + self._report_data['meta_subtitle'] = ReportMeta.reports[self._report_key]['metaSubtitle'] # Appears in the Description section of the PDF Document Properties as Subject. if self._report_key in (ReportTypes.SEARCH_DETAIL_REPORT, @@ -414,9 +508,9 @@ def _set_meta_info(self): self._report_data['footer_content'] = f'MHR Number Search - "{criteria}"' else: self._report_data['footer_content'] = f'MHR {search_desc} Search - "{criteria}"' - elif self._report_key == ReportTypes.MHR_REGISTRATION: + elif self._report_key in (ReportTypes.MHR_REGISTRATION, ReportTypes.MHR_COVER): reg_num = self._report_data.get('mhrNumber', '') - self._report_data['footer_content'] = f'MH Registraton Number {reg_num}' + self._report_data['footer_content'] = f'Manufactured Home Registration #{reg_num}' self._report_data['meta_subject'] = f'Manufactured Home Registration Number: {reg_num}' if self._get_environment() != '': self._report_data['footer_content'] = 'TEST DATA | ' + self._report_data['footer_content'] @@ -436,6 +530,8 @@ def _get_environment(): @staticmethod def _to_report_datetime(date_time: str, include_time: bool = True, expiry: bool = False): """Convert ISO formatted date time or date string to report format.""" + if len(date_time) < 10: # Legacy may be empty string. + return date_time if len(date_time) == 10: # Legacy has some date only data. report_date = model_utils.date_from_iso_format(date_time) return report_date.strftime('%B %-d, %Y') @@ -459,28 +555,39 @@ class ReportMeta: # pylint: disable=too-few-public-methods """Helper class to maintain the report meta information.""" reports = { + ReportTypes.MHR_COVER: { + 'reportDescription': 'MHRCover', + 'fileName': 'coverV2', + 'metaTitle': 'VERIFICATION OF SERVICE', + 'metaSubtitle': 'Manufactured Home Registry', + 'metaSubject': '' + }, ReportTypes.MHR_REGISTRATION: { 'reportDescription': 'MHRRegistration', 'fileName': 'registrationV2', - 'metaTitle': 'Manufactured Home Registration', + 'metaTitle': 'VERIFICATION OF SERVICE', + 'metaSubtitle': 'Manufactured Home Registry', 'metaSubject': '' }, ReportTypes.SEARCH_DETAIL_REPORT: { 'reportDescription': 'SearchResult', 'fileName': 'searchResultV2', 'metaTitle': 'Manufactured Home Registry Search Result', + 'metaSubtitle': 'BC Registries and Online Services', 'metaSubject': '' }, ReportTypes.SEARCH_TOC_REPORT: { 'reportDescription': 'SearchResult', 'fileName': 'searchResultTOCV2', 'metaTitle': 'Personal Property Registry Search Result', + 'metaSubtitle': 'BC Registries and Online Services', 'metaSubject': '' }, ReportTypes.SEARCH_BODY_REPORT: { 'reportDescription': 'SearchResult', 'fileName': 'searchResultBodyV2', 'metaTitle': 'Manufactured Home Registry Search Result', + 'metaSubtitle': 'BC Registries and Online Services', 'metaSubject': '' } } diff --git a/mhr_api/src/mhr_api/reports/v2/report_utils.py b/mhr_api/src/mhr_api/reports/v2/report_utils.py old mode 100644 new mode 100755 index baaf94d11..d10d9b7fa --- a/mhr_api/src/mhr_api/reports/v2/report_utils.py +++ b/mhr_api/src/mhr_api/reports/v2/report_utils.py @@ -21,13 +21,21 @@ HEADER_PATH = '/static/v2/header_replace.html' -FOOTER_PAGES_PATH = '/static/v2/footer.html' +HEADER_COVER_PATH = '/static/v2/header_cover.html' +HEADER_MAIL_PATH = '/static/v2/header_mail.html' +HEADER_REG_PATH = '/static/v2/header_registration.html' +FOOTER_PATH = '/static/v2/footer.html' +FOOTER_COVER_PATH = '/static/v2/footer_cover.html' +FOOTER_MAIL_PATH = '/static/v2/footer_mail.html' HEADER_TITLE_REPLACE = '{{TITLE}}' +HEADER_SUBTITLE_REPLACE = '{{SUBTITLE}}' +HEADER_SUBJECT_REPLACE = '{{SUBJECT}}' FOOTER_TEXT_REPLACE = '{{FOOTER-TEXT}}' -# marginTop 1.5 +MARGIN_TOP_REG_REPORT = 1.93 +# marginTop 1.5 bottom 0.7 REPORT_META_DATA = { 'marginTop': 1.25, - 'marginBottom': 0.7, + 'marginBottom': 0.75, 'marginLeft': 0.4, 'marginRight': 0.4, 'printBackground': True @@ -43,7 +51,9 @@ class ReportTypes(BaseEnum): """Render an Enum of the MHR PDF report types.""" + MHR_COVER = 'mhrCover' MHR_REGISTRATION = 'mhrRegistration' + MHR_REGISTRATION_MAIL = 'mhrRegistrationMail' SEARCH_DETAIL_REPORT = 'searchDetail' # Gotenberg SEARCH_TOC_REPORT = 'searchTOC' @@ -54,7 +64,12 @@ class Config: # pylint: disable=too-few-public-methods """Configuration that loads report template static data.""" HEADER_TEMPLATE: str = None + HEADER_COVER_TEMPLATE: str = None + HEADER_MAIL_TEMPLATE: str = None + HEADER_REG_TEMPLATE: str = None FOOTER_TEMPLATE: str = None + FOOTER_COVER_TEMPLATE: str = None + FOOTER_MAIL_TEMPLATE: str = None @classmethod def get_header_template(cls) -> str: @@ -68,11 +83,47 @@ def get_header_template(cls) -> str: current_app.logger.error(f'Error loading header template from path={file_path}: ' + str(err)) return cls.HEADER_TEMPLATE + @classmethod + def get_reg_header_template(cls) -> str: + """Fetch registration header template data from the file system.""" + if not cls.HEADER_REG_TEMPLATE: + file_path = current_app.config.get('REPORT_TEMPLATE_PATH', '') + HEADER_REG_PATH + try: + cls.HEADER_REG_TEMPLATE = Path(file_path).read_text() + current_app.logger.info(f'Loaded registration header file from path {file_path}') + except Exception as err: # noqa: B902; just logging + current_app.logger.error(f'Error loading reg header template from path={file_path}: ' + str(err)) + return cls.HEADER_REG_TEMPLATE + + @classmethod + def get_cover_header_template(cls) -> str: + """Fetch mail cover letter header template data from the file system.""" + if not cls.HEADER_COVER_TEMPLATE: + file_path = current_app.config.get('REPORT_TEMPLATE_PATH', '') + HEADER_COVER_PATH + try: + cls.HEADER_COVER_TEMPLATE = Path(file_path).read_text() + current_app.logger.info(f'Loaded mail cover header file from path {file_path}') + except Exception as err: # noqa: B902; just logging + current_app.logger.error(f'Error loading mail cover header template from path={file_path}: ' + str(err)) + return cls.HEADER_COVER_TEMPLATE + + @classmethod + def get_mail_header_template(cls) -> str: + """Fetch mail registration header template data from the file system.""" + if not cls.HEADER_MAIL_TEMPLATE: + file_path = current_app.config.get('REPORT_TEMPLATE_PATH', '') + HEADER_MAIL_PATH + try: + cls.HEADER_MAIL_TEMPLATE = Path(file_path).read_text() + current_app.logger.info(f'Loaded mail registration header file from path {file_path}') + except Exception as err: # noqa: B902; just logging + current_app.logger.error(f'Error loading mail reg header template from path={file_path}: ' + str(err)) + return cls.HEADER_MAIL_TEMPLATE + @classmethod def get_footer_template(cls) -> str: """Fetch footer template data from the file system.""" if not cls.FOOTER_TEMPLATE: - file_path = current_app.config.get('REPORT_TEMPLATE_PATH', '') + FOOTER_PAGES_PATH + file_path = current_app.config.get('REPORT_TEMPLATE_PATH', '') + FOOTER_PATH try: cls.FOOTER_TEMPLATE = Path(file_path).read_text() current_app.logger.info(f'Loaded footer file from path {file_path}') @@ -80,43 +131,128 @@ def get_footer_template(cls) -> str: current_app.logger.error(f'Error loading footer template from path={file_path}: ' + str(err)) return cls.FOOTER_TEMPLATE + @classmethod + def get_cover_footer_template(cls) -> str: + """Fetch cover letter footer template data from the file system.""" + if not cls.FOOTER_COVER_TEMPLATE: + file_path = current_app.config.get('REPORT_TEMPLATE_PATH', '') + FOOTER_COVER_PATH + try: + cls.FOOTER_COVER_TEMPLATE = Path(file_path).read_text() + current_app.logger.info(f'Loaded mail cover footer file from path {file_path}') + except Exception as err: # noqa: B902; just logging + current_app.logger.error(f'Error loading mail cover footer template from path={file_path}: ' + str(err)) + return cls.FOOTER_COVER_TEMPLATE + + @classmethod + def get_mail_footer_template(cls) -> str: + """Fetch footer template data from the file system.""" + if not cls.FOOTER_MAIL_TEMPLATE: + file_path = current_app.config.get('REPORT_TEMPLATE_PATH', '') + FOOTER_MAIL_PATH + try: + cls.FOOTER_MAIL_TEMPLATE = Path(file_path).read_text() + current_app.logger.info(f'Loaded mail footer file from path {file_path}') + except Exception as err: # noqa: B902; just logging + current_app.logger.error(f'Error loading mail footer template from path={file_path}: ' + str(err)) + return cls.FOOTER_MAIL_TEMPLATE + -def get_header_data(title: str) -> str: - """Get report header with the provided title.""" +def get_header_data(title: str, subtitle: str = '') -> str: + """Get report header with the provided titles.""" template = Config().get_header_template() if template: - return template.replace(HEADER_TITLE_REPLACE, title) + return template.replace(HEADER_TITLE_REPLACE, title).replace(HEADER_SUBTITLE_REPLACE, subtitle) return None -def get_footer_data(footer_text: str) -> str: +def get_reg_header_data(title: str, subtitle: str, subject: str, mail: bool = False) -> str: + """Get registration report header with the provided titles and subject.""" + if mail: + return get_mail_header_data(title, subtitle, subject) + template = Config().get_reg_header_template() + if template: + rep_template = template.replace(HEADER_TITLE_REPLACE, title).replace(HEADER_SUBTITLE_REPLACE, subtitle) + return rep_template.replace(HEADER_SUBJECT_REPLACE, subject) + return None + + +def get_mail_header_data(title: str, subtitle: str, subject: str) -> str: + """Get a mail registration report header with the provided titles and subject.""" + template = Config().get_mail_header_template() + if template: + rep_template = template.replace(HEADER_TITLE_REPLACE, title).replace(HEADER_SUBTITLE_REPLACE, subtitle) + return rep_template.replace(HEADER_SUBJECT_REPLACE, subject) + return None + + +def get_cover_header_data(title: str, subtitle: str, subject: str) -> str: + """Get a mail cover letter report header with the provided titles and subject.""" + template = Config().get_cover_header_template() + if template: + rep_template = template.replace(HEADER_TITLE_REPLACE, title).replace(HEADER_SUBTITLE_REPLACE, subtitle) + return rep_template.replace(HEADER_SUBJECT_REPLACE, subject) + return None + + +def get_footer_data(footer_text: str, mail: bool = False) -> str: """Get report footer with the provided text.""" + if mail: + return get_mail_footer_data(footer_text) template = Config().get_footer_template() if template: return template.replace(FOOTER_TEXT_REPLACE, footer_text) return None -def get_report_meta_data() -> dict: +def get_mail_footer_data(footer_text: str) -> str: + """Get mail report footer with the provided text.""" + template = Config().get_mail_footer_template() + if template: + return template.replace(FOOTER_TEXT_REPLACE, footer_text) + return None + + +def get_cover_footer_data(footer_text: str) -> str: + """Get mail cover letter report footer with the provided text.""" + template = Config().get_cover_footer_template() + if template: + return template.replace(FOOTER_TEXT_REPLACE, footer_text) + return None + + +def get_report_meta_data(report_type: str = '') -> dict: """Get gotenberg report configuration data.""" - return copy.deepcopy(REPORT_META_DATA) + if not report_type or report_type not in (ReportTypes.MHR_REGISTRATION, ReportTypes.MHR_COVER): + return copy.deepcopy(REPORT_META_DATA) + data = copy.deepcopy(REPORT_META_DATA) + data['marginTop'] = MARGIN_TOP_REG_REPORT + return data -def get_report_files(request_data: dict, report_type: str) -> dict: +def get_report_files(request_data: dict, report_type: str, mail: bool = False) -> dict: """Get gotenberg report generation source file data.""" files = copy.deepcopy(REPORT_FILES) files['index.html'] = get_html_from_data(request_data) - header_text = '' footer_text = '' if report_type in (ReportTypes.SEARCH_BODY_REPORT, ReportTypes.SEARCH_DETAIL_REPORT, ReportTypes.SEARCH_TOC_REPORT, + ReportTypes.MHR_COVER, ReportTypes.MHR_REGISTRATION): - header_text = request_data['templateVars'].get('meta_title', '') + title_text = request_data['templateVars'].get('meta_title', '') + subtitle_text = request_data['templateVars'].get('meta_subtitle', '') footer_text = request_data['templateVars'].get('footer_content', '') - - files['header.html'] = get_header_data(header_text) - files['footer.html'] = get_footer_data(footer_text) + if report_type in (ReportTypes.MHR_REGISTRATION, ReportTypes.MHR_COVER): + subject_text = request_data['templateVars'].get('meta_subject', '') + if report_type == ReportTypes.MHR_COVER: + files['header.html'] = get_cover_header_data(title_text, subtitle_text, subject_text) + else: + files['header.html'] = get_reg_header_data(title_text, subtitle_text, subject_text, mail) + else: + files['header.html'] = get_header_data(title_text, subtitle_text) + if report_type == ReportTypes.MHR_COVER: + files['footer.html'] = get_cover_footer_data(footer_text) + else: + files['footer.html'] = get_footer_data(footer_text, mail) return files @@ -148,3 +284,68 @@ def update_toc_page_numbers(json_data, reg_pdf_data): select['pageNumber'] = (i + 1) break return json_data + + +def set_cover(report_data): # pylint: disable=too-many-branches, too-many-statements + """Add cover page report data. Cover page envelope window lines up to a maximum of 4.""" + cover_info = {} + if report_data.get('submittingParty'): + party = report_data.get('submittingParty') + name = '' + line1: str = '' + line2: str = '' + line3: str = '' + line4: str = '' + address = party['address'] + country = address.get('country', '') + region = address.get('region', '') + if 'businessName' in party: + name = party['businessName'] + elif 'personName' in party: + name = party['personName']['first'] + ' ' + party['personName']['last'] + if name: + line1 = name + if len(line1) > 40: + line1 = line1[0:40] + if country == 'CA': + postal_code: str = address.get('postalCode', '') + postal_code = postal_code.replace('-', ' ') + if len(postal_code) == 6: + line4 = region + '\n' + postal_code[0:3] + ' ' + postal_code[3:] + else: + line4 = region + '\n' + postal_code + else: + line4 = region + ' ' + address.get('postalCode', '') + + if (len(address['city']) + len(line4)) < 40: + line4 = address['city'] + ' ' + line4 + else: + line3 = address['city'] + if 'street' in address: + street = address['street'] + if not line2: + line2 = street + if len(street) > 40 and line3 == '': + line3 = street[40:80] + line2 = street[0:40] + else: + line3 = street + if not line3 and 'streetAdditional' in address: + line3 = address['streetAdditional'] + if line2 and len(line2) > 40: + line2 = line2[0:40] + if line3 and len(line3) > 40: + line3 = line3[0:40] + if country != 'CA': + if not line3: + line3 = line4 + line4 = country + else: + line4 = line4 + ' ' + country + cover_info['line1'] = line1.strip() + if line2: + cover_info['line2'] = line2.strip() + if line3: + cover_info['line3'] = line3.strip() + cover_info['line4'] = line4.strip() + return cover_info diff --git a/mhr_api/src/mhr_api/resources/utils.py b/mhr_api/src/mhr_api/resources/utils.py index 2c5ed648c..b0c16f72f 100644 --- a/mhr_api/src/mhr_api/resources/utils.py +++ b/mhr_api/src/mhr_api/resources/utils.py @@ -241,3 +241,14 @@ def validate_registration(json_data, is_staff: bool = False): """Perform non-schema extra validation on a non-financing registrations.""" error_msg = registration_validator.validate_registration(json_data, is_staff) return error_msg + + +def valid_api_key(request) -> bool: + """Verify the callback request api key is valid.""" + key = get_apikey(request) + if not key: + return False + apikey = current_app.config.get('SUBSCRIPTION_API_KEY') + if not apikey: + return True + return key == apikey diff --git a/mhr_api/src/mhr_api/resources/v1/registration_report_callback.py b/mhr_api/src/mhr_api/resources/v1/registration_report_callback.py old mode 100644 new mode 100755 index aae76c069..a1d30a0b3 --- a/mhr_api/src/mhr_api/resources/v1/registration_report_callback.py +++ b/mhr_api/src/mhr_api/resources/v1/registration_report_callback.py @@ -16,7 +16,7 @@ from http import HTTPStatus # import requests -from flask import Blueprint, current_app # , jsonify, request +from flask import Blueprint, current_app, request from flask_cors import cross_origin from mhr_api.exceptions import DatabaseException @@ -50,6 +50,11 @@ def post_registration_report_callback(registration_id: str): # pylint: disable= current_app.logger.info(f'Registration report callback starting id={registration_id}.') if registration_id is None: return resource_utils.path_param_error_response('registration ID') + + # Authenticate with request api key + if not resource_utils.valid_api_key(request): + return resource_utils.unauthorized_error_response('MHR registration report callback') + # If exceeded max retries we're done. event_count: int = 0 events = EventTracking.find_by_key_id_type(registration_id, diff --git a/mhr_api/src/mhr_api/resources/v1/search_report_callback.py b/mhr_api/src/mhr_api/resources/v1/search_report_callback.py index 2ca574a09..dda2f160b 100644 --- a/mhr_api/src/mhr_api/resources/v1/search_report_callback.py +++ b/mhr_api/src/mhr_api/resources/v1/search_report_callback.py @@ -17,7 +17,7 @@ import json # import requests -from flask import Blueprint, current_app # , jsonify, request +from flask import Blueprint, current_app, request from flask_cors import cross_origin from mhr_api.exceptions import DatabaseException @@ -56,6 +56,10 @@ def post_search_report_callback( # pylint: disable=too-many-branches,too-many-l if search_id is None: return resource_utils.path_param_error_response('search ID') + # Authenticate with request api key + if not resource_utils.valid_api_key(request): + return resource_utils.unauthorized_error_response('MHR search report callback') + # If exceeded max retries we're done. event_count: int = 0 events = EventTracking.find_by_key_id_type(search_id, EventTracking.EventTrackingTypes.SEARCH_REPORT) diff --git a/mhr_api/src/mhr_api/version.py b/mhr_api/src/mhr_api/version.py index 9f9802e5f..e07072b62 100644 --- a/mhr_api/src/mhr_api/version.py +++ b/mhr_api/src/mhr_api/version.py @@ -22,4 +22,4 @@ Development release segment: .devN """ -__version__ = '1.0.6' # pylint: disable=invalid-name +__version__ = '1.0.7' # pylint: disable=invalid-name diff --git a/mhr_api/tests/unit/models/test_mhr_registration.py b/mhr_api/tests/unit/models/test_mhr_registration.py old mode 100644 new mode 100755 index cf7c19bd9..e9f9a2fd6 --- a/mhr_api/tests/unit/models/test_mhr_registration.py +++ b/mhr_api/tests/unit/models/test_mhr_registration.py @@ -127,6 +127,12 @@ def test_find_by_id(session, reg_id, has_results, legacy): assert report_json.get('notes') for note in report_json.get('notes'): assert note['documentDescription'] + registration.mail_version = True + report_json = registration.new_registration_json + # current_app.logger.debug(report_json) + assert report_json.get('documentId') + assert report_json.get('documentRegistrationId') + assert report_json.get('documentDescription') else: assert not registration diff --git a/mhr_api/tests/unit/reports/data/registration-common-example.json b/mhr_api/tests/unit/reports/data/registration-common-example.json old mode 100644 new mode 100755 index 5a41b836e..63ff101f6 --- a/mhr_api/tests/unit/reports/data/registration-common-example.json +++ b/mhr_api/tests/unit/reports/data/registration-common-example.json @@ -28,7 +28,7 @@ }, "documentId": "UT-00007", "location": { - "additionalDescription": "SPALLUMCHEEN INDIAN RESERVE NO. 2", + "additionalDescription": "", "address": { "city": "VICTORIA", "country": "CA", @@ -37,24 +37,24 @@ "street": "940 BLANSHARD STREET" }, "block": "", - "dealerName": "NOR-TEC DESIGN GROUP LTD.", + "dealerName": "", "districtLot": "", "exceptionPlan": "", - "landDistrict": "PEACE RIVER", + "landDistrict": "", "leaveProvince": false, - "lot": "3", + "lot": "", "meridian": "", "pad": "20", - "parcel": "A (69860M)", + "parcel": "", "parkName": "HIDDEN VALLEY TRAILER COURT", "partOf": "", - "pidNumber": "011625490", + "pidNumber": "", "plan": "", - "range": "16", - "section": "34", + "range": "", + "section": "", "status": "ACTIVE", "taxCertificate": false, - "township": "78" + "township": "" }, "mhrNumber": "150075", "ownerGroups": [ diff --git a/mhr_api/tests/unit/reports/data/registration-joint-example.json b/mhr_api/tests/unit/reports/data/registration-joint-example.json new file mode 100644 index 000000000..771593a2e --- /dev/null +++ b/mhr_api/tests/unit/reports/data/registration-joint-example.json @@ -0,0 +1,121 @@ +{ + "attentionReference": "TESTING", + "clientReferenceId": "UT-MHREG0007 ", + "createDateTime": "2022-08-17T12:22:12+00:00", + "declaredValue": 0, + "description": { + "baseInformation": { + "make": "WATSON IND. (ALTA) DUCHESS", + "model": "", + "year": "2018" + }, + "csaNumber": "786356", + "csaStandard": "Z240", + "engineerName": "Dave Smith ENG. LTD.", + "engineerDate": "2018-02-22T07", + "manufacturer": "STARLINE", + "otherRemarks": "BC SAFETY AUTHORITY #339556, PERMIT# EL-721296-2018", + "rebuiltRemarks": "REBUILT AS A DOUBLE WIDE, MAKE/MODEL CUSTOM, YEAR 1995", + "sectionCount": 1, + "sections": [ + { + "lengthFeet": 52, + "lengthInches": 0, + "serialNumber": "52D70556", + "widthFeet": 12, + "widthInches": 0 + } + ] + }, + "documentId": "UT-00007", + "location": { + "additionalDescription": "LEGAL SUBDIVISION 8", + "address": { + "city": "VICTORIA", + "country": "CA", + "postalCode": "", + "region": "BC", + "street": "940 BLANSHARD STREET" + }, + "block": "87", + "dealerName": "", + "districtLot": "1290", + "exceptionPlan": "H14991, B7426 & KAP47383", + "landDistrict": "PEACE RIVER", + "leaveProvince": false, + "lot": "3", + "meridian": "W6M", + "pad": "", + "parcel": "A (69860M)", + "parkName": "", + "partOf": "WEST 1/2", + "pidNumber": "011625490", + "plan": "39062", + "range": "16", + "section": "34", + "status": "ACTIVE", + "taxCertificate": false, + "township": "78" + }, + "mhrNumber": "150075", + "ownerGroups": [ + { + "groupId": 2, + "interest": "", + "interestNumerator": 0, + "owners": [ + { + "address": { + "city": "LANGLEY", + "country": "CA", + "postalCode": "V3A 6H4", + "region": "BC", + "street": "6665 238TH STREET" + }, + "individualName": { + "first": "MARY-ANNE", + "last": "BICKNELL" + }, + "suffix": "STEPHENIE", + "phoneNumber": "604462027", + "type": "JOINT" + }, + { + "address": { + "city": "LANGLEY", + "country": "CA", + "postalCode": "V3A 6H4", + "region": "BC", + "street": "6665 238TH STREET" + }, + "individualName": { + "first": "JOHN", + "last": "CONNOLLY", + "middle": "STEVEN" + }, + "phoneNumber": "604462027", + "type": "JOINT" + } + ], + "status": "ACTIVE", + "tenancySpecified": true, + "type": "JOINT" + } + ], + "payment": { + "invoiceId": "21665", + "receipt": "/api/v1/payment-requests/21665/receipts" + }, + "status": "ACTIVE", + "submittingParty": { + "address": { + "city": "VICTORIA", + "country": "CA", + "postalCode": "V8W 2V8", + "region": "BC", + "street": "222 SUMMER STREET" + }, + "businessName": "ABC SEARCHING COMPANY", + "emailAddress": "bsmith@abc-search.com" + } +} diff --git a/mhr_api/tests/unit/reports/data/registration-sole-example.json b/mhr_api/tests/unit/reports/data/registration-sole-example.json old mode 100644 new mode 100755 index 4de9c939d..94cea27ac --- a/mhr_api/tests/unit/reports/data/registration-sole-example.json +++ b/mhr_api/tests/unit/reports/data/registration-sole-example.json @@ -13,8 +13,8 @@ "csaStandard": "Z240", "engineerName": "", "manufacturer": "STARLINE", - "otherRemarks": "BC SAFETY AUTHORITY #339556, PERMIT# EL-721296-2018", - "rebuiltRemarks": "REBUILT AS A DOUBLE WIDE, MAKE/MODEL CUSTOM, YEAR 1995", + "otherRemarks": "", + "rebuiltRemarks": "", "sectionCount": 1, "sections": [ { @@ -26,9 +26,11 @@ } ] }, - "documentId": "UT-00007", + "documentId": "40590422", + "documentRegistrationId": "00022911", + "documentDescription": "Register New Unit", "location": { - "additionalDescription": "SPALLUMCHEEN INDIAN RESERVE NO. 2", + "additionalDescription": "", "address": { "city": "VICTORIA", "country": "CA", @@ -40,21 +42,21 @@ "dealerName": "NOR-TEC DESIGN GROUP LTD.", "districtLot": "", "exceptionPlan": "", - "landDistrict": "PEACE RIVER", + "landDistrict": "", "leaveProvince": false, - "lot": "3", + "lot": "", "meridian": "", - "pad": "20", - "parcel": "A (69860M)", - "parkName": "HIDDEN VALLEY TRAILER COURT", + "pad": "", + "parcel": "", + "parkName": "", "partOf": "", - "pidNumber": "011625490", + "pidNumber": "", "plan": "", - "range": "16", - "section": "34", + "range": "", + "section": "", "status": "ACTIVE", "taxCertificate": false, - "township": "78" + "township": "" }, "mhrNumber": "150075", "ownerGroups": [ diff --git a/mhr_api/tests/unit/reports/test_report_registration_v2.py b/mhr_api/tests/unit/reports/test_report_registration_v2.py old mode 100644 new mode 100755 index 2f9b81df9..4122d307c --- a/mhr_api/tests/unit/reports/test_report_registration_v2.py +++ b/mhr_api/tests/unit/reports/test_report_registration_v2.py @@ -29,6 +29,10 @@ REGISTRATON_SOLE_PDFFILE = 'tests/unit/reports/data/registration-sole-example.pdf' REGISTRATON_COMMON_DATAFILE = 'tests/unit/reports/data/registration-common-example.json' REGISTRATON_COMMON_PDFFILE = 'tests/unit/reports/data/registration-common-example.pdf' +REGISTRATON_JOINT_DATAFILE = 'tests/unit/reports/data/registration-joint-example.json' +REGISTRATON_JOINT_PDFFILE = 'tests/unit/reports/data/registration-joint-example.pdf' +REGISTRATON_MAIL_PDFFILE = 'tests/unit/reports/data/registration-mail-example.pdf' +REGISTRATON_COVER_PDFFILE = 'tests/unit/reports/data/registration-cover-example.pdf' REPORT_VERSION_V2 = '2' @@ -58,6 +62,45 @@ def test_registration_common(session, client, jwt): check_response(content, status, REGISTRATON_COMMON_PDFFILE) +def test_registration_joint(session, client, jwt): + """Assert that generation of an owner joint tenants type report is as expected.""" + # setup + if is_report_v2(): + json_data = get_json_from_file(REGISTRATON_JOINT_DATAFILE) + report = Report(json_data, 'PS12345', ReportTypes.MHR_REGISTRATION, 'Account Name') + # test + content, status, headers = report.get_pdf() + assert headers + # verify + check_response(content, status, REGISTRATON_JOINT_PDFFILE) + + +def test_cover_registration(session, client, jwt): + """Assert that generation of a mail cover page report is as expected.""" + # setup + if is_report_v2(): + json_data = get_json_from_file(REGISTRATON_SOLE_DATAFILE) + report = Report(json_data, 'PS12345', ReportTypes.MHR_COVER, 'Account Name') + # test + content, status, headers = report.get_pdf() + assert headers + # verify + check_response(content, status, REGISTRATON_COVER_PDFFILE) + + +def test_mail_registration(session, client, jwt): + """Assert that generation of a mail report is as expected.""" + # setup + if is_report_v2(): + json_data = get_json_from_file(REGISTRATON_SOLE_DATAFILE) + report = Report(json_data, 'PS12345', ReportTypes.MHR_REGISTRATION_MAIL, 'Account Name') + # test + content, status, headers = report.get_pdf() + assert headers + # verify + check_response(content, status, REGISTRATON_MAIL_PDFFILE) + + def get_json_from_file(data_file: str): """Get json data from report data file.""" text_data = None