From 213dc289b8fa2bba5ee95a5889a1e149818d1079 Mon Sep 17 00:00:00 2001 From: Kyle Huang Junyuan Date: Fri, 25 Mar 2022 15:48:23 +0800 Subject: [PATCH] feat: recovery healthcert schema v2.0 (#72) --- src/index.ts | 11 +- .../gov/moh/recovery-healthcert/2.0/index.ts | 3 + .../2.0/sample-document.json | 157 +++++++++++++ .../moh/recovery-healthcert/2.0/schema.json | 47 ++++ .../recovery-healthcert/2.0/schema.test.ts | 222 ++++++++++++++++++ 5 files changed, 439 insertions(+), 1 deletion(-) create mode 100644 src/sg/gov/moh/recovery-healthcert/2.0/index.ts create mode 100644 src/sg/gov/moh/recovery-healthcert/2.0/sample-document.json create mode 100644 src/sg/gov/moh/recovery-healthcert/2.0/schema.json create mode 100644 src/sg/gov/moh/recovery-healthcert/2.0/schema.test.ts diff --git a/src/index.ts b/src/index.ts index 3c01516..546dfcf 100644 --- a/src/index.ts +++ b/src/index.ts @@ -4,6 +4,15 @@ import * as pdtHealthCertV2 from "./__generated__/sg/gov/moh/pdt-healthcert/2.0/ import * as vaccinationHealthcert from "./__generated__/sg/gov/moh/vaccination-healthcert/1.0/schema"; import * as pdtHealthCertV2Schema from "./sg/gov/moh/pdt-healthcert/2.0"; +import * as recHealthCertV2Schema from "./sg/gov/moh/recovery-healthcert/2.0"; import * as fhirSchema from "./sg/gov/moh/fhir/4.0.1"; -export { geekout, notarise, pdtHealthCertV2, vaccinationHealthcert, pdtHealthCertV2Schema, fhirSchema }; +export { + geekout, + notarise, + pdtHealthCertV2, + vaccinationHealthcert, + pdtHealthCertV2Schema, + recHealthCertV2Schema, + fhirSchema +}; diff --git a/src/sg/gov/moh/recovery-healthcert/2.0/index.ts b/src/sg/gov/moh/recovery-healthcert/2.0/index.ts new file mode 100644 index 0000000..33d11d7 --- /dev/null +++ b/src/sg/gov/moh/recovery-healthcert/2.0/index.ts @@ -0,0 +1,3 @@ +import schema from "./schema.json"; + +export { schema }; diff --git a/src/sg/gov/moh/recovery-healthcert/2.0/sample-document.json b/src/sg/gov/moh/recovery-healthcert/2.0/sample-document.json new file mode 100644 index 0000000..609fd90 --- /dev/null +++ b/src/sg/gov/moh/recovery-healthcert/2.0/sample-document.json @@ -0,0 +1,157 @@ +{ + "id": "76caf3f9-5591-4ef1-b756-1cb47a76dede", + "version": "rec-healthcert-v2.0", + "type": "PCR", + "validFrom": "2022-03-12T04:30:35.065Z", + "validUntil": "2023-08-28T04:30:35.065Z", + "fhirVersion": "4.0.1", + "fhirBundle": { + "resourceType": "Bundle", + "type": "collection", + "entry": [ + { + "fullUrl": "urn:uuid:ba7b7c8d-c509-4d9d-be4e-f99b6de29e23", + "resource": { + "resourceType": "Patient", + "extension": [ + { + "url": "http://hl7.org/fhir/StructureDefinition/patient-nationality", + "extension": [ + { + "url": "code", + "valueCodeableConcept": { + "text": "Patient Nationality", + "coding": [{ "system": "urn:iso:std:iso:3166", "code": "SG" }] + } + } + ] + } + ], + "identifier": [ + { + "id": "PPN", + "type": { + "coding": [ + { + "system": "http://terminology.hl7.org/CodeSystem/v2-0203", + "code": "PPN", + "display": "Passport Number" + } + ] + }, + "value": "E7831177G" + }, + { "id": "NRIC-FIN", "value": "S9098989Z" } + ], + "name": [{ "text": "Tan Chen Chen" }], + "gender": "female", + "birthDate": "1990-01-15" + } + }, + { + "fullUrl": "urn:uuid:7729970e-ab26-469f-b3e5-36a42ec24146", + "resource": { + "resourceType": "Observation", + "specimen": { + "type": "Specimen", + "reference": "urn:uuid:0275bfaf-48fb-44e0-80cd-9c504f80e6ae" + }, + "identifier": [ + { + "id": "ACSN", + "type": { + "coding": [ + { + "system": "http://terminology.hl7.org/CodeSystem/v2-0203", + "code": "ACSN", + "display": "Accession ID" + } + ] + }, + "value": "123456789" + } + ], + "category": [ + { + "coding": [ + { + "system": "http://snomed.info/sct", + "code": "840539006", + "display": "COVID-19" + } + ] + } + ], + "code": { + "coding": [ + { + "system": "http://loinc.org", + "code": "94531-1", + "display": "SARS-CoV-2 (COVID-19) RNA panel - Respiratory specimen by NAA with probe detection" + } + ] + }, + "valueCodeableConcept": { + "coding": [ + { + "system": "http://snomed.info/sct", + "code": "260385009", + "display": "Positive" + } + ] + }, + "effectiveDateTime": "2022-03-01T04:30:35.065Z", + "status": "final" + } + }, + { + "fullUrl": "urn:uuid:0275bfaf-48fb-44e0-80cd-9c504f80e6ae", + "resource": { + "resourceType": "Specimen", + "type": { + "coding": [ + { + "system": "http://snomed.info/sct", + "code": "258500001", + "display": "Nasopharyngeal swab" + } + ] + }, + "collection": { "collectedDateTime": "2022-03-01T04:30:35.065Z" } + } + }, + { + "fullUrl": "urn:uuid:bc7065ee-42aa-473a-a614-afd8a7b30b1e", + "resource": { + "resourceType": "Organization", + "name": "Ministry of Health (MOH)", + "type": [ + { + "coding": [ + { + "system": "http://terminology.hl7.org/CodeSystem/organization-type", + "code": "govt", + "display": "Government" + } + ] + } + ], + "contact": [ + { + "telecom": [ + { "system": "url", "value": "https://www.moh.gov.sg" }, + { "system": "phone", "value": "+6563259220" } + ], + "address": { + "type": "physical", + "use": "work", + "text": "Ministry of Health, 16 College Road, College of Medicine Building, Singapore 169854" + } + } + ] + } + } + ] + }, + "logo": "" +} diff --git a/src/sg/gov/moh/recovery-healthcert/2.0/schema.json b/src/sg/gov/moh/recovery-healthcert/2.0/schema.json new file mode 100644 index 0000000..0673800 --- /dev/null +++ b/src/sg/gov/moh/recovery-healthcert/2.0/schema.json @@ -0,0 +1,47 @@ +{ + "$schema": "http://json-schema.org/draft-07/schema#", + "$id": "https://schemata.openattestation.com/sg/gov/moh/pdt-healthcert/2.0/schema.json", + "type": "object", + "required": ["id", "version", "type", "validFrom", "validUntil", "fhirVersion", "fhirBundle"], + "definitions": { + "RecTypes": { + "type": "string", + "enum": ["PCR", "ART", "SER"] + } + }, + "properties": { + "id": { + "type": "string", + "examples": ["TEST001"] + }, + "version": { + "type": "string", + "enum": ["rec-healthcert-v2.0"] + }, + "type": { + "$ref": "#/definitions/RecTypes" + }, + "logo": { + "type": "string", + "description": "base64 encoded image" + }, + "validFrom": { + "type": "string", + "format": "date-time", + "description": "Date and time from which the document is considered valid" + }, + "validUntil": { + "type": "string", + "format": "date-time", + "description": "Date and time after which the document is considered invalid" + }, + "fhirVersion": { + "type": "string", + "examples": ["4.0.1"] + }, + "fhirBundle": { + "$ref": "https://schemata.openattestation.com/sg/gov/moh/fhir/4.0.1/lite-schema.json#/definitions/Bundle", + "description": "FHIR bundle for a collection of resources. Each resource and the entire bundle should be compliant against the base spec of FHIR. You may use a validator like: https://inferno.healthit.gov/validator/" + } + } +} diff --git a/src/sg/gov/moh/recovery-healthcert/2.0/schema.test.ts b/src/sg/gov/moh/recovery-healthcert/2.0/schema.test.ts new file mode 100644 index 0000000..e08f631 --- /dev/null +++ b/src/sg/gov/moh/recovery-healthcert/2.0/schema.test.ts @@ -0,0 +1,222 @@ +import Ajv from "ajv"; +import { cloneDeep, omit, set } from "lodash"; +import axios from "axios"; + +import liteFhirSchema from "../../fhir/4.0.1/lite-schema.json"; + +import schema from "./schema.json"; +import sampleDocument from "./sample-document.json"; + +function loadSchema(uri: string) { + return axios.get(uri).then(res => { + return res.data; + }); +} + +let validator: Ajv.ValidateFunction; + +describe("schema", () => { + beforeAll(async () => { + const ajv = new Ajv({ allErrors: true, loadSchema: loadSchema }); + ajv.addSchema(liteFhirSchema); + validator = await ajv.compileAsync(schema); + }); + + it("should work with valid json", () => { + expect(validator(sampleDocument)).toBe(true); + }); + + it("should work with valid single type", () => { + expect(validator(set(cloneDeep(sampleDocument), "type", "PCR"))).toBe(true); + expect(validator(set(cloneDeep(sampleDocument), "type", "ART"))).toBe(true); + expect(validator(set(cloneDeep(sampleDocument), "type", "SER"))).toBe(true); + }); + + it("should fail when type is missing", () => { + const document = omit(cloneDeep(sampleDocument), "type"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": "", + "keyword": "required", + "message": "should have required property 'type'", + "params": Object { + "missingProperty": "type", + }, + "schemaPath": "#/required", + }, + ] + `); + }); + + it("should fail when type is not part of enum", () => { + const document = set(cloneDeep(sampleDocument), "type", "foo"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": ".type", + "keyword": "enum", + "message": "should be equal to one of the allowed values", + "params": Object { + "allowedValues": Array [ + "PCR", + "ART", + "SER", + ], + }, + "schemaPath": "#/definitions/RecTypes/enum", + }, + ] + `); + }); + + it("should fail when type is multi an array", () => { + expect(validator(set(cloneDeep(sampleDocument), "type", ["PCR", "SER"]))).toBe(false); + }); + + it("should fail when id is missing", () => { + const document = omit(cloneDeep(sampleDocument), "id"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": "", + "keyword": "required", + "message": "should have required property 'id'", + "params": Object { + "missingProperty": "id", + }, + "schemaPath": "#/required", + }, + ] + `); + }); + + it("should fail when version is missing", () => { + const document = omit(cloneDeep(sampleDocument), "version"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": "", + "keyword": "required", + "message": "should have required property 'version'", + "params": Object { + "missingProperty": "version", + }, + "schemaPath": "#/required", + }, + ] + `); + }); + + it("should fail when validFrom is missing", () => { + const document = omit(cloneDeep(sampleDocument), "validFrom"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": "", + "keyword": "required", + "message": "should have required property 'validFrom'", + "params": Object { + "missingProperty": "validFrom", + }, + "schemaPath": "#/required", + }, + ] + `); + }); + + it("should fail when validFrom is not a date-time", () => { + const document = set(cloneDeep(sampleDocument), "validFrom", "FOO"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": ".validFrom", + "keyword": "format", + "message": "should match format \\"date-time\\"", + "params": Object { + "format": "date-time", + }, + "schemaPath": "#/properties/validFrom/format", + }, + ] + `); + }); + + it("should fail when validUntil is missing", () => { + const document = omit(cloneDeep(sampleDocument), "validUntil"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": "", + "keyword": "required", + "message": "should have required property 'validUntil'", + "params": Object { + "missingProperty": "validUntil", + }, + "schemaPath": "#/required", + }, + ] + `); + }); + + it("should fail when validUntil is not a date-time", () => { + const document = set(cloneDeep(sampleDocument), "validUntil", "FOO"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": ".validUntil", + "keyword": "format", + "message": "should match format \\"date-time\\"", + "params": Object { + "format": "date-time", + }, + "schemaPath": "#/properties/validUntil/format", + }, + ] + `); + }); + + it("should fail when fhirVersion is missing", () => { + const document = omit(cloneDeep(sampleDocument), "fhirVersion"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": "", + "keyword": "required", + "message": "should have required property 'fhirVersion'", + "params": Object { + "missingProperty": "fhirVersion", + }, + "schemaPath": "#/required", + }, + ] + `); + }); + + it("should fail when fhirBundle is missing", () => { + const document = omit(cloneDeep(sampleDocument), "fhirBundle"); + expect(validator(document)).toBe(false); + expect(validator.errors).toMatchInlineSnapshot(` + Array [ + Object { + "dataPath": "", + "keyword": "required", + "message": "should have required property 'fhirBundle'", + "params": Object { + "missingProperty": "fhirBundle", + }, + "schemaPath": "#/required", + }, + ] + `); + }); +});