diff --git a/CHANGELOG.md b/CHANGELOG.md index e929fb37e0a..ff156ef4c59 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -18,12 +18,18 @@ - Reoder the sytem user add/edit field for surname to be first, also change labels from `Last name` to `User's surname` and lastly remove the NID question from the form [#6830](https://github.com/opencrvs/opencrvs-core/issues/6830) - Corrected the total amount displayed for _certification_ and _correction_ fees on the Performance Page, ensuring accurate fee tracking across certification and correction sequences. [#7793](https://github.com/opencrvs/opencrvs-core/issues/7793) - Auth now allows registrar's token to be exchanged for a new token that strictly allows confirming or rejecting a specific record. Core now passes this token to country configuration instead of the registrar's token [#7728](https://github.com/opencrvs/opencrvs-core/issues/7728) [#7849](https://github.com/opencrvs/opencrvs-core/issues/7849) +- **Template Selection for Certified Copies**: Added support for multiple certificate templates for each event (birth, death, marriage). Users can now select a template during the certificate issuance process. +- **Template-based Payment Configuration**: Implemented payment differentiation based on the selected certificate template, ensuring the correct amount is charged. +- **Template Action Tracking**: Each template printed is tracked in the history table, showing which specific template was used. +- **Template Selection Dropdown**: Updated print workflow to include a dropdown menu for template selection when issuing a certificate. +- Auth now allows exchanging user's token for a new record-specific token [#7728](https://github.com/opencrvs/opencrvs-core/issues/7728) - A new GraphQL mutation `upsertRegistrationIdentifier` is added to allow updating the patient identifiers of a registration record such as NID [#8034](https://github.com/opencrvs/opencrvs-core/pull/8034) ### Improvements - Auth token, ip address, remote address redacted from server log - **Align Patient data model with FHIR**: Previously we were using `string[]` for `Patient.name.family` field instead of `string` as mentioned in the FHIR standard. We've now aligned the field with the standard. +- **Certificate Fetching**: Removed certificates from the database, allowing them to be fetched directly from the country configuration via a simplified API endpoint. ### Deprecated diff --git a/packages/client/graphql.schema.json b/packages/client/graphql.schema.json index 7adc9cbfafa..f18e9d1a158 100644 --- a/packages/client/graphql.schema.json +++ b/packages/client/graphql.schema.json @@ -3648,39 +3648,376 @@ "deprecationReason": null }, { - "name": "data", + "name": "hasShowedVerifiedDocument", + "description": null, + "args": [], + "type": { + "kind": "SCALAR", + "name": "Boolean", + "ofType": null + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "payments", + "description": null, + "args": [], + "type": { + "kind": "LIST", + "name": null, + "ofType": { + "kind": "OBJECT", + "name": "Payment", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "templateConfig", + "description": null, + "args": [], + "type": { + "kind": "OBJECT", + "name": "CertificateConfigData", + "ofType": null + }, + "isDeprecated": false, + "deprecationReason": null + } + ], + "inputFields": null, + "interfaces": [], + "enumValues": null, + "possibleTypes": null + }, + { + "kind": "OBJECT", + "name": "CertificateConfigData", + "description": null, + "fields": [ + { + "name": "event", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "fee", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "OBJECT", + "name": "CertificateFee", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "id", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "label", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "OBJECT", + "name": "CertificateLabel", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "lateRegistrationTarget", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Int", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "printInAdvance", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Boolean", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "registrationTarget", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Int", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "svgUrl", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + } + ], + "inputFields": null, + "interfaces": [], + "enumValues": null, + "possibleTypes": null + }, + { + "kind": "INPUT_OBJECT", + "name": "CertificateConfigDataInput", + "description": null, + "fields": null, + "inputFields": [ + { + "name": "event", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "fee", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "INPUT_OBJECT", + "name": "CertificateFeeInput", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "id", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "label", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "INPUT_OBJECT", + "name": "CertificateLabelInput", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "lateRegistrationTarget", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Int", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "printInAdvance", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Boolean", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "registrationTarget", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Int", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "svgUrl", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + } + ], + "interfaces": null, + "enumValues": null, + "possibleTypes": null + }, + { + "kind": "OBJECT", + "name": "CertificateFee", + "description": null, + "fields": [ + { + "name": "delayed", "description": null, "args": [], "type": { - "kind": "SCALAR", - "name": "String", - "ofType": null + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Float", + "ofType": null + } }, "isDeprecated": false, "deprecationReason": null }, { - "name": "hasShowedVerifiedDocument", + "name": "late", "description": null, "args": [], "type": { - "kind": "SCALAR", - "name": "Boolean", - "ofType": null + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Float", + "ofType": null + } }, "isDeprecated": false, "deprecationReason": null }, { - "name": "payments", + "name": "onTime", "description": null, "args": [], "type": { - "kind": "LIST", + "kind": "NON_NULL", "name": null, "ofType": { - "kind": "OBJECT", - "name": "Payment", + "kind": "SCALAR", + "name": "Float", "ofType": null } }, @@ -3695,28 +4032,75 @@ }, { "kind": "INPUT_OBJECT", - "name": "CertificateInput", + "name": "CertificateFeeInput", "description": null, "fields": null, "inputFields": [ { - "name": "collector", + "name": "delayed", "description": null, "type": { - "kind": "INPUT_OBJECT", - "name": "RelatedPersonInput", - "ofType": null + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Float", + "ofType": null + } }, "defaultValue": null, "isDeprecated": false, "deprecationReason": null }, { - "name": "data", + "name": "late", "description": null, "type": { - "kind": "SCALAR", - "name": "String", + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Float", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "onTime", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "Float", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + } + ], + "interfaces": null, + "enumValues": null, + "possibleTypes": null + }, + { + "kind": "INPUT_OBJECT", + "name": "CertificateInput", + "description": null, + "fields": null, + "inputFields": [ + { + "name": "collector", + "description": null, + "type": { + "kind": "INPUT_OBJECT", + "name": "RelatedPersonInput", "ofType": null }, "defaultValue": null, @@ -3750,6 +4134,136 @@ "defaultValue": null, "isDeprecated": false, "deprecationReason": null + }, + { + "name": "templateConfig", + "description": null, + "type": { + "kind": "INPUT_OBJECT", + "name": "CertificateConfigDataInput", + "ofType": null + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + } + ], + "interfaces": null, + "enumValues": null, + "possibleTypes": null + }, + { + "kind": "OBJECT", + "name": "CertificateLabel", + "description": null, + "fields": [ + { + "name": "defaultMessage", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "description", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "id", + "description": null, + "args": [], + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "isDeprecated": false, + "deprecationReason": null + } + ], + "inputFields": null, + "interfaces": [], + "enumValues": null, + "possibleTypes": null + }, + { + "kind": "INPUT_OBJECT", + "name": "CertificateLabelInput", + "description": null, + "fields": null, + "inputFields": [ + { + "name": "defaultMessage", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "description", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null + }, + { + "name": "id", + "description": null, + "type": { + "kind": "NON_NULL", + "name": null, + "ofType": { + "kind": "SCALAR", + "name": "String", + "ofType": null + } + }, + "defaultValue": null, + "isDeprecated": false, + "deprecationReason": null } ], "interfaces": null, @@ -7428,6 +7942,18 @@ "isDeprecated": false, "deprecationReason": null }, + { + "name": "templateConfig", + "description": null, + "args": [], + "type": { + "kind": "OBJECT", + "name": "CertificateConfigData", + "ofType": null + }, + "isDeprecated": false, + "deprecationReason": null + }, { "name": "user", "description": null, diff --git a/packages/client/src/declarations/index.ts b/packages/client/src/declarations/index.ts index 02400533891..90212662d6b 100644 --- a/packages/client/src/declarations/index.ts +++ b/packages/client/src/declarations/index.ts @@ -262,11 +262,14 @@ type RelationForCertificateCorrection = | 'CHILD' export type ICertificate = { - collector?: Partial<{ type: Relation | string }> + collector?: Partial<{ + type: Relation | string + certificateTemplateId: string + }> corrector?: Partial<{ type: RelationForCertificateCorrection | string }> hasShowedVerifiedDocument?: boolean payments?: Payment - data?: string + certificateTemplateId?: string } /* diff --git a/packages/client/src/declarations/submissionMiddleware.test.ts b/packages/client/src/declarations/submissionMiddleware.test.ts index 7a2372f2e6d..420d63993be 100644 --- a/packages/client/src/declarations/submissionMiddleware.test.ts +++ b/packages/client/src/declarations/submissionMiddleware.test.ts @@ -14,6 +14,7 @@ import { offlineDataReady } from '@client/offline/actions' import { createStore } from '@client/store' import { ACTION_STATUS_MAP, + mockBirthRegistrationSectionData, mockDeclarationData, mockOfflineDataDispatch } from '@client/tests/util' @@ -140,30 +141,7 @@ describe('Submission middleware', () => { } it(`should handle ${ACTION_STATUS_MAP[submissionAction]} ${event} declarations`, async () => { - mockDeclarationData.registration.certificates[0] = { - collector: { - relationship: 'OTHER', - affidavit: { - contentType: 'image/jpg', - data: '' - }, - individual: { - name: [{ firstNames: 'Doe', familyName: 'Jane', use: 'en' }], - identifier: [{ id: '123456', type: 'PASSPORT' }] - } - }, - hasShowedVerifiedDocument: true, - payments: [ - { - paymentId: '1234', - type: 'MANUAL', - amount: 50, - outcome: 'COMPLETED', - date: '2018-10-22' - } - ], - data: '' - } + mockDeclarationData.registration = mockBirthRegistrationSectionData const action = declarationReadyForStatusChange({ id: 'mockDeclaration', data: mockDeclarationData, diff --git a/packages/client/src/declarations/submissionMiddleware.ts b/packages/client/src/declarations/submissionMiddleware.ts index 879d84267e4..b7956d920db 100644 --- a/packages/client/src/declarations/submissionMiddleware.ts +++ b/packages/client/src/declarations/submissionMiddleware.ts @@ -306,12 +306,6 @@ export const submissionMiddleware: Middleware<{}, IStoreState> = details: graphqlPayload } }) - //delete data from certificates to identify event in workflow for markEventAsIssued - if (declaration.data.registration.certificates) { - delete ( - declaration.data.registration.certificates as ICertificate[] - )?.[0].data - } updateDeclaration(dispatch, { ...declaration, registrationStatus: RegStatus.Certified, diff --git a/packages/client/src/forms/certificate/fieldDefinitions/collectorSection.ts b/packages/client/src/forms/certificate/fieldDefinitions/collectorSection.ts index a41156755f8..a5010532813 100644 --- a/packages/client/src/forms/certificate/fieldDefinitions/collectorSection.ts +++ b/packages/client/src/forms/certificate/fieldDefinitions/collectorSection.ts @@ -13,8 +13,11 @@ import { CHECKBOX_GROUP, FIELD_WITH_DYNAMIC_DEFINITIONS, identityTypeMapper, + IFormData, IFormField, + IFormFieldValue, IFormSection, + IFormSectionData, IFormSectionGroup, IRadioGroupFormField, IRadioOption, @@ -36,6 +39,8 @@ import { identityHelperTextMapper, identityNameMapper } from './messages' import { EventType } from '@client/utils/gateway' import { IDeclaration } from '@client/declarations' import { issueMessages } from '@client/i18n/messages/issueCertificate' +import { ICertificateData } from '@client/utils/referenceApi' +import { IOfflineData } from '@client/offline/reducer' interface INameField { firstNamesField: string @@ -974,7 +979,24 @@ const affidavitCertCollectorGroup: IFormSectionGroup = { label: certificateMessages.noLabel, required: false, initialValue: [], - validator: [], + validator: [ + ( + value: IFormFieldValue, + drafts?: IFormData, + offlineCountryConfig?: IOfflineData, + form?: IFormSectionData + ) => + form && + !( + (form['noAffidavitAgreement'] as Array)?.length || + form['affidavitFile'] + ) + ? { + message: + certificateMessages.certificateOtherCollectorAffidavitError + } + : undefined + ], options: [ { value: 'AFFIDAVIT', @@ -1012,7 +1034,8 @@ const marriageIssueCollectorFormOptions = [ ] function getCertCollectorGroupForEvent( - declaration: IDeclaration + declaration: IDeclaration, + certificates: ICertificateData[] ): IFormSectionGroup { const informant = (declaration.data.informant.otherInformantType || declaration.data.informant.informantType) as string @@ -1039,12 +1062,30 @@ function getCertCollectorGroupForEvent( birthCertCollectorOptions, marriageCertCollectorOptions ) - + const certificateTemplateOptions = + certificates + .filter((x) => x.event === declaration.event) + .map((x) => ({ label: x.label, value: x.id })) || [] return { id: 'certCollector', title: certificateMessages.whoToCollect, - error: certificateMessages.certificateCollectorError, fields: [ + { + name: 'certificateTemplateId', + type: 'SELECT_WITH_OPTIONS', + label: certificateMessages.certificateTemplateSelectLabel, + required: true, + validator: [ + (value: IFormFieldValue) => { + return !value + ? { + message: certificateMessages.certificateCollectorTemplateError + } + : undefined + } + ], + options: certificateTemplateOptions + }, { name: 'type', type: RADIO_GROUP, @@ -1053,7 +1094,15 @@ function getCertCollectorGroupForEvent( hideHeader: true, required: true, initialValue: '', - validator: [], + validator: [ + (value: IFormFieldValue) => { + return !value + ? { + message: certificateMessages.certificateCollectorError + } + : undefined + } + ], options: finalOptions } ] @@ -1061,7 +1110,8 @@ function getCertCollectorGroupForEvent( } export function getCertificateCollectorFormSection( - declaration: IDeclaration + declaration: IDeclaration, + certificates: ICertificateData[] ): IFormSection { return { id: CertificateSection.Collector, @@ -1069,7 +1119,7 @@ export function getCertificateCollectorFormSection( name: certificateMessages.printCertificate, title: certificateMessages.certificateCollectionTitle, groups: [ - getCertCollectorGroupForEvent(declaration), + getCertCollectorGroupForEvent(declaration, certificates), otherCertCollectorFormGroup(declaration.event), affidavitCertCollectorGroup ] diff --git a/packages/client/src/forms/index.ts b/packages/client/src/forms/index.ts index 2218acf6489..65a84947250 100644 --- a/packages/client/src/forms/index.ts +++ b/packages/client/src/forms/index.ts @@ -1304,5 +1304,5 @@ export interface ICertificate { collector?: IFormSectionData hasShowedVerifiedDocument?: boolean payments?: Payment[] - data?: string + certificateTemplateId?: string } diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.test.ts b/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.test.ts index fcedb64513c..d54c2b672b9 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.test.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.test.ts @@ -32,46 +32,38 @@ describe('Birth registration mutation mapping related tests', () => { expect(transformedData.registration.trackingId).toEqual('BDSS0SE') expect(transformedData.registration.certificates).toEqual([ { + certificateTemplateId: 'birth-certificate', + hasShowedVerifiedDocument: true, collector: { relationship: 'OTHER', otherRelationship: 'Uncle', - name: [ - { - use: 'en', - firstNames: 'Mushraful', - familyName: 'Hoque' - } - ], - identifier: [ - { - id: '123456789', - type: 'PASSPORT' - } - ], - affidavit: [ - { - contentType: 'abc', - data: 'BASE64 data' - } - ] - }, - hasShowedVerifiedDocument: true + name: [{ use: 'en', firstNames: 'Mushraful', familyName: 'Hoque' }], + identifier: [{ id: '123456789', type: 'PASSPORT' }], + affidavit: [{ contentType: 'abc', data: 'BASE64 data' }] + } } ]) }) - it('Test certificate mapping without any data', () => { + it('Test certificate mapping template config data', () => { const transformedData: TransformedData = { registration: {} } + const mockBirthDeclaration = cloneDeep({ + ...mockDeclarationData, + registration: { + ...mockDeclarationData.registration, + certificates: [{}] + } + }) setBirthRegistrationSectionTransformer( transformedData, - mockDeclarationData, + mockBirthDeclaration, 'registration' ) expect(transformedData.registration).toBeDefined() expect(transformedData.registration.registrationNumber).toEqual( '201908122365BDSS0SE1' ) - expect(transformedData.registration.certificates).toEqual([{}]) + expect(transformedData.registration.certificates).toEqual([]) }) }) diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.ts b/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.ts index 4328580a695..13caa4efc6e 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/birth/mutation/registration-mappings.ts @@ -44,11 +44,20 @@ export function setBirthRegistrationSectionTransformer( }) } - if (draftData[sectionId].certificates) { - transformCertificateData( - transformedData, - (draftData[sectionId].certificates as ICertificate[])[0], - sectionId - ) + const certificates: ICertificate[] = draftData[sectionId] + .certificates as ICertificate[] + if ( + Array.isArray(certificates) && + certificates.length && + !draftData[sectionId].correction + ) { + const updatedCertificates = transformCertificateData(certificates.slice(-1)) + transformedData[sectionId].certificates = + updatedCertificates.length > 0 && + Object.keys(updatedCertificates[0]).length > 0 && + updatedCertificates[0].collector // making sure we are not sending empty object as certificate + ? updatedCertificates + : [] } + return transformedData } diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/birth/query/registration-mappings.ts b/packages/client/src/forms/register/mappings/event-specific-fields/birth/query/registration-mappings.ts index b4b14187824..be9da58de59 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/birth/query/registration-mappings.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/birth/query/registration-mappings.ts @@ -33,4 +33,17 @@ export function getBirthRegistrationSectionTransformer( if (queryData[sectionId].status) { transformStatusData(transformedData, queryData[sectionId].status, sectionId) } + + if ( + Array.isArray(queryData[sectionId].certificates) && + queryData[sectionId].certificates.length > 0 + ) { + const currentCertificate = + queryData[sectionId].certificates[ + queryData[sectionId].certificates.length - 1 + ] + if (currentCertificate?.collector?.relationship === 'PRINT_IN_ADVANCE') { + transformedData[sectionId].certificates = [currentCertificate] + } + } } diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/deceased-mappings.test.ts b/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/deceased-mappings.test.ts index 26fbe52da1b..8da67f91387 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/deceased-mappings.test.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/deceased-mappings.test.ts @@ -32,10 +32,9 @@ describe('Death registration mutation mapping related tests', () => { ) expect(transformedData.registration.certificates).toEqual([ { - collector: { - relationship: 'MOTHER' - }, - hasShowedVerifiedDocument: true + hasShowedVerifiedDocument: true, + certificateTemplateId: 'death-certificate', + collector: { relationship: 'MOTHER' } } ]) }) @@ -54,24 +53,14 @@ describe('Death registration mutation mapping related tests', () => { expect(transformedData.registration.trackingId).toEqual('DDSS0SE') expect(transformedData.registration.certificates).toEqual([ { + hasShowedVerifiedDocument: true, + certificateTemplateId: 'death-certificate', collector: { relationship: 'OTHER', otherRelationship: 'Uncle', - name: [ - { - use: 'en', - firstNames: 'Mushraful', - familyName: 'Hoque' - } - ], - identifier: [ - { - id: '123456789', - type: 'PASSPORT' - } - ] - }, - hasShowedVerifiedDocument: true + name: [{ use: 'en', firstNames: 'Mushraful', familyName: 'Hoque' }], + identifier: [{ id: '123456789', type: 'PASSPORT' }] + } } ]) }) diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/registration-mappings.ts b/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/registration-mappings.ts index 2741b81e4c8..e61a9452f63 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/registration-mappings.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/death/mutation/registration-mappings.ts @@ -51,14 +51,23 @@ export function setDeathRegistrationSectionTransformer( }) } - if (draftData.registration.certificates) { - transformCertificateData( - transformedData, - (draftData.registration.certificates as ICertificate[])[0], - 'registration' + const certificates: ICertificate[] = draftData[sectionId] + .certificates as ICertificate[] + if ( + Array.isArray(certificates) && + certificates.length && + !draftData[sectionId].correction + ) { + const updatedCertificates = transformCertificateData( + certificates.slice(-1) ) + transformedData[sectionId].certificates = + updatedCertificates.length > 0 && + Object.keys(updatedCertificates[0]).length > 0 && + updatedCertificates[0].collector // making sure we are not sending empty object as certificate + ? updatedCertificates + : [] } } - return transformedData } diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/death/query/registration-mappings.ts b/packages/client/src/forms/register/mappings/event-specific-fields/death/query/registration-mappings.ts index 8985caaf19c..35475161b45 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/death/query/registration-mappings.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/death/query/registration-mappings.ts @@ -16,37 +16,46 @@ import type { GQLRegWorkflow } from '@client/utils/gateway-deprecated-do-not-use export function getDeathRegistrationSectionTransformer( transformedData: IFormData, queryData: any, - _sectionId: string + sectionId: string ) { - if (!transformedData['registration']) { - transformedData['registration'] = {} + if (!transformedData[sectionId]) { + transformedData[sectionId] = {} } - if (queryData['registration'].id) { - transformedData['registration']._fhirID = queryData['registration'].id + if (queryData[sectionId].id) { + transformedData[sectionId]._fhirID = queryData[sectionId].id } - if (queryData['registration'].trackingId) { - transformedData['registration'].trackingId = - queryData['registration'].trackingId + if (queryData[sectionId].trackingId) { + transformedData[sectionId].trackingId = queryData[sectionId].trackingId } - if (queryData['registration'].registrationNumber) { - transformedData['registration'].registrationNumber = - queryData['registration'].registrationNumber + if (queryData[sectionId].registrationNumber) { + transformedData[sectionId].registrationNumber = + queryData[sectionId].registrationNumber } - if ( - queryData['registration'].type && - queryData['registration'].type === 'DEATH' - ) { - transformedData['registration'].type = EventType.Death + if (queryData[sectionId].type && queryData[sectionId].type === 'DEATH') { + transformedData[sectionId].type = EventType.Death } - if (queryData['registration'].status) { + if (queryData[sectionId].status) { transformStatusData( transformedData, - queryData['registration'].status as GQLRegWorkflow[], - 'registration' + queryData[sectionId].status as GQLRegWorkflow[], + sectionId ) } + + if ( + Array.isArray(queryData[sectionId].certificates) && + queryData[sectionId].certificates.length > 0 + ) { + const currentCertificate = + queryData[sectionId].certificates[ + queryData[sectionId].certificates.length - 1 + ] + if (currentCertificate?.collector?.relationship === 'PRINT_IN_ADVANCE') { + transformedData[sectionId].certificates = [currentCertificate] + } + } } diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/marriage/mutation/registration-mappings.ts b/packages/client/src/forms/register/mappings/event-specific-fields/marriage/mutation/registration-mappings.ts index fb9c8a5cbc5..e7c19c931b1 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/marriage/mutation/registration-mappings.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/marriage/mutation/registration-mappings.ts @@ -50,12 +50,23 @@ export function setMarriageRegistrationSectionTransformer( }) } - if (draftData[sectionId].certificates) { - transformCertificateData( - transformedData, - (draftData[sectionId].certificates as ICertificate[])[0], - sectionId + const certificates: ICertificate[] = draftData[sectionId] + .certificates as ICertificate[] + if ( + Array.isArray(certificates) && + certificates.length && + !draftData[sectionId].correction + ) { + const updatedCertificates = transformCertificateData( + certificates.slice(-1) ) + transformedData[sectionId].certificates = + updatedCertificates.length > 0 && + Object.keys(updatedCertificates[0]).length > 0 && + updatedCertificates[0].collector // making sure we are not sending empty object as certificate + ? updatedCertificates + : [] } } + return transformedData } diff --git a/packages/client/src/forms/register/mappings/event-specific-fields/marriage/query/registration-mappings.ts b/packages/client/src/forms/register/mappings/event-specific-fields/marriage/query/registration-mappings.ts index b729f08150f..b9b3dd66841 100644 --- a/packages/client/src/forms/register/mappings/event-specific-fields/marriage/query/registration-mappings.ts +++ b/packages/client/src/forms/register/mappings/event-specific-fields/marriage/query/registration-mappings.ts @@ -42,6 +42,19 @@ export function getMarriageRegistrationSectionTransformer( sectionId ) } + + if ( + Array.isArray(queryData[sectionId].certificates) && + queryData[sectionId].certificates.length > 0 + ) { + const currentCertificate = + queryData[sectionId].certificates[ + queryData[sectionId].certificates.length - 1 + ] + if (currentCertificate?.collector?.relationship === 'PRINT_IN_ADVANCE') { + transformedData[sectionId].certificates = [currentCertificate] + } + } } export function groomSignatureTransformer( diff --git a/packages/client/src/forms/register/mappings/mutation/utils.ts b/packages/client/src/forms/register/mappings/mutation/utils.ts index 802ca2fd848..21676c80d69 100644 --- a/packages/client/src/forms/register/mappings/mutation/utils.ts +++ b/packages/client/src/forms/register/mappings/mutation/utils.ts @@ -10,19 +10,33 @@ */ import type { GQLRelatedPersonInput } from '@client/utils/gateway-deprecated-do-not-use' -import { ICertificate, IFileValue, TransformedData } from '@client/forms' +import { ICertificate, IFileValue } from '@client/forms' import { omit } from 'lodash' -export function transformCertificateData( - transformedData: TransformedData, - certificateData: ICertificate, - sectionId: string -) { - transformedData[sectionId].certificates = [ +export function stripTypename(obj: any): any { + if (Array.isArray(obj)) { + return obj.map(stripTypename) + } else if (obj !== null && typeof obj === 'object') { + const newObj: any = {} + for (const key in obj) { + if (key !== '__typename' && Object.hasOwn(obj, key)) { + newObj[key] = stripTypename(obj[key]) + } + } + return newObj + } + return obj +} +export function transformCertificateData(certificates: ICertificate[]) { + const certificateData = stripTypename(certificates[0]) + + // Prepare the base certificate data + const updatedCertificates: ICertificate[] = [ { ...omit(certificateData, 'collector') } ] + // for collector mapping if (certificateData && certificateData.collector) { let collector: GQLRelatedPersonInput = {} @@ -58,6 +72,9 @@ export function transformCertificateData( } ] } - transformedData[sectionId].certificates[0].collector = collector + updatedCertificates[0].collector = collector as any } + + // Return the processed certificates array + return updatedCertificates } diff --git a/packages/client/src/i18n/messages/views/certificate.ts b/packages/client/src/i18n/messages/views/certificate.ts index 3c3601da743..3002b0a496e 100644 --- a/packages/client/src/i18n/messages/views/certificate.ts +++ b/packages/client/src/i18n/messages/views/certificate.ts @@ -14,6 +14,7 @@ interface ICertificateMessages extends Record { certificateCollectionTitle: MessageDescriptor addAnotherSignature: MessageDescriptor + certificateTemplateSelectLabel: MessageDescriptor certificateConfirmationTxt: MessageDescriptor certificateIsCorrect: MessageDescriptor certificateReceiptHeader: MessageDescriptor @@ -58,6 +59,7 @@ interface ICertificateMessages receiptPaidAmount: MessageDescriptor receiptService: MessageDescriptor selectSignature: MessageDescriptor + selectedCertificateTemplateLabel: MessageDescriptor service: MessageDescriptor amountDue: MessageDescriptor typeOfID: MessageDescriptor @@ -72,6 +74,7 @@ interface ICertificateMessages toastMessage: MessageDescriptor otherCollectorFormTitle: MessageDescriptor certificateCollectorError: MessageDescriptor + certificateCollectorTemplateError: MessageDescriptor certificateOtherCollectorInfoError: MessageDescriptor certificateOtherCollectorAffidavitFormTitle: MessageDescriptor certificateOtherCollectorAffidavitError: MessageDescriptor @@ -95,6 +98,11 @@ const messagesToDefine: ICertificateMessages = { description: 'The title of print certificate action', id: 'print.certificate.section.title' }, + certificateTemplateSelectLabel: { + defaultMessage: 'Type', + description: 'The title of select certificate template action', + id: 'certificate.selectTemplate' + }, certificateConfirmationTxt: { defaultMessage: 'Edit', description: 'Edit', @@ -330,6 +338,11 @@ const messagesToDefine: ICertificateMessages = { description: 'The label for choose signature select', id: 'print.certificate.selectSignature' }, + selectedCertificateTemplateLabel: { + defaultMessage: 'Selected certificate template', + description: 'The title of selected certificate template label', + id: 'certificate.selectedTemplate' + }, service: { defaultMessage: 'Service: Birth registration after {service, plural, =0 {0 month} one {1 month} other{{service} months}} of D.o.B.
Amount Due:', @@ -404,6 +417,11 @@ const messagesToDefine: ICertificateMessages = { description: 'Form level error for collector form', id: 'print.certificate.collector.form.error' }, + certificateCollectorTemplateError: { + defaultMessage: 'Please select certificate type', + description: 'Form level error for collector certificate template', + id: 'print.certificate.collector.form.error.template' + }, certificateOtherCollectorInfoError: { defaultMessage: 'Complete all the mandatory fields', description: 'Form level error for other collector information form', diff --git a/packages/client/src/offline/actions.ts b/packages/client/src/offline/actions.ts index 6dec4c81459..fbb126282c9 100644 --- a/packages/client/src/offline/actions.ts +++ b/packages/client/src/offline/actions.ts @@ -28,6 +28,7 @@ import { LoadConditionalsResponse, LoadFormsResponse, LoadHandlebarHelpersResponse, + ICertificateData, LoadValidatorsResponse } from '@client/utils/referenceApi' import { UserDetails } from '@client/utils/userUtils' @@ -110,23 +111,15 @@ type ApplicationConfigLoadedAction = { payload: IApplicationConfigResponse } -const CERTIFICATE_LOAD_FAILED = 'OFFLINE/CERTIFICATE_LOAD_FAILED' -type CertificateLoadFailedAction = { - type: typeof CERTIFICATE_LOAD_FAILED - payload: Error -} - -export const CERTIFICATE_CONFIGURATION_LOADED = - 'OFFLINE/CERTIFICATE_CONFIGURATION_LOADED' -type CertificateConfigurationLoadedAction = { - type: typeof CERTIFICATE_CONFIGURATION_LOADED - payload: CertificateConfiguration +export const CERTIFICATES_LOADED = 'OFFLINE/CERTIFICATES_LOADED' +type CertificatesLoadedAction = { + type: typeof CERTIFICATES_LOADED + payload: ICertificateData[] } -export const CERTIFICATE_CONFIGURATION_LOAD_FAILED = - 'OFFLINE/CERTIFICATE_CONFIGURATION_LOAD_FAILED' -type CertificateConfigurationLoadFailedAction = { - type: typeof CERTIFICATE_CONFIGURATION_LOAD_FAILED +export const CERTIFICATES_LOAD_FAILED = 'OFFLINE/CERTIFICATES_LOAD_FAILED' +type CertificatesLoadFailedAction = { + type: typeof CERTIFICATES_LOAD_FAILED payload: Error } @@ -267,17 +260,10 @@ export const configLoaded = ( payload: payload }) -export const certificateConfigurationLoaded = ( - payload: CertificateConfiguration -): CertificateConfigurationLoadedAction => ({ - type: CERTIFICATE_CONFIGURATION_LOADED, - payload -}) - -export const certificateConfigurationLoadFailed = ( - payload: CertificateConfigurationLoadFailedAction['payload'] -): CertificateConfigurationLoadFailedAction => ({ - type: CERTIFICATE_CONFIGURATION_LOAD_FAILED, +export const certificatesLoaded = ( + payload: ICertificateData[] +): CertificatesLoadedAction => ({ + type: CERTIFICATES_LOADED, payload }) @@ -345,12 +331,11 @@ export type Action = | ContentFailedAction | ContentLoadedAction | ApplicationConfigLoadedAction + | CertificatesLoadedAction + | CertificatesLoadFailedAction | ApplicationConfigAnonymousUserAction | ApplicationConfigFailedAction | ApplicationConfigUpdatedAction - | CertificateLoadFailedAction - | CertificateConfigurationLoadedAction - | CertificateConfigurationLoadFailedAction | UpdateOfflineSystemsAction | IFilterLocationsAction | ReturnType diff --git a/packages/client/src/offline/reducer.ts b/packages/client/src/offline/reducer.ts index e9897ce4708..e042f0a9cb5 100644 --- a/packages/client/src/offline/reducer.ts +++ b/packages/client/src/offline/reducer.ts @@ -8,43 +8,44 @@ * * Copyright (C) The OpenCRVS Authors located at https://github.com/opencrvs/opencrvs-core/blob/master/AUTHORS. */ -import { ISerializedForm } from '@client/forms' -import { initConditionals } from '@client/forms/conditionals' -import { initHandlebarHelpers } from '@client/forms/handlebarHelpers' -import { initValidators } from '@client/forms/validators' -import { ILanguage } from '@client/i18n/reducer' import { - Action as NotificationAction, - configurationErrorNotification -} from '@client/notification/actions' + loop, + Cmd, + Loop, + liftState, + getModel, + getCmd, + RunCmd +} from 'redux-loop' import * as actions from '@client/offline/actions' -import { ISVGTemplate } from '@client/pdfRenderer' import * as profileActions from '@client/profile/profileActions' import { storage } from '@client/storage' -import { isNavigatorOnline } from '@client/utils' -import { EventType, System } from '@client/utils/gateway' -import { filterLocations } from '@client/utils/locationUtils' import { - CertificateConfiguration, IApplicationConfig, IApplicationConfigAnonymous, - IFacilitiesDataResponse, ILocationDataResponse, + referenceApi, + CertificateConfiguration, + IFacilitiesDataResponse, IOfficesDataResponse, - referenceApi + ICertificateData } from '@client/utils/referenceApi' +import { ILanguage } from '@client/i18n/reducer' +import { filterLocations } from '@client/utils/locationUtils' +import { System } from '@client/utils/gateway' import { UserDetails } from '@client/utils/userUtils' +import { isOfflineDataLoaded } from './selectors' import { merge } from 'lodash' +import { isNavigatorOnline } from '@client/utils' +import { ISerializedForm } from '@client/forms' +import { initConditionals } from '@client/forms/conditionals' +import { initHandlebarHelpers } from '@client/forms/handlebarHelpers' +import { initValidators } from '@client/forms/validators' import { - Cmd, - Loop, - RunCmd, - getCmd, - getModel, - liftState, - loop -} from 'redux-loop' -import { isOfflineDataLoaded } from './selectors' + Action as NotificationAction, + configurationErrorNotification +} from '@client/notification/actions' +import { getToken } from '@client/utils/authUtils' export const OFFLINE_LOCATIONS_KEY = 'locations' export const OFFLINE_FACILITIES_KEY = 'facilities' @@ -103,13 +104,10 @@ export interface IOfflineData { languages: ILanguage[] templates: { fonts?: CertificateConfiguration['fonts'] - // Certificates might not be defined in the case of - // a field agent using the app. - certificates?: { - birth: ISVGTemplate - death: ISVGTemplate - marriage: ISVGTemplate - } + // TODO: Certificates need to be restricted in the case of + // a user who does not have permission to print certificate + // (ex: a field agent using the app) + certificates: ICertificateData[] } assets: { logo: string @@ -138,6 +136,43 @@ async function saveOfflineData(offlineData: IOfflineData) { return storage.setItem('offline', JSON.stringify(offlineData)) } +async function loadSingleCertificate(certificate: ICertificateData) { + const { id } = certificate + const res = await fetch(certificate.svgUrl, { + headers: { + Authorization: getToken(), + 'If-None-Match': certificate?.hash ?? '' + } + }) + if (res.status === 304) { + return { + ...certificate, + svg: certificate.svg, + hash: certificate!.hash! + } + } + if (!res.ok) { + return Promise.reject( + new Error(`Fetching certificate with id: "${id}" failed`) + ) + } + return res.text().then((svg) => { + return { + ...certificate, + svg, + hash: res.headers.get('etag')! + } + }) +} + +async function loadCertificates( + savedCertificates: IOfflineData['templates']['certificates'] +) { + return await Promise.all( + savedCertificates.map((cert) => loadSingleCertificate(cert)) + ) +} + function checkIfDone( oldState: IOfflineDataState, loopOrState: IOfflineDataState | Loop @@ -200,14 +235,6 @@ const CONFIG_CMD = Cmd.run(() => referenceApi.loadConfig(), { failActionCreator: actions.configFailed }) -const CERTIFICATE_CONFIG_CMD = Cmd.run( - () => referenceApi.loadCertificateConfiguration(), - { - successActionCreator: actions.certificateConfigurationLoaded, - failActionCreator: actions.certificateConfigurationLoadFailed - } -) - const CONTENT_CMD = Cmd.run(() => referenceApi.loadContent(), { successActionCreator: actions.contentLoaded, failActionCreator: actions.contentFailed @@ -242,7 +269,6 @@ function getDataLoadingCommands() { FACILITIES_CMD, LOCATIONS_CMD, CONFIG_CMD, - CERTIFICATE_CONFIG_CMD, CONDITIONALS_CMD, VALIDATORS_CMD, HANDLEBARS_CMD, @@ -368,65 +394,57 @@ function reducer( case actions.APPLICATION_CONFIG_LOADED: { const { certificates, config, systems } = action.payload merge(window.config, config) - const birthCertificateTemplate = certificates.find( - ({ event }) => event === EventType.Birth - ) - const deathCertificateTemplate = certificates.find( - ({ event }) => event === EventType.Death - ) + const newOfflineData = { + ...state.offlineData, + config, + systems, + templates: { + ...state.offlineData.templates, + certificates: (certificates as ICertificateData[]).map((x) => { + const baseUrl = window.location.origin + if (x.fonts) { + x.fonts = Object.fromEntries( + Object.entries(x.fonts).map(([fontFamily, fontStyles]) => [ + fontFamily, + { + normal: `${baseUrl}${fontStyles.normal}`, + bold: `${baseUrl}${fontStyles.bold}`, + italics: `${baseUrl}${fontStyles.italics}`, + bolditalics: `${baseUrl}${fontStyles.bolditalics}` + } + ]) + ) + } + return x + }) + } + } - const marriageCertificateTemplate = certificates.find( - ({ event }) => event === EventType.Marriage + return loop( + { + ...state, + offlineDataLoaded: isOfflineDataLoaded(newOfflineData), + offlineData: newOfflineData + }, + Cmd.run(loadCertificates, { + successActionCreator: actions.certificatesLoaded, + args: [newOfflineData.templates?.certificates] + }) ) + } - let newOfflineData: Partial - - if ( - birthCertificateTemplate && - deathCertificateTemplate && - marriageCertificateTemplate - ) { - const certificatesTemplates = { - birth: { - definition: birthCertificateTemplate.svgCode - }, - death: { - definition: deathCertificateTemplate.svgCode - }, - marriage: { - definition: marriageCertificateTemplate.svgCode - } - } - - newOfflineData = { + case actions.CERTIFICATES_LOADED: { + const certificates = action.payload + return { + ...state, + offlineData: { ...state.offlineData, - config, - systems, templates: { ...state.offlineData.templates, - certificates: certificatesTemplates + certificates } } - } else { - newOfflineData = { - ...state.offlineData, - config, - systems, - - // Field agents do not get certificate templates from the config service. - // Our loading logic depends on certificates being present and the app would load infinitely - // without a value here. - // This is a quickfix for the issue. If done properly, we should amend the "is loading" check - // to not expect certificate templates when a field agent is logged in. - templates: {} - } - } - - return { - ...state, - offlineDataLoaded: isOfflineDataLoaded(newOfflineData), - offlineData: newOfflineData } } @@ -477,23 +495,6 @@ function reducer( ) } - case actions.CERTIFICATE_CONFIGURATION_LOADED: { - return { - ...state, - offlineData: { - ...state.offlineData, - templates: { - ...state.offlineData.templates, - fonts: action.payload.fonts - } - } - } - } - - case actions.CERTIFICATE_CONFIGURATION_LOAD_FAILED: { - return loop(state, delay(CERTIFICATE_CONFIG_CMD, RETRY_TIMEOUT)) - } - /* * Locations */ diff --git a/packages/client/src/setupTests.ts b/packages/client/src/setupTests.ts index f986361115f..d3e92326211 100644 --- a/packages/client/src/setupTests.ts +++ b/packages/client/src/setupTests.ts @@ -40,11 +40,7 @@ const config = { BIRTH: { REGISTRATION_TARGET: 45, LATE_REGISTRATION_TARGET: 365, - FEE: { - ON_TIME: 0, - LATE: 0, - DELAYED: 0 - } + PRINT_IN_ADVANCE: true }, COUNTRY: 'BGD', CURRENCY: { @@ -53,10 +49,7 @@ const config = { }, DEATH: { REGISTRATION_TARGET: 45, - FEE: { - ON_TIME: 0, - DELAYED: 0 - } + PRINT_IN_ADVANCE: true }, FEATURES: { DEATH_REGISTRATION: true, @@ -201,7 +194,6 @@ vi.doMock( languages: mockOfflineData.languages }), loadConfig: () => Promise.resolve(mockConfigResponse), - loadCertificateConfiguration: () => Promise.resolve({}), loadConfigAnonymousUser: () => Promise.resolve(mockConfigResponse), loadForms: () => Promise.resolve(mockOfflineData.forms.forms), importConditionals: () => Promise.resolve({}), diff --git a/packages/client/src/src-sw.ts b/packages/client/src/src-sw.ts index 4c435b1a97d..e30e80f4dec 100644 --- a/packages/client/src/src-sw.ts +++ b/packages/client/src/src-sw.ts @@ -82,7 +82,7 @@ registerRoute(/http(.+)config$/, new NetworkFirst()) registerRoute( import.meta.env.DEV ? /http(.+)localhost:3535\/ocrvs\/.+/ - : /http(.+)minio\.(.+)\/ocrvs\/.+/, + : /http(.+)minio\.(.+)\/.*ocrvs.*\/.+/, new CacheFirst() ) diff --git a/packages/client/src/tests/languages.json b/packages/client/src/tests/languages.json index e3329737a31..ab5d4e0bf42 100644 --- a/packages/client/src/tests/languages.json +++ b/packages/client/src/tests/languages.json @@ -52,6 +52,8 @@ "buttons.upload": "Upload", "buttons.yes": "Yes", "certificate.confirmCorrect": "Please confirm that the informant has reviewed that the information on the certificate is correct and that you are ready to print.", + "certificate.selectTemplate": "Select certificate template", + "certificate.selectedTemplate": "Selected certificate template", "certificate.isCertificateCorrect": "Is the {event} certificate correct?", "certificate.label.birth": "Birth", "certificate.label.death": "Death", @@ -1200,6 +1202,8 @@ "buttons.upload": "আপলোড", "buttons.yes": "হ্যাঁ", "certificate.confirmCorrect": "অনুগ্রহ করে নিশ্চিত করুন যে নিবন্ধনটি পর্যালোচনা হয়েছে তার তথ্য সঠিক এবং আপনি মুদ্রণ করতে প্রস্তুত", + "certificate.selectTemplate": "নিবন্ধন টেমপ্লেট নির্বাচন করুন", + "certificate.selectedTemplate": "নির্বাচিত নিবন্ধন টেমপ্লেট", "certificate.isCertificateCorrect": "জন্ম নিবন্ধনটি কি সঠিক?", "certificate.label.birth": "জন্ম", "certificate.label.death": "মৃত্যু", diff --git a/packages/client/src/tests/mock-offline-data.ts b/packages/client/src/tests/mock-offline-data.ts index db4eca8bee6..93f6e2bf063 100644 --- a/packages/client/src/tests/mock-offline-data.ts +++ b/packages/client/src/tests/mock-offline-data.ts @@ -415,27 +415,14 @@ export const mockOfflineData = { BIRTH: { REGISTRATION_TARGET: 45, LATE_REGISTRATION_TARGET: 365, - FEE: { - ON_TIME: 0, - LATE: 15, - DELAYED: 20 - }, PRINT_IN_ADVANCE: true }, DEATH: { REGISTRATION_TARGET: 45, - FEE: { - ON_TIME: 0, - DELAYED: 0 - }, PRINT_IN_ADVANCE: true }, MARRIAGE: { REGISTRATION_TARGET: 45, - FEE: { - ON_TIME: 0, - DELAYED: 0 - }, PRINT_IN_ADVANCE: true }, FEATURES: { diff --git a/packages/client/src/tests/schema.graphql b/packages/client/src/tests/schema.graphql index 5453f94cb75..aa106ca8ee7 100644 --- a/packages/client/src/tests/schema.graphql +++ b/packages/client/src/tests/schema.graphql @@ -295,7 +295,6 @@ input AvatarInput { type Birth { REGISTRATION_TARGET: Int LATE_REGISTRATION_TARGET: Int - FEE: BirthFee PRINT_IN_ADVANCE: Boolean } @@ -319,22 +318,9 @@ type BirthEventSearchSet implements EventSearchSet { fatherIdentifier: String } -type BirthFee { - ON_TIME: Float - LATE: Float - DELAYED: Float -} - -input BirthFeeInput { - ON_TIME: Float - LATE: Float - DELAYED: Float -} - input BirthInput { REGISTRATION_TARGET: Int LATE_REGISTRATION_TARGET: Int - FEE: BirthFeeInput PRINT_IN_ADVANCE: Boolean } @@ -405,41 +391,14 @@ type Certificate { collector: RelatedPerson hasShowedVerifiedDocument: Boolean payments: [Payment] - data: String + certificateTemplateId: String } input CertificateInput { collector: RelatedPersonInput hasShowedVerifiedDocument: Boolean payments: [PaymentInput] - data: String -} - -enum CertificateStatus { - ACTIVE - INACTIVE -} - -type CertificateSVG { - id: ID! - svgCode: String! - svgFilename: String! - svgDateUpdated: String! - svgDateCreated: String! - user: String! - event: Event! - status: CertificateStatus! -} - -input CertificateSVGInput { - id: ID - svgCode: String! - svgFilename: String! - svgDateUpdated: Int - svgDateCreated: Int - user: String! - event: Event! - status: CertificateStatus! + certificateTemplateId: String } type CertificationMetric { @@ -574,7 +533,6 @@ scalar FieldValue type Death { REGISTRATION_TARGET: Int - FEE: DeathFee PRINT_IN_ADVANCE: Boolean } @@ -588,19 +546,8 @@ type DeathEventSearchSet implements EventSearchSet { operationHistories: [OperationHistorySearchSet] } -type DeathFee { - ON_TIME: Float - DELAYED: Float -} - -input DeathFeeInput { - ON_TIME: Float - DELAYED: Float -} - input DeathInput { REGISTRATION_TARGET: Int - FEE: DeathFeeInput PRINT_IN_ADVANCE: Boolean } @@ -935,7 +882,6 @@ scalar Map type Marriage { REGISTRATION_TARGET: Int - FEE: MarriageFee PRINT_IN_ADVANCE: Boolean } @@ -951,19 +897,8 @@ type MarriageEventSearchSet implements EventSearchSet { operationHistories: [OperationHistorySearchSet] } -type MarriageFee { - ON_TIME: Float - DELAYED: Float -} - -input MarriageFeeInput { - ON_TIME: Float - DELAYED: Float -} - input MarriageInput { REGISTRATION_TARGET: Int - FEE: MarriageFeeInput PRINT_IN_ADVANCE: Boolean } @@ -1418,8 +1353,6 @@ type Query { sortBy: String sortOrder: String ): [SystemRole!] - getCertificateSVG(status: CertificateStatus!, event: Event!): CertificateSVG - getActiveCertificatesSVG: [CertificateSVG!] fetchSystem(clientId: ID!): System informantSMSNotifications: [SMSNotification!] } diff --git a/packages/client/src/tests/templates.json b/packages/client/src/tests/templates.json index bc7f832fae7..7b4a0eb0865 100644 --- a/packages/client/src/tests/templates.json +++ b/packages/client/src/tests/templates.json @@ -7,19 +7,136 @@ "bolditalics": "NotoSans-Regular.ttf" } }, - "certificates": { - "birth": { - "definition": " {eventDate} Was born on Place of birth {placeOfBirth} {informantName} This is to certify that {registrationNumber} Birth Registration No Date of issuance of certificate:{certificateDate} {registrarName} ({role}) This event was registered at{registrationLocation} ", - "fileName": "farajaland-birth-certificate-v3.svg" + "certificates": [ + { + "id": "birth-certificate", + "event": "birth", + "label": { + "id": "certificates.birth.certificate", + "defaultMessage": "Birth Certificate", + "description": "The label for a birth certificate" + }, + + "fee": { + "onTime": 0, + "late": 5.5, + "delayed": 15 + }, + "isDefault": true, + "svgUrl": "/api/countryconfig/certificates/birth-certificate.svg", + "svg": "", + "fonts": { + "Noto Sans": { + "normal": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bold": "/api/countryconfig/fonts/NotoSans-Bold.ttf", + "italics": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bolditalics": "/api/countryconfig/fonts/NotoSans-Regular.ttf" + } + } }, - "death": { - "definition": " {eventDate} Died on Place of death {placeOfDeath} {informantName} This is to certify that {registrationNumber} Death Registration No Date of issuance of certificate:{certificateDate} {registrarName} ({role}) This event was registered at{registrationLocation} ", - "fileName": "farajaland-death-certificate-v3.svg" + { + "id": "birth-certificate-copy", + "event": "birth", + "label": { + "id": "certificates.birth.certificate.copy", + "defaultMessage": "Birth Certificate certified copy", + "description": "The label for a birth certificate" + }, + + "fee": { + "onTime": 0, + "late": 5.5, + "delayed": 15 + }, + "isDefault": false, + "svgUrl": "/api/countryconfig/certificates/birth-certificate-certified-copy.svg", + "svg": "", + "fonts": { + "Noto Sans": { + "normal": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bold": "/api/countryconfig/fonts/NotoSans-Bold.ttf", + "italics": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bolditalics": "/api/countryconfig/fonts/NotoSans-Regular.ttf" + } + } }, - "marriage": { - "definition": " {eventDate} Died on Place of death {placeOfDeath} {informantName} This is to certify that {registrationNumber} Death Registration No Date of issuance of certificate:{certificateDate} {registrarName} ({role}) This event was registered at{registrationLocation} ", - "fileName": "farajaland-marriage-certificate-v1.svg", - "lastModifiedDate": "1678106545001" + { + "id": "death-certificate", + "event": "death", + "label": { + "id": "certificates.death.certificate", + "defaultMessage": "Death Certificate", + "description": "The label for a death certificate" + }, + + "fee": { + "onTime": 0, + "late": 5.5, + "delayed": 0 + }, + "isDefault": true, + "svgUrl": "/api/countryconfig/certificates/death-certificate.svg", + "svg": "", + "fonts": { + "Noto Sans": { + "normal": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bold": "/api/countryconfig/fonts/NotoSans-Bold.ttf", + "italics": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bolditalics": "/api/countryconfig/fonts/NotoSans-Regular.ttf" + } + } + }, + { + "id": "death-certificate-copy", + "event": "death", + "label": { + "id": "certificates.death.certificate", + "defaultMessage": "Death Certificate Certified Copy", + "description": "The label for a death certificate" + }, + + "fee": { + "onTime": 0, + "late": 5.5, + "delayed": 15 + }, + "isDefault": true, + "svgUrl": "/api/countryconfig/certificates/death-certificate-copy.svg", + "svg": "", + "fonts": { + "Noto Sans": { + "normal": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bold": "/api/countryconfig/fonts/NotoSans-Bold.ttf", + "italics": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bolditalics": "/api/countryconfig/fonts/NotoSans-Regular.ttf" + } + } + }, + { + "id": "marriage-certificate", + "event": "marriage", + "label": { + "id": "certificates.marriage.certificate", + "defaultMessage": "Marriage Certificate", + "description": "The label for a marriage certificate" + }, + + "fee": { + "onTime": 0, + "late": 5.5, + "delayed": 15 + }, + "isDefault": true, + "svgUrl": "/api/countryconfig/certificates/marriage-certificate.svg", + "svg": "", + "fonts": { + "Noto Sans": { + "normal": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bold": "/api/countryconfig/fonts/NotoSans-Bold.ttf", + "italics": "/api/countryconfig/fonts/NotoSans-Regular.ttf", + "bolditalics": "/api/countryconfig/fonts/NotoSans-Regular.ttf" + } + } } - } + ] } diff --git a/packages/client/src/tests/util.tsx b/packages/client/src/tests/util.tsx index b106544eb37..025937e09ba 100644 --- a/packages/client/src/tests/util.tsx +++ b/packages/client/src/tests/util.tsx @@ -541,7 +541,6 @@ export const mockDeclarationData = { }, registration: { informantsSignature: '', - registrationNumber: '201908122365BDSS0SE1', regStatus: { type: 'REGISTERED', @@ -550,7 +549,7 @@ export const mockDeclarationData = { officeAddressLevel3: 'Gazipur', officeAddressLevel4: 'Dhaka' }, - certificates: [{}] + certificates: [{ certificateTemplateId: 'birth-certificate' }] }, documents: {} } @@ -650,7 +649,8 @@ export const mockDeathDeclarationData = { collector: { type: 'MOTHER' }, - hasShowedVerifiedDocument: true + hasShowedVerifiedDocument: true, + certificateTemplateId: 'death-certificate' } ] } @@ -679,7 +679,8 @@ export const mockMarriageDeclarationData = { collector: { type: 'BRIDE' }, - hasShowedVerifiedDocument: true + hasShowedVerifiedDocument: true, + certificateTemplateId: 'marriage-certificate' } ] }, @@ -771,6 +772,7 @@ export const mockBirthRegistrationSectionData = { data: 'BASE64 data' } }, + certificateTemplateId: 'birth-certificate', hasShowedVerifiedDocument: true } ] @@ -798,33 +800,112 @@ export const mockDeathRegistrationSectionData = { iDType: 'PASSPORT', iD: '123456789' }, - hasShowedVerifiedDocument: true + hasShowedVerifiedDocument: true, + certificateTemplateId: 'death-certificate' } ] } const mockFetchCertificatesTemplatesDefinition = [ { - id: '12313546', - event: EventType.Birth, - status: 'ACTIVE', - svgCode: - '\n\n\n\n{registrarName}
({role}) \n \n \nThis event was registered at {registrationLocation} \n{eventDate} \nDied on \nPlace of death \n \n{placeOfDeath} \n{informantName} \nThis is to certify that \n{registrationNumber} \nDeath Registration No \nDate of issuance of certificate: {certificateDate}\n\n\n\n\n\n\n\n\n\n\n', - svgDateCreated: '1640696680593', - svgDateUpdated: '1644326332088', - svgFilename: 'oCRVS_DefaultZambia_Death_v1.svg', - user: '61d42359f1a2c25ea01beb4b' + id: 'birth-certificate', + event: 'birth' as EventType, + label: { + id: 'certificates.birth.certificate', + defaultMessage: 'Birth Certificate', + description: 'The label for a birth certificate' + }, + fee: { + onTime: 0, + late: 5.5, + delayed: 15 + }, + isDefault: true, + svgUrl: '/api/countryconfig/certificates/birth-certificate.svg', + svg: '', + fonts: { + 'Noto Sans': { + normal: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bold: '/api/countryconfig/fonts/NotoSans-Bold.ttf', + italics: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bolditalics: '/api/countryconfig/fonts/NotoSans-Regular.ttf' + } + } + }, + { + id: 'birth-certificate-copy', + event: 'birth' as EventType, + label: { + id: 'certificates.birth-certificate-copy', + defaultMessage: 'Birth Certificate certified copy', + description: 'The label for a birth certificate' + }, + fee: { + onTime: 0, + late: 5.5, + delayed: 15 + }, + isDefault: false, + svgUrl: '/api/countryconfig/certificates/birth-certificate-copy.svg', + svg: '', + fonts: { + 'Noto Sans': { + normal: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bold: '/api/countryconfig/fonts/NotoSans-Bold.ttf', + italics: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bolditalics: '/api/countryconfig/fonts/NotoSans-Regular.ttf' + } + } + }, + { + id: 'death-certificate', + event: 'death' as EventType, + label: { + id: 'certificates.death.certificate', + defaultMessage: 'Death Certificate', + description: 'The label for a death certificate' + }, + fee: { + onTime: 0, + late: 5.5, + delayed: 15 + }, + isDefault: true, + svgUrl: '/api/countryconfig/certificates/death-certificate.svg', + svg: '', + fonts: { + 'Noto Sans': { + normal: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bold: '/api/countryconfig/fonts/NotoSans-Bold.ttf', + italics: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bolditalics: '/api/countryconfig/fonts/NotoSans-Regular.ttf' + } + } }, { - id: '25313546', - event: EventType.Death, - status: 'ACTIVE', - svgCode: - '\n\n\n\n{registrarName}
({role}) \n \n \nThis event was registered at {registrationLocation} \n{eventDate} \nWas born on \nPlace of birth \n \n{placeOfBirth} \n{informantName} \nThis is to certify that \n{registrationNumber} \nBirth Registration No \nDate of issuance of certificate: {certificateDate}\n\n\n\n\n\n\n\n\n\n\n', - svgDateCreated: '1640696804785', - svgDateUpdated: '1643885502999', - svgFilename: 'oCRVS_DefaultZambia_Birth_v1.svg', - user: '61d42359f1a2c25ea01beb4b' + id: 'marriage-certificate', + event: 'marriage' as EventType, + label: { + id: 'certificates.marriage.certificate', + defaultMessage: 'Marriage Certificate', + description: 'The label for a marriage certificate' + }, + fee: { + onTime: 0, + late: 5.5, + delayed: 15 + }, + isDefault: true, + svgUrl: '/api/countryconfig/certificates/marriage-certificate.svg', + svg: '', + fonts: { + 'Noto Sans': { + normal: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bold: '/api/countryconfig/fonts/NotoSans-Bold.ttf', + italics: '/api/countryconfig/fonts/NotoSans-Regular.ttf', + bolditalics: '/api/countryconfig/fonts/NotoSans-Regular.ttf' + } + } } ] diff --git a/packages/client/src/transformer/index.ts b/packages/client/src/transformer/index.ts index f801defa8f4..1f0733fa135 100644 --- a/packages/client/src/transformer/index.ts +++ b/packages/client/src/transformer/index.ts @@ -287,7 +287,8 @@ export const draftToGqlTransformer = ( draftData[section.id][fieldDef.name] !== null && draftData[section.id][fieldDef.name] !== undefined && draftData[section.id][fieldDef.name] !== '' && - !conditionalActions.includes('hide') + (!conditionalActions.includes('hide') || + fieldDef.name === 'detailsExist') // https://github.com/opencrvs/opencrvs-core/issues/7821#issuecomment-2514398986 ) { if (fieldDef.mapping && fieldDef.mapping.mutation) { fieldDef.mapping.mutation( @@ -433,7 +434,14 @@ export const gqlToDraftTransformer = ( transformedData[section.id]._fhirID = queryData[section.id].id } if (section.mapping && section.mapping.query) { - section.mapping.query(transformedData, queryData, section.id) + section.mapping.query( + transformedData, + queryData, + section.id, + undefined, + undefined, + offlineData + ) } if (section.mapping?.template) { if (!transformedData.template) { diff --git a/packages/client/src/utils/gateway.ts b/packages/client/src/utils/gateway.ts index ccc4153714b..4b284927e27 100644 --- a/packages/client/src/utils/gateway.ts +++ b/packages/client/src/utils/gateway.ts @@ -396,16 +396,29 @@ export type BookmarkedSeachItem = { export type Certificate = { __typename?: 'Certificate' collector?: Maybe - data?: Maybe hasShowedVerifiedDocument?: Maybe payments?: Maybe>> + certificateTemplateId?: Maybe } export type CertificateInput = { collector?: InputMaybe - data?: InputMaybe hasShowedVerifiedDocument?: InputMaybe payments?: InputMaybe>> + certificateTemplateId?: InputMaybe +} + +export type CertificateLabel = { + __typename?: 'CertificateLabel' + defaultMessage: Scalars['String'] + description: Scalars['String'] + id: Scalars['String'] +} + +export type CertificateLabelInput = { + defaultMessage: Scalars['String'] + description: Scalars['String'] + id: Scalars['String'] } export type CertificationMetric = { @@ -786,6 +799,7 @@ export type History = { signature?: Maybe statusReason?: Maybe system?: Maybe + certificateTemplateId?: Maybe user?: Maybe } @@ -3437,6 +3451,28 @@ export type FetchBirthRegistrationForReviewQuery = { type?: RegistrationType | null trackingId?: string | null registrationNumber?: string | null + certificates?: Array<{ + __typename?: 'Certificate' + hasShowedVerifiedDocument?: boolean | null + certificateTemplateId?: string | null + collector?: { + __typename?: 'RelatedPerson' + relationship?: string | null + otherRelationship?: string | null + name?: Array<{ + __typename?: 'HumanName' + use?: string | null + firstNames?: string | null + familyName?: string | null + } | null> | null + telecom?: Array<{ + __typename?: 'ContactPoint' + system?: string | null + value?: string | null + use?: string | null + } | null> | null + } | null + } | null> | null duplicates?: Array<{ __typename?: 'DuplicatesInfo' compositionId?: string | null @@ -3505,6 +3541,7 @@ export type FetchBirthRegistrationForReviewQuery = { reason?: string | null duplicateOf?: string | null potentialDuplicates?: Array | null + certificateTemplateId?: string | null documents: Array<{ __typename?: 'Attachment' id: string @@ -7549,76 +7586,82 @@ export type GetRegistrationsListByFilterQueryVariables = Exact<{ size: Scalars['Int'] }> +export type RegistrationsListByLocationFilter = { + __typename: 'TotalMetricsByLocation' + total?: number | null + results: Array<{ + __typename?: 'EventMetricsByLocation' + total: number + late: number + delayed: number + home: number + healthFacility: number + location: { __typename?: 'Location'; name?: string | null } + }> +} + +export type RegistrationsListByRegistrarFilter = { + __typename: 'TotalMetricsByRegistrar' + total?: number | null + results: Array<{ + __typename?: 'EventMetricsByRegistrar' + total: number + late: number + delayed: number + registrarPractitioner?: { + __typename?: 'User' + id: string + systemRole: SystemRoleType + role: { + __typename?: 'Role' + _id: string + labels: Array<{ + __typename?: 'RoleLabel' + lang: string + label: string + }> + } + primaryOffice?: { + __typename?: 'Location' + name?: string | null + id: string + } | null + name: Array<{ + __typename?: 'HumanName' + firstNames?: string | null + familyName?: string | null + use?: string | null + }> + avatar?: { + __typename?: 'Avatar' + type: string + data: string + } | null + } | null + }> +} + +export type RegistrationsListByTimeFilter = { + __typename: 'TotalMetricsByTime' + total?: number | null + results: Array<{ + __typename?: 'EventMetricsByTime' + total: number + delayed: number + late: number + home: number + healthFacility: number + month: string + time: string + }> +} + export type GetRegistrationsListByFilterQuery = { __typename?: 'Query' getRegistrationsListByFilter?: - | { - __typename: 'TotalMetricsByLocation' - total?: number | null - results: Array<{ - __typename?: 'EventMetricsByLocation' - total: number - late: number - delayed: number - home: number - healthFacility: number - location: { __typename?: 'Location'; name?: string | null } - }> - } - | { - __typename: 'TotalMetricsByRegistrar' - total?: number | null - results: Array<{ - __typename?: 'EventMetricsByRegistrar' - total: number - late: number - delayed: number - registrarPractitioner?: { - __typename?: 'User' - id: string - systemRole: SystemRoleType - role: { - __typename?: 'Role' - _id: string - labels: Array<{ - __typename?: 'RoleLabel' - lang: string - label: string - }> - } - primaryOffice?: { - __typename?: 'Location' - name?: string | null - id: string - } | null - name: Array<{ - __typename?: 'HumanName' - firstNames?: string | null - familyName?: string | null - use?: string | null - }> - avatar?: { - __typename?: 'Avatar' - type: string - data: string - } | null - } | null - }> - } - | { - __typename: 'TotalMetricsByTime' - total?: number | null - results: Array<{ - __typename?: 'EventMetricsByTime' - total: number - delayed: number - late: number - home: number - healthFacility: number - month: string - time: string - }> - } + | RegistrationsListByLocationFilter + | RegistrationsListByRegistrarFilter + | RegistrationsListByTimeFilter | null } diff --git a/packages/client/src/utils/referenceApi.ts b/packages/client/src/utils/referenceApi.ts index b2343d90c80..4fd63a0cda5 100644 --- a/packages/client/src/utils/referenceApi.ts +++ b/packages/client/src/utils/referenceApi.ts @@ -77,10 +77,29 @@ interface ILoginBackground { backgroundImage?: string imageFit?: string } -export interface ICertificateTemplateData { +export interface ICertificateConfigData { + id: string event: EventType - svgCode: string + label: { + id: string + defaultMessage: string + description: string + } + isDefault: boolean + fee: { + onTime: number + late: number + delayed: number + } + svgUrl: string + fonts?: Record +} + +export interface ICertificateData extends ICertificateConfigData { + hash?: string + svg: string } + export interface ICurrency { isoCode: string languagesAndCountry: string[] @@ -98,29 +117,16 @@ export interface IApplicationConfig { BIRTH: { REGISTRATION_TARGET: number LATE_REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - LATE: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } COUNTRY_LOGO: ICountryLogo CURRENCY: ICurrency DEATH: { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } MARRIAGE: { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } FEATURES: { @@ -143,7 +149,7 @@ export interface IApplicationConfig { } export interface IApplicationConfigResponse { config: IApplicationConfig - certificates: ICertificateTemplateData[] + certificates: ICertificateConfigData[] systems: System[] } @@ -250,33 +256,6 @@ async function importHandlebarHelpers(): Promise { return {} } } -async function loadCertificateConfiguration(): Promise { - const url = `${window.config.COUNTRY_CONFIG_URL}/certificate-configuration` - - const res = await fetch(url, { - method: 'GET' - }) - - // for backward compatibility, if the endpoint is unimplemented - if (res.status === 404) { - return { - fonts: { - notosans: { - normal: 'NotoSans-Light.ttf', - bold: 'NotoSans-Regular.ttf', - italics: 'NotoSans-Light.ttf', - bolditalics: 'NotoSans-Regular.ttf' - } - } - } - } - - if (!res.ok) { - throw Error(res.statusText) - } - - return res.json() -} async function loadContent(): Promise { const url = `${window.config.COUNTRY_CONFIG_URL}/content/client` @@ -415,7 +394,6 @@ async function loadFacilities(): Promise { export const referenceApi = { loadLocations, loadFacilities, - loadCertificateConfiguration, loadContent, loadConfig, loadForms, diff --git a/packages/client/src/views/CorrectionForm/utils.ts b/packages/client/src/views/CorrectionForm/utils.ts index 2f0cb483904..ff093047ecb 100644 --- a/packages/client/src/views/CorrectionForm/utils.ts +++ b/packages/client/src/views/CorrectionForm/utils.ts @@ -412,9 +412,13 @@ const getFormFieldValue = ( let tempField: IFormField for (const key in sectionDraftData) { tempField = sectionDraftData[key] as IFormField - return (tempField && + if ( + tempField && tempField.nestedFields && - tempField.nestedFields[field.name]) as IFormFieldValue + field.name in tempField.nestedFields + ) { + return tempField.nestedFields[field.name] as IFormFieldValue + } } return '' } diff --git a/packages/client/src/views/DataProvider/birth/queries.ts b/packages/client/src/views/DataProvider/birth/queries.ts index 441ac15bc78..205b59422a1 100644 --- a/packages/client/src/views/DataProvider/birth/queries.ts +++ b/packages/client/src/views/DataProvider/birth/queries.ts @@ -149,6 +149,24 @@ const GET_BIRTH_REGISTRATION_FOR_REVIEW = gql` contactRelationship contactPhoneNumber contactEmail + certificates { + hasShowedVerifiedDocument + certificateTemplateId + collector { + relationship + otherRelationship + name { + use + firstNames + familyName + } + telecom { + system + value + use + } + } + } duplicates { compositionId trackingId @@ -206,6 +224,7 @@ const GET_BIRTH_REGISTRATION_FOR_REVIEW = gql` requesterOther noSupportingDocumentationRequired hasShowedVerifiedDocument + certificateTemplateId date action regStatus @@ -294,6 +313,7 @@ const GET_BIRTH_REGISTRATION_FOR_REVIEW = gql` } certificates { hasShowedVerifiedDocument + certificateTemplateId collector { relationship otherRelationship @@ -569,6 +589,7 @@ export const GET_BIRTH_REGISTRATION_FOR_CERTIFICATE = gql` } certificates { hasShowedVerifiedDocument + certificateTemplateId collector { relationship otherRelationship diff --git a/packages/client/src/views/DataProvider/death/queries.ts b/packages/client/src/views/DataProvider/death/queries.ts index 86aecd6e7ce..42bcbcdd5c1 100644 --- a/packages/client/src/views/DataProvider/death/queries.ts +++ b/packages/client/src/views/DataProvider/death/queries.ts @@ -213,6 +213,24 @@ const GET_DEATH_REGISTRATION_FOR_REVIEW = gql` contactRelationship contactPhoneNumber contactEmail + certificates { + hasShowedVerifiedDocument + certificateTemplateId + collector { + relationship + otherRelationship + name { + use + firstNames + familyName + } + telecom { + system + value + use + } + } + } duplicates { compositionId trackingId @@ -288,6 +306,7 @@ const GET_DEATH_REGISTRATION_FOR_REVIEW = gql` requester requesterOther hasShowedVerifiedDocument + certificateTemplateId noSupportingDocumentationRequired date action @@ -363,6 +382,7 @@ const GET_DEATH_REGISTRATION_FOR_REVIEW = gql` } certificates { hasShowedVerifiedDocument + certificateTemplateId collector { relationship otherRelationship @@ -628,6 +648,7 @@ export const GET_DEATH_REGISTRATION_FOR_CERTIFICATION = gql` } certificates { hasShowedVerifiedDocument + certificateTemplateId collector { relationship otherRelationship diff --git a/packages/client/src/views/DataProvider/marriage/queries.ts b/packages/client/src/views/DataProvider/marriage/queries.ts index 2eb8d12cb2a..ba215418f0d 100644 --- a/packages/client/src/views/DataProvider/marriage/queries.ts +++ b/packages/client/src/views/DataProvider/marriage/queries.ts @@ -158,6 +158,24 @@ const GET_MARRIAGE_REGISTRATION_FOR_REVIEW = gql` contactRelationship contactPhoneNumber contactEmail + certificates { + hasShowedVerifiedDocument + certificateTemplateId + collector { + relationship + otherRelationship + name { + use + firstNames + familyName + } + telecom { + system + value + use + } + } + } groomSignature brideSignature witnessOneSignature @@ -227,6 +245,7 @@ const GET_MARRIAGE_REGISTRATION_FOR_REVIEW = gql` otherReason requester hasShowedVerifiedDocument + certificateTemplateId noSupportingDocumentationRequired date action diff --git a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueCollectorForm.tsx b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueCollectorForm.tsx index 0e732426569..b8d8f5041a8 100644 --- a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueCollectorForm.tsx +++ b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueCollectorForm.tsx @@ -64,7 +64,8 @@ export function IssueCollectorForm({ certificates: [ { collector: collector, - hasShowedVerifiedDocument: false + hasShowedVerifiedDocument: false, + certificateTemplateId: certificate.certificateTemplateId } ] } @@ -78,6 +79,7 @@ export function IssueCollectorForm({ const relationship = declaration.data.registration.certificates[0].collector?.type const event = declaration.event + if (!relationship) return if (relationship === 'OTHER') { navigate( diff --git a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueFormForOthers.tsx b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueFormForOthers.tsx index f7951329677..41d62e12346 100644 --- a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueFormForOthers.tsx +++ b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssueFormForOthers.tsx @@ -63,7 +63,11 @@ export const IssueCollectorFormForOthers = ({ const navigate = useNavigate() const config = useSelector(getOfflineData) const user = useSelector(getUserDetails) - + const { relationship, ...collectorForm }: { [key: string]: any } = + (declaration && + declaration.data.registration.certificates && + declaration.data.registration.certificates[0].collector) || + {} const fields: IFormField[] = collectorFormFieldsForOthers(declaration.event) const handleChange = ( sectionData: ICertificate['collector'], @@ -83,7 +87,8 @@ export const IssueCollectorFormForOthers = ({ certificates: [ { collector: collector, - hasShowedVerifiedDocument: false + hasShowedVerifiedDocument: false, + certificateTemplateId: certificate.certificateTemplateId } ] } @@ -148,12 +153,7 @@ export const IssueCollectorFormForOthers = ({ setAllFieldsDirty={false} fields={replaceInitialValues( fields, - (declaration && - declaration.data.registration.certificates && - declaration.data.registration.certificates[ - declaration.data.registration.certificates.length - 1 - ].collector) || - {}, + collectorForm, declaration && declaration.data, config, user diff --git a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.test.tsx b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.test.tsx index b28b5c2ebff..e04b04aef2f 100644 --- a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.test.tsx +++ b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.test.tsx @@ -56,6 +56,14 @@ const deathDeclaration = { id: 'mockDeath1234', data: { ...mockDeathDeclarationData, + registration: { + ...mockDeathDeclarationData.registration, + certificates: [ + { + certificateTemplateId: 'death-certificate-copy' + } + ] + }, history: [ { date: '2021-03-21T08:16:24.467+00:00', @@ -96,7 +104,7 @@ describe('verify collector tests for issuance', () => { } ) expect(testComponent.find('#service').hostNodes().text()).toContain('Birth') - expect(testComponent.find('#amountDue').hostNodes().text()).toContain('20') + expect(testComponent.find('#amountDue').hostNodes().text()).toContain('15') testComponent.find('#Continue').hostNodes().simulate('click') }) @@ -123,7 +131,7 @@ describe('verify collector tests for issuance', () => { }) }) -describe('in case of death declaration renders issue payment component', () => { +describe('in case of birth declaration renders issue payment component', () => { const { store } = createStore() beforeAll(async () => { getItem.mockReturnValue(validToken) @@ -147,7 +155,7 @@ describe('in case of death declaration renders issue payment component', () => { } ) expect(testComponent.find('#service').hostNodes().text()).toContain('Death') - expect(testComponent.find('#amountDue').hostNodes().text()).toContain('0.0') + expect(testComponent.find('#amountDue').hostNodes().text()).toContain('15') testComponent.find('#Continue').hostNodes().simulate('click') }) }) diff --git a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.tsx b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.tsx index a4d6043925d..a2d4bdd2595 100644 --- a/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.tsx +++ b/packages/client/src/views/IssueCertificate/IssueCollectorForm/IssuePayment.tsx @@ -10,6 +10,7 @@ */ import { + ICertificate, IDeclaration, IPrintableDeclaration, modifyDeclaration, @@ -87,7 +88,8 @@ export const IssuePayment = () => { event, eventDate, registeredDate, - offlineCountryConfig + offlineCountryConfig, + certificate as ICertificate ) certificate.payments = { type: 'MANUAL' as const, @@ -128,7 +130,8 @@ export const IssuePayment = () => { event, eventDate, registeredDate, - offlineCountryConfig + offlineCountryConfig, + declaration.data.registration.certificates[0] ) const serviceMessage = getServiceMessage( diff --git a/packages/client/src/views/PrintCertificate/PDFUtils.ts b/packages/client/src/views/PrintCertificate/PDFUtils.ts index d8d72568334..c9c73983ac9 100644 --- a/packages/client/src/views/PrintCertificate/PDFUtils.ts +++ b/packages/client/src/views/PrintCertificate/PDFUtils.ts @@ -14,11 +14,7 @@ import { createIntl, createIntlCache } from 'react-intl' -import { - AdminStructure, - ILocation, - IOfflineData -} from '@client/offline/reducer' +import { AdminStructure, ILocation } from '@client/offline/reducer' import { IPDFTemplate } from '@client/pdfRenderer' import { certificateBaseTemplate } from '@client/templates/register' import * as Handlebars from 'handlebars' @@ -27,7 +23,10 @@ import { getOfflineData } from '@client/offline/selectors' import isValid from 'date-fns/isValid' import format from 'date-fns/format' import { getHandlebarHelpers } from '@client/forms/handlebarHelpers' -import { FontFamilyTypes } from '@client/utils/referenceApi' +import { + CertificateConfiguration, + FontFamilyTypes +} from '@client/utils/referenceApi' import htmlToPdfmake from 'html-to-pdfmake' import { Content } from 'pdfmake/interfaces' @@ -225,23 +224,23 @@ src: url("${url}") format("truetype"); const serializer = new XMLSerializer() return serializer.serializeToString(svg) } -export function svgToPdfTemplate(svg: string, offlineResource: IOfflineData) { - const initialDefaultFont = offlineResource.templates.fonts - ? Object.keys(offlineResource.templates.fonts)[0] - : null +export function svgToPdfTemplate( + svg: string, + certificateFonts: CertificateConfiguration +) { const pdfTemplate: IPDFTemplate = { ...certificateBaseTemplate, definition: { ...certificateBaseTemplate.definition, defaultStyle: { font: - initialDefaultFont || + Object.keys(certificateFonts)[0] || certificateBaseTemplate.definition.defaultStyle.font } }, fonts: { ...certificateBaseTemplate.fonts, - ...offlineResource.templates.fonts + ...certificateFonts } } diff --git a/packages/client/src/views/PrintCertificate/Payment.test.tsx b/packages/client/src/views/PrintCertificate/Payment.test.tsx index 4d68db07bbb..5bcfade1b99 100644 --- a/packages/client/src/views/PrintCertificate/Payment.test.tsx +++ b/packages/client/src/views/PrintCertificate/Payment.test.tsx @@ -73,7 +73,27 @@ describe('verify collector tests', () => { await store.dispatch(checkAuth()) await flushPromises() // @ts-ignore - store.dispatch(storeDeclaration(birthDeclaration)) + store.dispatch( + storeDeclaration({ + ...birthDeclaration, + // @ts-ignore + data: { + ...birthDeclaration.data, + registration: { + ...birthDeclaration.data.registration, + certificates: [ + { + collector: { + type: 'MOTHER' + }, + hasShowedVerifiedDocument: true, + certificateTemplateId: 'birth-certificate' + } + ] + } + } + }) + ) }) it('when mother is collector renders Payment component', async () => { @@ -96,7 +116,7 @@ describe('verify collector tests', () => { ) expect(testComponent.find('#amountDue').hostNodes().text()).toContain( - '20' + '15' ) testComponent.find('#Continue').hostNodes().simulate('click') diff --git a/packages/client/src/views/PrintCertificate/Payment.tsx b/packages/client/src/views/PrintCertificate/Payment.tsx index 24bdb410736..aac264e66d8 100644 --- a/packages/client/src/views/PrintCertificate/Payment.tsx +++ b/packages/client/src/views/PrintCertificate/Payment.tsx @@ -135,7 +135,8 @@ const PaymentComponent = ({ event, eventDate, registeredDate, - offlineCountryConfig + offlineCountryConfig, + declaration.data.registration.certificates[0] ) const serviceMessage = getServiceMessage( diff --git a/packages/client/src/views/PrintCertificate/ReviewCertificateAction.test.tsx b/packages/client/src/views/PrintCertificate/ReviewCertificateAction.test.tsx index 3b12f9d780f..0b04271dece 100644 --- a/packages/client/src/views/PrintCertificate/ReviewCertificateAction.test.tsx +++ b/packages/client/src/views/PrintCertificate/ReviewCertificateAction.test.tsx @@ -85,9 +85,11 @@ describe('back button behavior tests of review certificate action', () => { ] } mockBirthDeclarationData.registration.certificates[0] = { + //@ts-ignore collector: { type: 'PRINT_IN_ADVANCE' - } + }, + certificateTemplateId: 'death-certificate' } it('takes user history back when navigated from inside app', async () => { @@ -177,9 +179,11 @@ describe('when user wants to review birth certificate', () => { const mockBirthDeclarationData = cloneDeep(mockDeclarationData) mockBirthDeclarationData.registration.certificates[0] = { + //@ts-ignore collector: { type: 'PRINT_IN_ADVANCE' - } + }, + certificateTemplateId: 'birth-certificate' } loginAsFieldAgent(store) await flushPromises() diff --git a/packages/client/src/views/PrintCertificate/ReviewCertificateAction.tsx b/packages/client/src/views/PrintCertificate/ReviewCertificateAction.tsx index 7938b0aba03..edc449cfbdd 100644 --- a/packages/client/src/views/PrintCertificate/ReviewCertificateAction.tsx +++ b/packages/client/src/views/PrintCertificate/ReviewCertificateAction.tsx @@ -125,10 +125,11 @@ const ReviewCertificateFrame = ({ } export const ReviewCertificate = () => { - const { registrationId } = useParams<{ registrationId: string }>() - + const { registrationId } = useParams<{ + registrationId: string + }>() const { - svg, + svgCode, handleCertify, isPrintInAdvance, canUserEditRecord, @@ -138,7 +139,7 @@ export const ReviewCertificate = () => { const intl = useIntl() const [modal, openModal] = useModal() - if (!svg) { + if (!svgCode) { return ( @@ -199,7 +200,7 @@ export const ReviewCertificate = () => { { if ( isFreeOfCost( - event, + declaration.data.registration.certificates[0], eventDate, registeredDate, offlineCountryConfiguration diff --git a/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.test.tsx b/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.test.tsx index 529d798113b..24f2677c4bd 100644 --- a/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.test.tsx +++ b/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.test.tsx @@ -209,8 +209,11 @@ describe('Certificate collector test for a birth registration without father det it('prompt error when no option is selected', async () => { component.find('#confirm_form').hostNodes().simulate('click') - await waitForElement(component, '#form_error') - expect(component.find('#form_error').hostNodes().text()).toBe( + await waitForElement(component, '#certificateTemplateId_error') + expect( + component.find('#certificateTemplateId_error').hostNodes().text() + ).toBe('Please select certificate type') + expect(component.find('#type_error').hostNodes().text()).toBe( 'Please select who is collecting the certificate' ) }) @@ -220,6 +223,11 @@ describe('Certificate collector test for a birth registration without father det .find('#type_FATHER') .hostNodes() .simulate('change', { target: { value: 'FATHER' } }) + await selectOption( + component, + '#certificateTemplateId', + 'Birth Certificate' + ) await new Promise((resolve) => { setTimeout(resolve, 500) @@ -242,6 +250,11 @@ describe('Certificate collector test for a birth registration without father det .find('#type_FATHER') .hostNodes() .simulate('change', { target: { value: 'FATHER' } }) + await selectOption( + component, + '#certificateTemplateId', + 'Birth Certificate' + ) component.update() component.find('#confirm_form').hostNodes().simulate('click') @@ -260,6 +273,11 @@ describe('Certificate collector test for a birth registration without father det .find('#type_OTHER') .hostNodes() .simulate('change', { target: { value: 'OTHER' } }) + await selectOption( + component, + '#certificateTemplateId', + 'Birth Certificate' + ) await new Promise((resolve) => { setTimeout(resolve, 500) @@ -311,6 +329,11 @@ describe('Certificate collector test for a birth registration without father det .find('#type_OTHER') .hostNodes() .simulate('change', { target: { value: 'OTHER' } }) + await selectOption( + component, + '#certificateTemplateId', + 'Birth Certificate' + ) // Continue form.find('#confirm_form').hostNodes().simulate('click') @@ -569,6 +592,12 @@ describe('Certificate collector test for a death registration', () => { .hostNodes() .simulate('change', { target: { value: 'PRINT_IN_ADVANCE' } }) + await selectOption( + component, + '#certificateTemplateId', + 'Death Certificate Certified Copy' + ) + const $confirm = await waitForElement(component, '#confirm_form') $confirm.hostNodes().simulate('click') @@ -624,6 +653,11 @@ describe('Certificate collector test for a marriage registration', () => { .hostNodes() .simulate('change', { target: { value: 'PRINT_IN_ADVANCE' } }) + await selectOption( + component, + '#certificateTemplateId', + 'Marriage Certificate' + ) const $confirm = await waitForElement(component, '#confirm_form') $confirm.hostNodes().simulate('click') diff --git a/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.tsx b/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.tsx index 5a41b734540..6b9370905f1 100644 --- a/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.tsx +++ b/packages/client/src/views/PrintCertificate/collectorForm/CollectorForm.tsx @@ -61,12 +61,10 @@ import { isCertificateForPrintInAdvance, filterPrintInAdvancedOption } from '@client/views/PrintCertificate/utils' -import { flatten } from 'lodash' import * as React from 'react' import { WrappedComponentProps as IntlShapeProps, injectIntl } from 'react-intl' import { connect } from 'react-redux' import { Navigate } from 'react-router-dom' -import { IValidationResult } from '@client/utils/validate' import { getRegisterForm } from '@client/forms/register/declaration-selectors' import { getCertificateCollectorFormSection } from '@client/forms/certificate/fieldDefinitions/collectorSection' import { replaceInitialValues } from '@client/views/RegisterForm/RegisterForm' @@ -80,6 +78,7 @@ import { RouteComponentProps, withRouter } from '@client/components/WithRouterProps' +import { FormikTouched, FormikValues } from 'formik' const ErrorWrapper = styled.div` margin-top: -3px; @@ -168,39 +167,27 @@ const getErrorsOnFieldsBySection = ( user ) - return { - [sectionId]: fields.reduce((fields, field) => { - const validationErrors: IValidationResult[] = ( - errors[field.name as keyof typeof errors] as IFieldErrors - ).errors - - const value = draft.data[sectionId] - ? draft.data[sectionId][field.name] - : null - - const informationMissing = - validationErrors.length > 0 || value === null ? validationErrors : [] - - return { ...fields, [field.name]: informationMissing } - }, {}) - } + return Object.values(errors) + .map((field) => field.errors[0]?.message) + .filter(Boolean) } interface IState { - showError: boolean showModalForNoSignedAffidavit: boolean isFileUploading: boolean + showError: boolean } class CollectorFormComponent extends React.Component { constructor(props: IProps) { super(props) this.state = { - showError: false, showModalForNoSignedAffidavit: false, - isFileUploading: false + isFileUploading: false, + showError: false } } + setAllFormFieldsTouched!: (touched: FormikTouched) => void onUploadingStateChanged = (isUploading: boolean) => { this.setState({ @@ -216,7 +203,6 @@ class CollectorFormComponent extends React.Component { const certificates = declaration.data.registration.certificates const certificate = (certificates && certificates[0]) || {} const collector = { ...(certificate.collector || {}), ...sectionData } - this.props.modifyDeclaration({ ...declaration, data: { @@ -226,7 +212,8 @@ class CollectorFormComponent extends React.Component { certificates: [ { collector: collector, - hasShowedVerifiedDocument: false + hasShowedVerifiedDocument: false, + certificateTemplateId: collector.certificateTemplateId } ] } @@ -252,10 +239,6 @@ class CollectorFormComponent extends React.Component { this.props.offlineCountryConfiguration, this.props.userDetails ) - const errorValues = Object.values(errors).map(Object.values) - const errLength = flatten(errorValues).filter( - (errs) => errs.length > 0 - ).length const certificates = draft.data.registration.certificates const certificate = (certificates && certificates[0]) || {} @@ -263,11 +246,18 @@ class CollectorFormComponent extends React.Component { sectionId as keyof typeof certificate ] as IFormSectionData - if (errLength > 0) { + if (errors.length > 0) { + const formGroup = ( + this.props as PropsWhenDeclarationIsFound + ).formGroup.fields.reduce( + (acc, { name }) => ({ ...acc, [name]: true }), + {} + ) + + this.setAllFormFieldsTouched(formGroup) this.setState({ showError: true }) - return } @@ -346,7 +336,7 @@ class CollectorFormComponent extends React.Component { .props as PropsWhenDeclarationIsFound if ( isFreeOfCost( - event, + declaration.data.registration.certificates[0], getEventDate(declaration.data, event), getRegisteredDate(declaration.data), offlineCountryConfiguration @@ -413,7 +403,10 @@ class CollectorFormComponent extends React.Component { id="collector_form" hideBackground title={formSection.title && intl.formatMessage(formSection.title)} - goBack={() => this.props.router.navigate(-1)} + goBack={() => { + this.setState({ showError: false }) + this.props.router.navigate(-1) + }} goHome={() => this.props.router.navigate( generateGoToHomeTabUrl({ @@ -452,7 +445,7 @@ class CollectorFormComponent extends React.Component { ]} > - {showError && ( + {showError && formGroup.error && ( {(formGroup.error && intl.formatMessage(formGroup.error)) || @@ -475,6 +468,9 @@ class CollectorFormComponent extends React.Component { fields={formGroup.fields} draftData={declarationToBeCertified.data} onUploadingStateChanged={this.onUploadingStateChanged} + onSetTouched={(setTouchedFunc) => + (this.setAllFormFieldsTouched = setTouchedFunc) + } /> @@ -544,7 +540,10 @@ const mapStateToProps = ( const userOfficeId = userDetails?.primaryOffice?.id const registeringOfficeId = getRegisteringOfficeId(declaration) - const certFormSection = getCertificateCollectorFormSection(declaration) + const certFormSection = getCertificateCollectorFormSection( + declaration, + state.offline.offlineData.templates?.certificates || [] + ) const isAllowPrintInAdvance = event === EventType.Birth @@ -582,7 +581,7 @@ const mapStateToProps = ( declaration.data.registration.certificates && declaration.data.registration.certificates[ declaration.data.registration.certificates.length - 1 - ].collector) || + ]?.collector) || {}, declaration && declaration.data, offlineCountryConfiguration, diff --git a/packages/client/src/views/PrintCertificate/usePrintableCertificate.ts b/packages/client/src/views/PrintCertificate/usePrintableCertificate.ts index b2c9c14f90e..c7215819bd5 100644 --- a/packages/client/src/views/PrintCertificate/usePrintableCertificate.ts +++ b/packages/client/src/views/PrintCertificate/usePrintableCertificate.ts @@ -48,6 +48,34 @@ import { isCertificateForPrintInAdvance } from './utils' import { useNavigate } from 'react-router-dom' +import { ICertificateData } from '@client/utils/referenceApi' +import { fetchImageAsBase64 } from '@client/utils/imageUtils' + +async function replaceMinioUrlWithBase64(template: Record) { + const regex = /\/[^\/?]+\.(jpg|png|jpeg|svg)(?=\?|$)/i + + async function recursiveTransform(obj: any) { + if (typeof obj !== 'object' || obj === null) { + return obj + } + + const transformedObject = Array.isArray(obj) ? [...obj] : { ...obj } + + for (const key in obj) { + const value = obj[key] + if (typeof value === 'string' && regex.test(value)) { + transformedObject[key] = await fetchImageAsBase64(value) + } else if (typeof value === 'object') { + transformedObject[key] = await recursiveTransform(value) + } else { + transformedObject[key] = value + } + } + + return transformedObject + } + return recursiveTransform(template) +} const withEnhancedTemplateVariables = ( declaration: IPrintableDeclaration | undefined, @@ -64,7 +92,8 @@ const withEnhancedTemplateVariables = ( declaration.event, eventDate, registeredDate, - offlineData + offlineData, + declaration.data.registration.certificates[0] ) const locationKey = userDetails?.primaryOffice?.id @@ -125,28 +154,28 @@ export const usePrintableCertificate = (declarationId?: string) => { declaration?.event !== EventType.Marriage && (hasRegisterScope(scope) || hasRegistrationClerkScope(scope)) - let svg = undefined - const certificateTemplate = - declaration && - offlineData.templates.certificates?.[declaration.event].definition - if (certificateTemplate) { - const svgWithoutFonts = compileSvg( - certificateTemplate, - { ...declaration.data.template, preview: true }, - state - ) - const svgWithFonts = addFontsToSvg( - svgWithoutFonts, - offlineData.templates.fonts ?? {} + const certificateTemplateConfig: ICertificateData | undefined = + offlineData.templates.certificates.find( + (x) => + x.id === + declaration?.data.registration.certificates[0].certificateTemplateId ) - svg = svgWithFonts - } + if (!certificateTemplateConfig) return { svgCode: null } + + const certificateFonts = certificateTemplateConfig?.fonts ?? {} + const svgTemplate = certificateTemplateConfig?.svg + + if (!svgTemplate) return { svgCode: null } + + const svgWithoutFonts = compileSvg( + svgTemplate, + { ...declaration?.data.template, preview: true }, + state + ) + const svgCode = addFontsToSvg(svgWithoutFonts, certificateFonts) const handleCertify = async () => { - if ( - !declaration || - !offlineData.templates.certificates?.[declaration.event].definition - ) { + if (!declaration || !certificateTemplateConfig) { return } const draft = cloneDeep(declaration) @@ -164,7 +193,8 @@ export const usePrintableCertificate = (declarationId?: string) => { draft.event, eventDate, registeredDate, - offlineData + offlineData, + declaration.data.registration.certificates[0] ) certificate.payments = { type: 'MANUAL' as const, @@ -174,23 +204,25 @@ export const usePrintableCertificate = (declarationId?: string) => { } } - const svg = await compileSvg( - offlineData.templates.certificates[draft.event].definition, - { ...draft.data.template, preview: false }, - state + const base64ReplacedTemplate = await replaceMinioUrlWithBase64( + draft.data.template ) + const svg = compileSvg( + svgTemplate, + { ...base64ReplacedTemplate, preview: false }, + state + ) draft.data.registration = { ...draft.data.registration, certificates: [ { - ...certificate, - data: svg || '' + ...certificate } ] } - const pdfTemplate = svgToPdfTemplate(svg, offlineData) + const pdfTemplate = svgToPdfTemplate(svg, certificateFonts) printPDF(pdfTemplate, draft.id) @@ -238,7 +270,7 @@ export const usePrintableCertificate = (declarationId?: string) => { } return { - svg, + svgCode, handleCertify, isPrintInAdvance, canUserEditRecord, diff --git a/packages/client/src/views/PrintCertificate/utils.ts b/packages/client/src/views/PrintCertificate/utils.ts index 6b7c63dd745..65f4288722b 100644 --- a/packages/client/src/views/PrintCertificate/utils.ts +++ b/packages/client/src/views/PrintCertificate/utils.ts @@ -8,15 +8,15 @@ * * Copyright (C) The OpenCRVS Authors located at https://github.com/opencrvs/opencrvs-core/blob/master/AUTHORS. */ -import { IPrintableDeclaration } from '@client/declarations' import { IFormData, IFormSectionGroup, ISelectOption } from '@client/forms' +import { Event, EventType } from '@client/utils/gateway' import { dynamicMessages } from '@client/i18n/messages/views/certificate' -import { ILanguageState } from '@client/i18n/reducer' import { getAvailableLanguages } from '@client/i18n/utils' +import { ILanguageState } from '@client/i18n/reducer' +import { ICertificate, IPrintableDeclaration } from '@client/declarations' +import { IntlShape } from 'react-intl' import { IOfflineData } from '@client/offline/reducer' -import { EventType } from '@client/utils/gateway' import differenceInDays from 'date-fns/differenceInDays' -import { IntlShape } from 'react-intl' const MONTH_IN_DAYS = 30 const YEAR_IN_DAYS = 365 @@ -57,72 +57,79 @@ export function getCountryTranslations( return certificateCountries } -interface IDayRange { - rangeData: { [key in EventType]?: IRange[] } -} - -function getDayRanges(offlineData: IOfflineData): IDayRange { - const BIRTH_REGISTRATION_TARGET = offlineData.config.BIRTH.REGISTRATION_TARGET - const BIRTH_LATE_REGISTRATION_TARGET = - offlineData.config.BIRTH.LATE_REGISTRATION_TARGET - const BIRTH_ON_TIME_FEE = offlineData.config.BIRTH.FEE.ON_TIME - const BIRTH_LATE_FEE = offlineData.config.BIRTH.FEE.LATE - const BIRTH_DELAYED_FEE = offlineData.config.BIRTH.FEE.DELAYED - - const DEATH_REGISTRATION_TARGET = offlineData.config.DEATH.REGISTRATION_TARGET - const DEATH_ON_TIME_FEE = offlineData.config.DEATH.FEE.ON_TIME - const DEATH_DELAYED_FEE = offlineData.config.DEATH.FEE.DELAYED - - const MARRIAGE_REGISTRATION_TARGET = - offlineData.config.MARRIAGE.REGISTRATION_TARGET - const MARRIAGE_ON_TIME_FEE = offlineData.config.MARRIAGE.FEE.ON_TIME - const MARRIAGE_DELAYED_FEE = offlineData.config.MARRIAGE.FEE.DELAYED - - const birthRanges = [ - { start: 0, end: BIRTH_REGISTRATION_TARGET, value: BIRTH_ON_TIME_FEE }, - { - start: BIRTH_REGISTRATION_TARGET + 1, - end: BIRTH_LATE_REGISTRATION_TARGET, - value: BIRTH_LATE_FEE - }, - { start: BIRTH_LATE_REGISTRATION_TARGET + 1, value: BIRTH_DELAYED_FEE } - ] +function getDayRanges( + offlineData: IOfflineData, + certificate: ICertificate +): IRange[] { + const templateConfig = offlineData.templates.certificates.find( + (x) => x.id === certificate.certificateTemplateId + ) + switch (templateConfig?.event) { + case EventType.Birth: { + const BIRTH_REGISTRATION_TARGET = + offlineData.config.BIRTH.REGISTRATION_TARGET + const BIRTH_LATE_REGISTRATION_TARGET = + offlineData.config.BIRTH.LATE_REGISTRATION_TARGET + const BIRTH_ON_TIME_FEE = templateConfig?.fee.onTime + const BIRTH_LATE_FEE = templateConfig?.fee.late + const BIRTH_DELAYED_FEE = templateConfig?.fee.delayed + const birthRanges = [ + { start: 0, end: BIRTH_REGISTRATION_TARGET, value: BIRTH_ON_TIME_FEE }, + { + start: BIRTH_REGISTRATION_TARGET + 1, + end: BIRTH_LATE_REGISTRATION_TARGET, + value: BIRTH_LATE_FEE + }, + { start: BIRTH_LATE_REGISTRATION_TARGET + 1, value: BIRTH_DELAYED_FEE } + ] + return birthRanges + } - const deathRanges = [ - { start: 0, end: DEATH_REGISTRATION_TARGET, value: DEATH_ON_TIME_FEE }, - { start: DEATH_REGISTRATION_TARGET + 1, value: DEATH_DELAYED_FEE } - ] + case EventType.Death: { + const DEATH_REGISTRATION_TARGET = + offlineData.config.DEATH.REGISTRATION_TARGET + const DEATH_ON_TIME_FEE = templateConfig?.fee.onTime + const DEATH_DELAYED_FEE = templateConfig?.fee.delayed - const marriageRanges = [ - { - start: 0, - end: MARRIAGE_REGISTRATION_TARGET, - value: MARRIAGE_ON_TIME_FEE - }, - { start: MARRIAGE_REGISTRATION_TARGET + 1, value: MARRIAGE_DELAYED_FEE } - ] + const deathRanges = [ + { start: 0, end: DEATH_REGISTRATION_TARGET, value: DEATH_ON_TIME_FEE }, + { start: DEATH_REGISTRATION_TARGET + 1, value: DEATH_DELAYED_FEE } + ] + return deathRanges + } + case EventType.Marriage: { + const MARRIAGE_REGISTRATION_TARGET = + offlineData.config.MARRIAGE.REGISTRATION_TARGET + const MARRIAGE_ON_TIME_FEE = templateConfig?.fee.onTime + const MARRIAGE_DELAYED_FEE = templateConfig?.fee.delayed + const marriageRanges = [ + { + start: 0, + end: MARRIAGE_REGISTRATION_TARGET, + value: MARRIAGE_ON_TIME_FEE + }, + { start: MARRIAGE_REGISTRATION_TARGET + 1, value: MARRIAGE_DELAYED_FEE } + ] - return { - rangeData: { - [EventType.Birth]: birthRanges, - [EventType.Death]: deathRanges, - [EventType.Marriage]: marriageRanges + return marriageRanges } + default: + return [] } } function getValue( offlineData: IOfflineData, - event: EventType, + certificate: ICertificate, check: number ): IRange['value'] { - const rangeByEvent = getDayRanges(offlineData).rangeData[event] as IRange[] + const rangeByEvent = getDayRanges(offlineData, certificate) as IRange[] const foundRange = rangeByEvent.find((range) => range.end ? check >= range.start && check <= range.end : check >= range.start ) - return foundRange ? foundRange.value : rangeByEvent[0].value + return foundRange ? foundRange.value : rangeByEvent[0]?.value || 0 } export function calculateDaysFromToday(doE: string) { @@ -162,10 +169,12 @@ export function calculatePrice( event: EventType, eventDate: string, registeredDate: string, - offlineData: IOfflineData + offlineData: IOfflineData, + certificate: ICertificate ) { + if (!certificate) return 0 const days = calculateDays(eventDate, registeredDate) - const result = getValue(offlineData, event, days) + const result = getValue(offlineData, certificate, days) return result } @@ -220,13 +229,13 @@ export function getServiceMessage( } export function isFreeOfCost( - event: EventType, + certificate: ICertificate, eventDate: string, registeredDate: string, offlineData: IOfflineData ): boolean { const days = calculateDays(eventDate, registeredDate) - const result = getValue(offlineData, event, days) + const result = getValue(offlineData, certificate, days) return result === 0 } diff --git a/packages/client/src/views/RecordAudit/ActionDetailsModal.tsx b/packages/client/src/views/RecordAudit/ActionDetailsModal.tsx index 6e6cc3baac7..b2e8cd407f2 100644 --- a/packages/client/src/views/RecordAudit/ActionDetailsModal.tsx +++ b/packages/client/src/views/RecordAudit/ActionDetailsModal.tsx @@ -452,6 +452,27 @@ const ActionDetailsModalListTable = ({ width: 100 } ] + + const selectedCertificateTemplate = [ + { + key: 'certTemplate', + label: intl.formatMessage( + certificateMessages.selectedCertificateTemplateLabel + ), + width: 200 + } + ] + + const certificateTemplateMessageDescriptor = + offlineData.templates?.certificates?.find( + (x) => x.id === actionDetailsData.certificateTemplateId + )?.label + + const selectedCertificateTemplateName = { + certTemplate: certificateTemplateMessageDescriptor + ? intl.formatMessage(certificateTemplateMessageDescriptor) + : '' + } const pageChangeHandler = (cp: number) => setCurrentPage(cp) const content = prepareComments(actionDetailsData, draft) const requesterLabel = requesterLabelMapper( @@ -609,6 +630,17 @@ const ActionDetailsModalListTable = ({ onPageChange={pageChangeHandler} /> )} + {!isEmpty(collectorData) && !!actionDetailsData.certificateTemplateId && ( + + )} {/* Matched to */} {actionDetailsData.potentialDuplicates && diff --git a/packages/client/typings/window.d.ts b/packages/client/typings/window.d.ts index c67120df95b..fa123a9b073 100644 --- a/packages/client/typings/window.d.ts +++ b/packages/client/typings/window.d.ts @@ -16,11 +16,6 @@ interface Window { BIRTH: { REGISTRATION_TARGET: number LATE_REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - LATE: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } CONFIG_API_URL: string @@ -35,18 +30,10 @@ interface Window { } DEATH: { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } MARRIAGE: { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } FEATURES: { diff --git a/packages/commons/src/fhir/extension.ts b/packages/commons/src/fhir/extension.ts index 002c5bc2897..50607bfc166 100644 --- a/packages/commons/src/fhir/extension.ts +++ b/packages/commons/src/fhir/extension.ts @@ -128,6 +128,10 @@ export type StringExtensionType = { valueString?: string valueBoolean: boolean } + 'http://opencrvs.org/specs/extension/certificateTemplateId': { + url: 'http://opencrvs.org/specs/extension/certificateTemplateId' + valueString?: string + } 'http://opencrvs.org/specs/extension/regLastOffice': { url: 'http://opencrvs.org/specs/extension/regLastOffice' valueReference: { reference: ResourceIdentifier } diff --git a/packages/commons/src/fhir/transformers/input.ts b/packages/commons/src/fhir/transformers/input.ts index 43890b41993..d55e67ab300 100644 --- a/packages/commons/src/fhir/transformers/input.ts +++ b/packages/commons/src/fhir/transformers/input.ts @@ -207,7 +207,7 @@ interface Certificate { collector?: RelatedPerson hasShowedVerifiedDocument?: boolean payments?: Array - data?: string + certificateTemplateId: string } interface Deceased { deceased?: boolean diff --git a/packages/commons/src/fhir/transformers/utils.ts b/packages/commons/src/fhir/transformers/utils.ts index 15e62866d4b..267689253dd 100644 --- a/packages/commons/src/fhir/transformers/utils.ts +++ b/packages/commons/src/fhir/transformers/utils.ts @@ -60,7 +60,6 @@ import { isObservation, isURLReference, urlReferenceToResourceIdentifier, - BundleEntryWithFullUrl, findEntryFromBundle } from '..' @@ -743,9 +742,9 @@ export function setInformantReference( throw new Error(`${sectionCode} not found in composition!`) } const personSectionEntry = section.entry[0] - const personEntry = fhirBundle.entry.find( - (entry): entry is BundleEntryWithFullUrl => - entry.fullUrl === personSectionEntry.reference + const personEntry = findEntryFromBundle( + fhirBundle, + personSectionEntry.reference ) if (!personEntry) { return diff --git a/packages/config/src/handlers/application/applicationConfigHandler.test.ts b/packages/config/src/handlers/application/applicationConfigHandler.test.ts index ce69652eb3e..8fa569a6aa1 100644 --- a/packages/config/src/handlers/application/applicationConfigHandler.test.ts +++ b/packages/config/src/handlers/application/applicationConfigHandler.test.ts @@ -37,11 +37,6 @@ const mockConfig = { BIRTH: { REGISTRATION_TARGET: 45, LATE_REGISTRATION_TARGET: 365, - FEE: { - ON_TIME: 0, - LATE: 0, - DELAYED: 0 - }, PRINT_IN_ADVANCE: true }, COUNTRY_LOGO: { @@ -54,18 +49,10 @@ const mockConfig = { }, DEATH: { REGISTRATION_TARGET: 45, - FEE: { - ON_TIME: 0, - DELAYED: 0 - }, PRINT_IN_ADVANCE: true }, MARRIAGE: { REGISTRATION_TARGET: 45, - FEE: { - ON_TIME: 0, - DELAYED: 0 - }, PRINT_IN_ADVANCE: true }, PHONE_NUMBER_PATTERN: '^0(7|9)[0-9]{8}$', diff --git a/packages/config/src/handlers/application/applicationConfigHandler.ts b/packages/config/src/handlers/application/applicationConfigHandler.ts index 798fc5187fb..6c3f0991a36 100644 --- a/packages/config/src/handlers/application/applicationConfigHandler.ts +++ b/packages/config/src/handlers/application/applicationConfigHandler.ts @@ -35,7 +35,7 @@ export default async function configHandler( ) { try { const [certificates, config, systems] = await Promise.all([ - getCertificates(request, h), + getCertificatesConfig(request, h), getApplicationConfig(request, h), getSystems(request, h) ]) @@ -53,7 +53,10 @@ export default async function configHandler( } } -async function getCertificates(request: Hapi.Request, h: Hapi.ResponseToolkit) { +async function getCertificatesConfig( + request: Hapi.Request, + h: Hapi.ResponseToolkit +) { const authToken = getToken(request) const decodedOrError = pipe(authToken, verifyToken) if (decodedOrError._tag === 'Left') { @@ -67,12 +70,18 @@ async function getCertificates(request: Hapi.Request, h: Hapi.ResponseToolkit) { scope.includes(RouteScope.VALIDATE) || scope.includes(RouteScope.NATLSYSADMIN)) ) { - return Promise.all( - (['birth', 'death', 'marriage'] as const).map(async (event) => { - const response = await getEventCertificate(event, getToken(request)) - return response - }) - ) + const url = new URL(`/certificates`, env.COUNTRY_CONFIG_URL).toString() + + const res = await fetch(url, { + headers: { Authorization: `Bearer ${authToken}` } + }) + + if (!res.ok) { + throw new Error( + `Failed to fetch certificates configuration: ${res.statusText} ${url}` + ) + } + return res.json() } return [] } @@ -86,27 +95,6 @@ async function getConfigFromCountry(authToken?: string) { return res.json() } -async function getEventCertificate( - event: 'birth' | 'death' | 'marriage', - authToken: string -) { - const url = new URL( - `/certificates/${event}.svg`, - env.COUNTRY_CONFIG_URL - ).toString() - - const res = await fetch(url, { - headers: { Authorization: `Bearer ${authToken}` } - }) - - if (!res.ok) { - throw new Error(`Failed to fetch ${event} certificate: ${res.statusText}`) - } - const responseText = await res.text() - - return { svgCode: responseText, event } -} - async function getApplicationConfig( request?: Hapi.Request, h?: Hapi.ResponseToolkit @@ -172,37 +160,18 @@ const applicationConfigResponseValidation = Joi.object({ .keys({ REGISTRATION_TARGET: Joi.number().required(), LATE_REGISTRATION_TARGET: Joi.number().required(), - FEE: Joi.object() - .keys({ - ON_TIME: Joi.number().required(), - LATE: Joi.number().required(), - DELAYED: Joi.number().required() - }) - .required(), PRINT_IN_ADVANCE: Joi.boolean().required() }) .required(), DEATH: Joi.object() .keys({ REGISTRATION_TARGET: Joi.number().required(), - FEE: Joi.object() - .keys({ - ON_TIME: Joi.number().required(), - DELAYED: Joi.number().required() - }) - .required(), PRINT_IN_ADVANCE: Joi.boolean().required() }) .required(), MARRIAGE: Joi.object() .keys({ REGISTRATION_TARGET: Joi.number().required(), - FEE: Joi.object() - .keys({ - ON_TIME: Joi.number().required(), - DELAYED: Joi.number().required() - }) - .required(), PRINT_IN_ADVANCE: Joi.boolean().required() }) .required(), diff --git a/packages/config/src/models/config.ts b/packages/config/src/models/config.ts index ff78df0bb2c..740818f885e 100644 --- a/packages/config/src/models/config.ts +++ b/packages/config/src/models/config.ts @@ -12,27 +12,14 @@ import { model, Schema, Document } from 'mongoose' interface IBirth { REGISTRATION_TARGET: number LATE_REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - LATE: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } interface IDeath { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } interface IMarriage { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } interface ICurrency { @@ -66,29 +53,16 @@ export interface IApplicationConfigurationModel extends Document { const birthSchema = new Schema({ REGISTRATION_TARGET: { type: Number, default: 45 }, LATE_REGISTRATION_TARGET: { type: Number, default: 365 }, - FEE: { - ON_TIME: Number, - LATE: Number, - DELAYED: Number - }, PRINT_IN_ADVANCE: { type: Boolean, default: true } }) const deathSchema = new Schema({ REGISTRATION_TARGET: { type: Number, default: 45 }, - FEE: { - ON_TIME: Number, - DELAYED: Number - }, PRINT_IN_ADVANCE: { type: Boolean, default: true } }) const marriageSchema = new Schema({ REGISTRATION_TARGET: { type: Number, default: 45 }, - FEE: { - ON_TIME: { type: Number, default: 10 }, - DELAYED: { type: Number, default: 45 } - }, PRINT_IN_ADVANCE: { type: Boolean, default: true } }) diff --git a/packages/gateway/src/features/registration/schema.graphql b/packages/gateway/src/features/registration/schema.graphql index 6a1d16902e9..1fbbcb56987 100644 --- a/packages/gateway/src/features/registration/schema.graphql +++ b/packages/gateway/src/features/registration/schema.graphql @@ -159,6 +159,7 @@ type History { requester: String requesterOther: String hasShowedVerifiedDocument: Boolean + certificateTemplateId: String noSupportingDocumentationRequired: Boolean otherReason: String #This doesn't resolve to the System model properly rather @@ -418,14 +419,14 @@ input CertificateInput { collector: RelatedPersonInput hasShowedVerifiedDocument: Boolean payments: [PaymentInput] - data: String + certificateTemplateId: String } type Certificate { # -> Document Reference collector: RelatedPerson # -> .extension hasShowedVerifiedDocument: Boolean # -> .extension payments: [Payment] # -> .extension - data: String # -> .content.attachment.data base64 + certificateTemplateId: String } input QuestionnaireQuestionInput { diff --git a/packages/gateway/src/features/registration/type-resolvers.ts b/packages/gateway/src/features/registration/type-resolvers.ts index 231adb35b62..edeb78a2aa1 100644 --- a/packages/gateway/src/features/registration/type-resolvers.ts +++ b/packages/gateway/src/features/registration/type-resolvers.ts @@ -1236,6 +1236,13 @@ export const typeResolvers: GQLResolver = { } return false + }, + certificateTemplateId(docRef: DocumentReference, _) { + const certificateTemplateId = findExtension( + `${OPENCRVS_SPECIFICATION_URL}extension/certificateTemplateId`, + docRef.extension as Extension[] + ) + return certificateTemplateId?.valueString } }, Identifier: { @@ -1401,7 +1408,6 @@ export const typeResolvers: GQLResolver = { `${OPENCRVS_SPECIFICATION_URL}extension/hasShowedVerifiedDocument`, task.extension as Extension[] ) - if (hasShowedDocument?.valueString) { return Boolean(hasShowedDocument?.valueString) } @@ -1412,7 +1418,13 @@ export const typeResolvers: GQLResolver = { return false }, - + certificateTemplateId: (task: Task) => { + const certificateTemplateId = findExtension( + `${OPENCRVS_SPECIFICATION_URL}extension/certificateTemplateId`, + task.extension as Extension[] + ) + return certificateTemplateId?.valueString + }, noSupportingDocumentationRequired: (task: Task) => { const hasShowedDocument = findExtension( NO_SUPPORTING_DOCUMENTATION_REQUIRED, diff --git a/packages/gateway/src/graphql/schema.d.ts b/packages/gateway/src/graphql/schema.d.ts index e9382585d55..18acb49229b 100644 --- a/packages/gateway/src/graphql/schema.d.ts +++ b/packages/gateway/src/graphql/schema.d.ts @@ -827,6 +827,7 @@ export interface GQLHistory { requester?: string requesterOther?: string hasShowedVerifiedDocument?: boolean + certificateTemplateId?: string noSupportingDocumentationRequired?: boolean otherReason?: string system?: GQLIntegratedSystem @@ -1345,7 +1346,7 @@ export interface GQLCertificate { collector?: GQLRelatedPerson hasShowedVerifiedDocument?: boolean payments?: Array - data?: string + certificateTemplateId?: string } export interface GQLDuplicatesInfo { @@ -1698,7 +1699,7 @@ export interface GQLCertificateInput { collector?: GQLRelatedPersonInput hasShowedVerifiedDocument?: boolean payments?: Array - data?: string + certificateTemplateId?: string } export interface GQLIdentityInput { @@ -6667,6 +6668,7 @@ export interface GQLHistoryTypeResolver { requester?: HistoryToRequesterResolver requesterOther?: HistoryToRequesterOtherResolver hasShowedVerifiedDocument?: HistoryToHasShowedVerifiedDocumentResolver + certificateTemplateId?: HistoryToCertificateTemplateIdResolver noSupportingDocumentationRequired?: HistoryToNoSupportingDocumentationRequiredResolver otherReason?: HistoryToOtherReasonResolver system?: HistoryToSystemResolver @@ -6786,6 +6788,18 @@ export interface HistoryToHasShowedVerifiedDocumentResolver< ): TResult } +export interface HistoryToCertificateTemplateIdResolver< + TParent = any, + TResult = any +> { + ( + parent: TParent, + args: {}, + context: Context, + info: GraphQLResolveInfo + ): TResult +} + export interface HistoryToNoSupportingDocumentationRequiredResolver< TParent = any, TResult = any @@ -8374,7 +8388,7 @@ export interface GQLCertificateTypeResolver { collector?: CertificateToCollectorResolver hasShowedVerifiedDocument?: CertificateToHasShowedVerifiedDocumentResolver payments?: CertificateToPaymentsResolver - data?: CertificateToDataResolver + certificateTemplateId?: CertificateToCertificateTemplateIdResolver } export interface CertificateToCollectorResolver { @@ -8407,7 +8421,10 @@ export interface CertificateToPaymentsResolver { ): TResult } -export interface CertificateToDataResolver { +export interface CertificateToCertificateTemplateIdResolver< + TParent = any, + TResult = any +> { ( parent: TParent, args: {}, diff --git a/packages/gateway/src/graphql/schema.graphql b/packages/gateway/src/graphql/schema.graphql index a46a4924920..af29cf90c26 100644 --- a/packages/gateway/src/graphql/schema.graphql +++ b/packages/gateway/src/graphql/schema.graphql @@ -964,6 +964,7 @@ type History { requester: String requesterOther: String hasShowedVerifiedDocument: Boolean + certificateTemplateId: String noSupportingDocumentationRequired: Boolean otherReason: String system: IntegratedSystem @@ -1439,7 +1440,7 @@ type Certificate { collector: RelatedPerson hasShowedVerifiedDocument: Boolean payments: [Payment] - data: String + certificateTemplateId: String } type DuplicatesInfo { @@ -1791,7 +1792,7 @@ input CertificateInput { collector: RelatedPersonInput hasShowedVerifiedDocument: Boolean payments: [PaymentInput] - data: String + certificateTemplateId: String } input IdentityInput { diff --git a/packages/metrics/src/configApi.ts b/packages/metrics/src/configApi.ts index 2a56205689f..caca8aa1238 100644 --- a/packages/metrics/src/configApi.ts +++ b/packages/metrics/src/configApi.ts @@ -21,28 +21,15 @@ import { interface IBirth { REGISTRATION_TARGET: number LATE_REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - LATE: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } interface IDeath { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } interface IMarriage { REGISTRATION_TARGET: number - FEE: { - ON_TIME: number - DELAYED: number - } PRINT_IN_ADVANCE: boolean } export interface ICountryLogo { diff --git a/packages/migration/src/migrations/application-config/20241002232752-remove-certificate-collection.ts b/packages/migration/src/migrations/application-config/20241002232752-remove-certificate-collection.ts new file mode 100644 index 00000000000..5bbe022ece3 --- /dev/null +++ b/packages/migration/src/migrations/application-config/20241002232752-remove-certificate-collection.ts @@ -0,0 +1,66 @@ +/* + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at https://mozilla.org/MPL/2.0/. + * + * OpenCRVS is also distributed under the terms of the Civil Registration + * & Healthcare Disclaimer located at http://opencrvs.org/license. + * + * Copyright (C) The OpenCRVS Authors located at https://github.com/opencrvs/opencrvs-core/blob/master/AUTHORS. + */ + +import { Db, MongoClient } from 'mongodb' + +export const up = async (db: Db, client: MongoClient) => { + const session = client.startSession() + try { + await session.withTransaction(async () => { + const collectionExists = await db + .listCollections({ name: 'certificates' }) + .hasNext() + + if (collectionExists) { + await db.collection('certificates').drop() + console.log('Certificates collection removed successfully') + } else { + console.log('Certificates collection does not exist, skipping removal') + } + }) + } catch (error) { + console.error( + 'Error occurred while removing certificates collection:', + error + ) + throw error + } finally { + session.endSession() + } +} + +export const down = async (db: Db, client: MongoClient) => { + const session = client.startSession() + try { + await session.withTransaction(async () => { + const collectionExists = await db + .listCollections({ name: 'certificates' }) + .hasNext() + + if (!collectionExists) { + await db.createCollection('certificates') + console.log('Certificates collection recreated successfully') + } else { + console.log( + 'Certificates collection already exists, skipping recreation' + ) + } + }) + } catch (error) { + console.error( + 'Error occurred while recreating certificates collection:', + error + ) + throw error + } finally { + session.endSession() + } +} diff --git a/packages/migration/src/migrations/hearth/20241029171702-add-certificateTemplateId-to-missing-docs.ts b/packages/migration/src/migrations/hearth/20241029171702-add-certificateTemplateId-to-missing-docs.ts new file mode 100644 index 00000000000..9b3f6fbd842 --- /dev/null +++ b/packages/migration/src/migrations/hearth/20241029171702-add-certificateTemplateId-to-missing-docs.ts @@ -0,0 +1,157 @@ +/* + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at https://mozilla.org/MPL/2.0/. + * + * OpenCRVS is also distributed under the terms of the Civil Registration + * & Healthcare Disclaimer located at http://opencrvs.org/license. + * + * Copyright (C) The OpenCRVS Authors located at https://github.com/opencrvs/opencrvs-core/blob/master/AUTHORS. + */ +import { Db, MongoClient } from 'mongodb' + +export const up = async (db: Db, client: MongoClient) => { + const session = client.startSession() + try { + await session.withTransaction(async () => { + const bulkOps = [ + { + $match: { + 'extension.url': { + $ne: 'http://opencrvs.org/specs/extension/certificateTemplateId' + } + } + }, + { + $set: { + certType: { + $arrayElemAt: [ + { + $filter: { + input: '$type.coding', + as: 'coding', + cond: { + $regexMatch: { + input: '$$coding.system', + regex: /certificate-type/ + } + } + } + }, + 0 + ] + } + } + }, + { + $set: { + certificateTemplateId: { + $switch: { + branches: [ + { + case: { $eq: ['$certType.code', 'BIRTH'] }, + then: 'birth-certificate' + }, + { + case: { $eq: ['$certType.code', 'DEATH'] }, + then: 'death-certificate' + }, + { + case: { $eq: ['$certType.code', 'MARRIAGE'] }, + then: 'marriage-certificate' + } + ], + default: null + } + } + } + }, + { + $match: { + certificateTemplateId: { $ne: null } + } + }, + { + $set: { + extension: { + $concatArrays: [ + '$extension', + [ + { + url: 'http://opencrvs.org/specs/extension/certificateTemplateId', + valueString: '$certificateTemplateId' + } + ] + ] + } + } + }, + { + $unset: ['certType', 'certificateTemplateId'] + }, + { + $merge: { + into: 'DocumentReference', + whenMatched: 'replace' + } + } + ] + + await db.collection('DocumentReference').aggregate(bulkOps).toArray() + }) + } catch (error) { + console.error('Error occurred while updating document references:', error) + throw error + } finally { + session.endSession() + } +} + +export const down = async (db: Db, client: MongoClient) => { + const session = client.startSession() + try { + await session.withTransaction(async () => { + const bulkOps = [ + { + $match: { + extension: { + $elemMatch: { + url: 'http://opencrvs.org/specs/extension/certificateTemplateId' + } + } + } + }, + { + $set: { + extension: { + $filter: { + input: '$extension', + as: 'ext', + cond: { + $ne: [ + '$$ext.url', + 'http://opencrvs.org/specs/extension/certificateTemplateId' + ] + } + } + } + } + }, + { + $merge: { + into: 'DocumentReference', + whenMatched: 'merge', + whenNotMatched: 'discard' + } + } + ] + await db.collection('DocumentReference').aggregate(bulkOps).toArray() + console.log('Reverted certificateTemplateId extension from all documents') + }) + } catch (error) { + console.error('Error occurred while reverting document references:', error) + throw error + } finally { + session.endSession() + } +} diff --git a/packages/workflow/src/documents.ts b/packages/workflow/src/documents.ts index debe9bc2105..a9fcc4626eb 100644 --- a/packages/workflow/src/documents.ts +++ b/packages/workflow/src/documents.ts @@ -36,24 +36,6 @@ export async function uploadFileToMinio( return res.refUrl } -async function uploadSVGToMinio( - fileData: string, - authHeader: IAuthHeader -): Promise { - const suffix = '/upload-svg' - const request = { - method: 'POST', - headers: { - ...authHeader, - 'Content-Type': 'image/svg+xml' - }, - body: fileData - } - const result = await fetch(`${DOCUMENTS_URL}${suffix}`, request) - const res = await result.json() - return res.refUrl -} - export async function uploadCertificateAttachmentsToDocumentsStore< T extends CertifyInput | IssueInput >(certificateDetails: T, authHeader: IAuthHeader): Promise { @@ -65,12 +47,6 @@ export async function uploadCertificateAttachmentsToDocumentsStore< affidavit.data = await uploadFileToMinio(affidavit.data, authHeader) } } - if ('data' in certificateDetails) { - certificateDetails.data = await uploadSVGToMinio( - certificateDetails.data, - authHeader - ) - } return certificateDetails } diff --git a/packages/workflow/src/records/fhir.ts b/packages/workflow/src/records/fhir.ts index 89558fb4806..881d2d40959 100644 --- a/packages/workflow/src/records/fhir.ts +++ b/packages/workflow/src/records/fhir.ts @@ -218,6 +218,7 @@ export function createDocumentReferenceEntryForCertificate( temporaryRelatedPersonId: UUID, eventType: EVENT_TYPE, hasShowedVerifiedDocument: boolean, + certificateTemplateId?: string, attachmentUrl?: string, paymentUrl?: URNReference | ResourceIdentifier ): BundleEntry { @@ -240,6 +241,10 @@ export function createDocumentReferenceEntryForCertificate( url: 'http://opencrvs.org/specs/extension/hasShowedVerifiedDocument', valueBoolean: hasShowedVerifiedDocument }, + { + url: 'http://opencrvs.org/specs/extension/certificateTemplateId', + valueString: certificateTemplateId + }, ...(paymentUrl ? [ { @@ -912,8 +917,20 @@ export async function createUnassignedTask( return unassignedTask } -export function createCertifiedTask(previousTask: SavedTask): SavedTask { - return createNewTaskResource(previousTask, [], 'CERTIFIED') +export function createCertifiedTask( + previousTask: SavedTask, + certificateTemplateId: string +): SavedTask { + return createNewTaskResource( + previousTask, + [ + { + url: 'http://opencrvs.org/specs/extension/certificateTemplateId', + valueString: certificateTemplateId + } + ], + 'CERTIFIED' + ) } export function createIssuedTask( diff --git a/packages/workflow/src/records/handler/certify.test.ts b/packages/workflow/src/records/handler/certify.test.ts index 0b77f5614b5..8b86cc43abf 100644 --- a/packages/workflow/src/records/handler/certify.test.ts +++ b/packages/workflow/src/records/handler/certify.test.ts @@ -111,7 +111,7 @@ describe('Certify record endpoint', () => { event: 'BIRTH', certificate: { hasShowedVerifiedDocument: true, - data: 'data:application/pdf;base64,AXDWYZ', + certificateTemplateId: 'birth-certificate', collector: { relationship: 'INFORMANT' } @@ -215,7 +215,7 @@ describe('Certify record endpoint', () => { event: 'BIRTH', certificate: { hasShowedVerifiedDocument: true, - data: 'data:application/pdf;base64,AXDWYZ', + certificateTemplateId: 'birth-certificate', collector: { relationship: 'Other', otherRelationship: 'Uncle', diff --git a/packages/workflow/src/records/handler/issue.test.ts b/packages/workflow/src/records/handler/issue.test.ts index b7804701913..7a7ad897b59 100644 --- a/packages/workflow/src/records/handler/issue.test.ts +++ b/packages/workflow/src/records/handler/issue.test.ts @@ -112,6 +112,7 @@ describe('Issue record endpoint', () => { collector: { relationship: 'INFORMANT' }, + certificateTemplateId: 'birth-certificate', payment: { type: 'MANUAL', amount: 100, diff --git a/packages/workflow/src/records/state-transitions.ts b/packages/workflow/src/records/state-transitions.ts index 68f0dbb8a71..d35b3b30b94 100644 --- a/packages/workflow/src/records/state-transitions.ts +++ b/packages/workflow/src/records/state-transitions.ts @@ -977,7 +977,10 @@ export async function toCertified( certificateDetails: CertifyInput ): Promise { const previousTask = getTaskFromSavedBundle(record) - const taskWithoutPractitionerExtensions = createCertifiedTask(previousTask) + const taskWithoutPractitionerExtensions = createCertifiedTask( + previousTask, + certificateDetails.certificateTemplateId + ) const [certifiedTask, practitionerResourcesBundle] = await withPractitionerDetails(taskWithoutPractitionerExtensions, token) @@ -996,7 +999,7 @@ export async function toCertified( temporaryRelatedPersonId, eventType, certificateDetails.hasShowedVerifiedDocument, - certificateDetails.data + certificateDetails.certificateTemplateId ) const certificateSection: CompositionSection = { @@ -1082,6 +1085,7 @@ export async function toIssued( temporaryRelatedPersonId, eventType, certificateDetails.hasShowedVerifiedDocument, + certificateDetails.certificateTemplateId, undefined, paymentReconciliation.fullUrl ) diff --git a/packages/workflow/src/records/validations.ts b/packages/workflow/src/records/validations.ts index fb81ff2b635..477bc3106f7 100644 --- a/packages/workflow/src/records/validations.ts +++ b/packages/workflow/src/records/validations.ts @@ -18,7 +18,7 @@ export const CertifyRequestSchema = z.object({ event: z.custom(), certificate: z.object({ hasShowedVerifiedDocument: z.boolean(), - data: z.string(), + certificateTemplateId: z.string(), collector: z .object({ relationship: z.string(), @@ -63,9 +63,9 @@ const PaymentSchema = z.object({ export const IssueRequestSchema = z.object({ event: z.custom(), - certificate: CertifyRequestSchema.shape.certificate - .omit({ data: true }) - .and(z.object({ payment: PaymentSchema })) + certificate: CertifyRequestSchema.shape.certificate.and( + z.object({ payment: PaymentSchema }) + ) }) export const ChangedValuesInput = z.array(