@openhi/constructs 0.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +3 -0
- package/lib/app/index.d.ts +4 -0
- package/lib/app/index.js +21 -0
- package/lib/app/open-hi-app.d.ts +85 -0
- package/lib/app/open-hi-app.js +127 -0
- package/lib/app/open-hi-environment.d.ts +59 -0
- package/lib/app/open-hi-environment.js +72 -0
- package/lib/app/open-hi-service.d.ts +169 -0
- package/lib/app/open-hi-service.js +195 -0
- package/lib/app/open-hi-stage.d.ts +71 -0
- package/lib/app/open-hi-stage.js +70 -0
- package/lib/components/acm/root-wildcard-certificate.d.ts +15 -0
- package/lib/components/acm/root-wildcard-certificate.js +35 -0
- package/lib/components/api-gateway/core-http-api.d.ts +10 -0
- package/lib/components/api-gateway/core-http-api.js +44 -0
- package/lib/components/api-gateway/http-lambda-integration-no-permissions.d.ts +18 -0
- package/lib/components/api-gateway/http-lambda-integration-no-permissions.js +26 -0
- package/lib/components/app-sync/core-graphql-api.d.ts +12 -0
- package/lib/components/app-sync/core-graphql-api.js +54 -0
- package/lib/components/auth.d.ts +75 -0
- package/lib/components/auth.js +100 -0
- package/lib/components/cognito/core-user-pool-client.d.ts +10 -0
- package/lib/components/cognito/core-user-pool-client.js +47 -0
- package/lib/components/cognito/core-user-pool-domain.d.ts +10 -0
- package/lib/components/cognito/core-user-pool-domain.js +41 -0
- package/lib/components/cognito/core-user-pool-kms-key.d.ts +10 -0
- package/lib/components/cognito/core-user-pool-kms-key.js +37 -0
- package/lib/components/cognito/core-user-pool.d.ts +10 -0
- package/lib/components/cognito/core-user-pool.js +54 -0
- package/lib/components/core.d.ts +102 -0
- package/lib/components/core.js +79 -0
- package/lib/components/dynamodb/dynamo-db-data-store.d.ts +33 -0
- package/lib/components/dynamodb/dynamo-db-data-store.js +107 -0
- package/lib/components/event-bridge/data-event-bus.d.ts +19 -0
- package/lib/components/event-bridge/data-event-bus.js +34 -0
- package/lib/components/event-bridge/ops-event-bus.d.ts +19 -0
- package/lib/components/event-bridge/ops-event-bus.js +34 -0
- package/lib/components/global.d.ts +36 -0
- package/lib/components/global.js +63 -0
- package/lib/components/index.d.ts +1 -0
- package/lib/components/index.js +18 -0
- package/lib/components/route-53/child-hosted-zone.d.ts +20 -0
- package/lib/components/route-53/child-hosted-zone.js +48 -0
- package/lib/components/route-53/root-hosted-zone.d.ts +10 -0
- package/lib/components/route-53/root-hosted-zone.js +20 -0
- package/lib/components/ssm/discoverable-string-parameter.d.ts +59 -0
- package/lib/components/ssm/discoverable-string-parameter.js +50 -0
- package/lib/components/ssm/index.d.ts +1 -0
- package/lib/components/ssm/index.js +18 -0
- package/lib/data/dynamo/ehr/r4/Patient.d.ts +180 -0
- package/lib/data/dynamo/ehr/r4/Patient.js +192 -0
- package/lib/data/dynamo/ehr/r4/ehr-r4-data-service.d.ts +162 -0
- package/lib/data/dynamo/ehr/r4/ehr-r4-data-service.js +37 -0
- package/lib/data/hello-world.d.ts +39 -0
- package/lib/data/hello-world.js +59 -0
- package/lib/data/import-patient-with-dynalite.d.ts +1 -0
- package/lib/data/import-patient-with-dynalite.js +87 -0
- package/lib/data/import-patient.d.ts +47 -0
- package/lib/data/import-patient.js +158 -0
- package/lib/data/lambda/rest-api-lambda.d.ts +13 -0
- package/lib/data/lambda/rest-api-lambda.handler.d.ts +1 -0
- package/lib/data/lambda/rest-api-lambda.handler.js +10 -0
- package/lib/data/lambda/rest-api-lambda.js +22 -0
- package/lib/data/middleware/open-hi-context.d.ts +13 -0
- package/lib/data/middleware/open-hi-context.js +31 -0
- package/lib/data/rest-api/ehr/r4/Patient.d.ts +16 -0
- package/lib/data/rest-api/ehr/r4/Patient.js +234 -0
- package/lib/data/rest-api/rest-api-local.d.ts +1 -0
- package/lib/data/rest-api/rest-api-local.js +8 -0
- package/lib/data/rest-api/rest-api-mockdata.d.ts +7 -0
- package/lib/data/rest-api/rest-api-mockdata.js +585 -0
- package/lib/data/rest-api/rest-api.d.ts +3 -0
- package/lib/data/rest-api/rest-api.js +26 -0
- package/lib/index.d.ts +3 -0
- package/lib/index.js +20 -0
- package/lib/lib/compression.d.ts +27 -0
- package/lib/lib/compression.js +87 -0
- package/lib/services/index.d.ts +5 -0
- package/lib/services/index.js +22 -0
- package/lib/services/open-hi-auth-service.d.ts +31 -0
- package/lib/services/open-hi-auth-service.js +31 -0
- package/lib/services/open-hi-core-service.d.ts +15 -0
- package/lib/services/open-hi-core-service.js +38 -0
- package/lib/services/open-hi-data-service.d.ts +18 -0
- package/lib/services/open-hi-data-service.js +18 -0
- package/lib/services/open-hi-global-service.d.ts +15 -0
- package/lib/services/open-hi-global-service.js +44 -0
- package/lib/services/open-hi-rest-api-service.d.ts +17 -0
- package/lib/services/open-hi-rest-api-service.js +107 -0
- package/package.json +67 -0
|
@@ -0,0 +1,234 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.patientRouter = void 0;
|
|
7
|
+
exports.listPatients = listPatients;
|
|
8
|
+
exports.getPatientById = getPatientById;
|
|
9
|
+
exports.createPatient = createPatient;
|
|
10
|
+
exports.updatePatient = updatePatient;
|
|
11
|
+
exports.deletePatient = deletePatient;
|
|
12
|
+
const express_1 = __importDefault(require("express"));
|
|
13
|
+
const compression_1 = require("../../../../lib/compression");
|
|
14
|
+
const ehr_r4_data_service_1 = require("../../../dynamo/ehr/r4/ehr-r4-data-service");
|
|
15
|
+
const import_patient_1 = require("../../../import-patient");
|
|
16
|
+
const BASE_PATH = "/ehr/r4/Patient";
|
|
17
|
+
const router = express_1.default.Router();
|
|
18
|
+
exports.patientRouter = router;
|
|
19
|
+
const SK = "CURRENT";
|
|
20
|
+
const TABLE_NAME = process.env.DYNAMO_TABLE_NAME ?? "jesttesttable";
|
|
21
|
+
/**
|
|
22
|
+
* GET /ehr/r4/Patient — search/list: returns a FHIR Bundle (searchset) from the data store.
|
|
23
|
+
* Uses GSI4 (Resource Type Index) to list all Patients in the workspace without a table scan.
|
|
24
|
+
* @see {@link https://github.com/codedrifters/openhi/blob/main/sites/www-docs/content/architecture/dynamodb-single-table-design.md | DynamoDB Single-Table Design} — GSI4, Query and Access Rules (no scans).
|
|
25
|
+
*/
|
|
26
|
+
async function listPatients(req, res) {
|
|
27
|
+
const { tenantId, workspaceId } = req.openhiContext;
|
|
28
|
+
const service = (0, ehr_r4_data_service_1.getEhrR4DataService)(TABLE_NAME);
|
|
29
|
+
try {
|
|
30
|
+
const result = await service.entities.patient.query
|
|
31
|
+
.gsi4({ tenantId, workspaceId })
|
|
32
|
+
.go();
|
|
33
|
+
const entries = (result.data ?? []).map((item) => {
|
|
34
|
+
const resource = JSON.parse((0, compression_1.decompressResource)(item.resource));
|
|
35
|
+
return {
|
|
36
|
+
fullUrl: `${BASE_PATH}/${item.id}`,
|
|
37
|
+
resource: { ...resource, id: item.id },
|
|
38
|
+
};
|
|
39
|
+
});
|
|
40
|
+
const bundle = {
|
|
41
|
+
resourceType: "Bundle",
|
|
42
|
+
type: "searchset",
|
|
43
|
+
total: entries.length,
|
|
44
|
+
link: [{ relation: "self", url: BASE_PATH }],
|
|
45
|
+
entry: entries,
|
|
46
|
+
};
|
|
47
|
+
return res.json(bundle);
|
|
48
|
+
}
|
|
49
|
+
catch (err) {
|
|
50
|
+
console.error("GET /Patient list error:", err);
|
|
51
|
+
return res.status(500).json({
|
|
52
|
+
resourceType: "OperationOutcome",
|
|
53
|
+
issue: [
|
|
54
|
+
{
|
|
55
|
+
severity: "error",
|
|
56
|
+
code: "exception",
|
|
57
|
+
diagnostics: String(err),
|
|
58
|
+
},
|
|
59
|
+
],
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
router.get("/", listPatients);
|
|
64
|
+
async function getPatientById(req, res) {
|
|
65
|
+
const id = String(req.params.id);
|
|
66
|
+
const { tenantId, workspaceId } = req.openhiContext;
|
|
67
|
+
const service = (0, ehr_r4_data_service_1.getEhrR4DataService)(TABLE_NAME);
|
|
68
|
+
try {
|
|
69
|
+
const result = await service.entities.patient
|
|
70
|
+
.get({ tenantId, workspaceId, id, sk: "CURRENT" })
|
|
71
|
+
.go();
|
|
72
|
+
if (!result.data) {
|
|
73
|
+
return res.status(404).json({
|
|
74
|
+
resourceType: "OperationOutcome",
|
|
75
|
+
issue: [
|
|
76
|
+
{
|
|
77
|
+
severity: "error",
|
|
78
|
+
code: "not-found",
|
|
79
|
+
diagnostics: `Patient ${id} not found`,
|
|
80
|
+
},
|
|
81
|
+
],
|
|
82
|
+
});
|
|
83
|
+
}
|
|
84
|
+
const resource = JSON.parse((0, compression_1.decompressResource)(result.data.resource));
|
|
85
|
+
return res.json({ ...resource, id: result.data.id });
|
|
86
|
+
}
|
|
87
|
+
catch (err) {
|
|
88
|
+
console.error("GET Patient error:", err);
|
|
89
|
+
return res.status(500).json({
|
|
90
|
+
resourceType: "OperationOutcome",
|
|
91
|
+
issue: [
|
|
92
|
+
{
|
|
93
|
+
severity: "error",
|
|
94
|
+
code: "exception",
|
|
95
|
+
diagnostics: String(err),
|
|
96
|
+
},
|
|
97
|
+
],
|
|
98
|
+
});
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
/** GET /ehr/r4/Patient/:id — read: returns a single Patient resource from the data store or 404. */
|
|
102
|
+
router.get("/:id", getPatientById);
|
|
103
|
+
/** POST /ehr/r4/Patient — create: accepts Patient in body, persists via data store, returns 201. */
|
|
104
|
+
async function createPatient(req, res) {
|
|
105
|
+
const ctx = req.openhiContext;
|
|
106
|
+
const { tenantId, workspaceId, date, userId, username } = ctx;
|
|
107
|
+
const body = req.body;
|
|
108
|
+
const id = body?.id ?? `patient-${Date.now()}`;
|
|
109
|
+
const patient = {
|
|
110
|
+
...body,
|
|
111
|
+
resourceType: "Patient",
|
|
112
|
+
id,
|
|
113
|
+
meta: {
|
|
114
|
+
...(body?.meta ?? {}),
|
|
115
|
+
lastUpdated: date,
|
|
116
|
+
versionId: "1",
|
|
117
|
+
},
|
|
118
|
+
};
|
|
119
|
+
const options = {
|
|
120
|
+
tenantId,
|
|
121
|
+
workspaceId,
|
|
122
|
+
createdDate: date,
|
|
123
|
+
createdById: userId,
|
|
124
|
+
createdByName: username,
|
|
125
|
+
modifiedDate: date,
|
|
126
|
+
modifiedById: userId,
|
|
127
|
+
modifiedByName: username,
|
|
128
|
+
};
|
|
129
|
+
const service = (0, ehr_r4_data_service_1.getEhrR4DataService)(TABLE_NAME);
|
|
130
|
+
try {
|
|
131
|
+
const attrs = (0, import_patient_1.patientToPutAttrs)(patient, options);
|
|
132
|
+
await service.entities.patient
|
|
133
|
+
.put(attrs)
|
|
134
|
+
.go();
|
|
135
|
+
return res.status(201).location(`${BASE_PATH}/${id}`).json(patient);
|
|
136
|
+
}
|
|
137
|
+
catch (err) {
|
|
138
|
+
console.error("POST Patient error:", err);
|
|
139
|
+
return res.status(500).json({
|
|
140
|
+
resourceType: "OperationOutcome",
|
|
141
|
+
issue: [
|
|
142
|
+
{ severity: "error", code: "exception", diagnostics: String(err) },
|
|
143
|
+
],
|
|
144
|
+
});
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
router.post("/", createPatient);
|
|
148
|
+
/** PUT /ehr/r4/Patient/:id — update: accepts Patient in body, persists via data store, returns 200. */
|
|
149
|
+
async function updatePatient(req, res) {
|
|
150
|
+
const id = String(req.params.id);
|
|
151
|
+
const ctx = req.openhiContext;
|
|
152
|
+
const { tenantId, workspaceId, date, userId, username } = ctx;
|
|
153
|
+
const body = req.body;
|
|
154
|
+
const patient = {
|
|
155
|
+
...body,
|
|
156
|
+
resourceType: "Patient",
|
|
157
|
+
id,
|
|
158
|
+
meta: {
|
|
159
|
+
...(body?.meta ?? {}),
|
|
160
|
+
lastUpdated: date,
|
|
161
|
+
versionId: "2",
|
|
162
|
+
},
|
|
163
|
+
};
|
|
164
|
+
const service = (0, ehr_r4_data_service_1.getEhrR4DataService)(TABLE_NAME);
|
|
165
|
+
try {
|
|
166
|
+
const existing = await service.entities.patient
|
|
167
|
+
.get({ tenantId, workspaceId, id, sk: SK })
|
|
168
|
+
.go();
|
|
169
|
+
if (!existing.data) {
|
|
170
|
+
return res.status(404).json({
|
|
171
|
+
resourceType: "OperationOutcome",
|
|
172
|
+
issue: [
|
|
173
|
+
{
|
|
174
|
+
severity: "error",
|
|
175
|
+
code: "not-found",
|
|
176
|
+
diagnostics: `Patient ${id} not found`,
|
|
177
|
+
},
|
|
178
|
+
],
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
const existingMeta = existing.data.resource != null
|
|
182
|
+
? JSON.parse((0, compression_1.decompressResource)(existing.data.resource)).meta
|
|
183
|
+
: undefined;
|
|
184
|
+
const patientWithMeta = {
|
|
185
|
+
...patient,
|
|
186
|
+
meta: (0, import_patient_1.mergeAuditIntoMeta)(patient.meta ?? existingMeta, {
|
|
187
|
+
modifiedDate: date,
|
|
188
|
+
modifiedById: userId,
|
|
189
|
+
modifiedByName: username,
|
|
190
|
+
}),
|
|
191
|
+
};
|
|
192
|
+
await service.entities.patient
|
|
193
|
+
.patch({ tenantId, workspaceId, id, sk: SK })
|
|
194
|
+
.set({
|
|
195
|
+
resource: (0, compression_1.compressResource)(JSON.stringify(patientWithMeta)),
|
|
196
|
+
lastUpdated: date,
|
|
197
|
+
})
|
|
198
|
+
.go();
|
|
199
|
+
return res.json(patientWithMeta);
|
|
200
|
+
}
|
|
201
|
+
catch (err) {
|
|
202
|
+
console.error("PUT Patient error:", err);
|
|
203
|
+
return res.status(500).json({
|
|
204
|
+
resourceType: "OperationOutcome",
|
|
205
|
+
issue: [
|
|
206
|
+
{ severity: "error", code: "exception", diagnostics: String(err) },
|
|
207
|
+
],
|
|
208
|
+
});
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
router.put("/:id", updatePatient);
|
|
212
|
+
/** DELETE /ehr/r4/Patient/:id — delete: removes from data store, returns 204. */
|
|
213
|
+
async function deletePatient(req, res) {
|
|
214
|
+
const id = String(req.params.id);
|
|
215
|
+
const { tenantId, workspaceId } = req.openhiContext;
|
|
216
|
+
const service = (0, ehr_r4_data_service_1.getEhrR4DataService)(TABLE_NAME);
|
|
217
|
+
try {
|
|
218
|
+
await service.entities.patient
|
|
219
|
+
.delete({ tenantId, workspaceId, id, sk: SK })
|
|
220
|
+
.go();
|
|
221
|
+
return res.status(204).send();
|
|
222
|
+
}
|
|
223
|
+
catch (err) {
|
|
224
|
+
console.error("DELETE Patient error:", err);
|
|
225
|
+
return res.status(500).json({
|
|
226
|
+
resourceType: "OperationOutcome",
|
|
227
|
+
issue: [
|
|
228
|
+
{ severity: "error", code: "exception", diagnostics: String(err) },
|
|
229
|
+
],
|
|
230
|
+
});
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
router.delete("/:id", deletePatient);
|
|
234
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Patient.js","sourceRoot":"","sources":["../../../../../src/data/rest-api/ehr/r4/Patient.ts"],"names":[],"mappings":"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","sourcesContent":["import express, { Request, Response } from \"express\";\nimport {\n  compressResource,\n  decompressResource,\n} from \"../../../../lib/compression\";\nimport { getEhrR4DataService } from \"../../../dynamo/ehr/r4/ehr-r4-data-service\";\nimport {\n  mergeAuditIntoMeta,\n  patientToPutAttrs,\n  type ImportPatientOptions,\n} from \"../../../import-patient\";\n\nconst BASE_PATH = \"/ehr/r4/Patient\";\nconst router: express.Router = express.Router();\nconst SK = \"CURRENT\";\n\nconst TABLE_NAME = process.env.DYNAMO_TABLE_NAME ?? \"jesttesttable\";\n\n/**\n * GET /ehr/r4/Patient — search/list: returns a FHIR Bundle (searchset) from the data store.\n * Uses GSI4 (Resource Type Index) to list all Patients in the workspace without a table scan.\n * @see {@link https://github.com/codedrifters/openhi/blob/main/sites/www-docs/content/architecture/dynamodb-single-table-design.md | DynamoDB Single-Table Design} — GSI4, Query and Access Rules (no scans).\n */\nexport async function listPatients(\n  req: Request,\n  res: Response,\n): Promise<Response> {\n  const { tenantId, workspaceId } = req.openhiContext!;\n  const service = getEhrR4DataService(TABLE_NAME);\n\n  try {\n    const result = await service.entities.patient.query\n      .gsi4({ tenantId, workspaceId })\n      .go();\n\n    const entries = (result.data ?? []).map((item) => {\n      const resource = JSON.parse(decompressResource(item.resource)) as Record<\n        string,\n        unknown\n      >;\n      return {\n        fullUrl: `${BASE_PATH}/${item.id}`,\n        resource: { ...resource, id: item.id },\n      };\n    });\n    const bundle = {\n      resourceType: \"Bundle\",\n      type: \"searchset\",\n      total: entries.length,\n      link: [{ relation: \"self\", url: BASE_PATH }],\n      entry: entries,\n    };\n    return res.json(bundle);\n  } catch (err: unknown) {\n    console.error(\"GET /Patient list error:\", err);\n    return res.status(500).json({\n      resourceType: \"OperationOutcome\",\n      issue: [\n        {\n          severity: \"error\",\n          code: \"exception\",\n          diagnostics: String(err),\n        },\n      ],\n    });\n  }\n}\n\nrouter.get(\"/\", listPatients);\n\nexport async function getPatientById(\n  req: Request,\n  res: Response,\n): Promise<Response> {\n  const id = String(req.params.id);\n  const { tenantId, workspaceId } = req.openhiContext!;\n  const service = getEhrR4DataService(TABLE_NAME);\n\n  try {\n    const result = await service.entities.patient\n      .get({ tenantId, workspaceId, id, sk: \"CURRENT\" })\n      .go();\n\n    if (!result.data) {\n      return res.status(404).json({\n        resourceType: \"OperationOutcome\",\n        issue: [\n          {\n            severity: \"error\",\n            code: \"not-found\",\n            diagnostics: `Patient ${id} not found`,\n          },\n        ],\n      });\n    }\n\n    const resource = JSON.parse(\n      decompressResource(result.data.resource),\n    ) as Record<string, unknown>;\n    return res.json({ ...resource, id: result.data.id });\n  } catch (err: unknown) {\n    console.error(\"GET Patient error:\", err);\n    return res.status(500).json({\n      resourceType: \"OperationOutcome\",\n      issue: [\n        {\n          severity: \"error\",\n          code: \"exception\",\n          diagnostics: String(err),\n        },\n      ],\n    });\n  }\n}\n\n/** GET /ehr/r4/Patient/:id — read: returns a single Patient resource from the data store or 404. */\nrouter.get(\"/:id\", getPatientById);\n\n/** POST /ehr/r4/Patient — create: accepts Patient in body, persists via data store, returns 201. */\nexport async function createPatient(\n  req: Request,\n  res: Response,\n): Promise<Response> {\n  const ctx = req.openhiContext!;\n  const { tenantId, workspaceId, date, userId, username } = ctx;\n  const body = req.body as Record<string, unknown>;\n  const id = (body?.id as string) ?? `patient-${Date.now()}`;\n  const patient = {\n    ...body,\n    resourceType: \"Patient\",\n    id,\n    meta: {\n      ...((body?.meta as object) ?? {}),\n      lastUpdated: date,\n      versionId: \"1\",\n    },\n  } as { resourceType: string; id: string; meta?: { lastUpdated?: string } };\n  const options: ImportPatientOptions = {\n    tenantId,\n    workspaceId,\n    createdDate: date,\n    createdById: userId,\n    createdByName: username,\n    modifiedDate: date,\n    modifiedById: userId,\n    modifiedByName: username,\n  };\n  const service = getEhrR4DataService(TABLE_NAME);\n\n  try {\n    const attrs = patientToPutAttrs(patient, options);\n    await service.entities.patient\n      .put(\n        attrs as unknown as Parameters<typeof service.entities.patient.put>[0],\n      )\n      .go();\n    return res.status(201).location(`${BASE_PATH}/${id}`).json(patient);\n  } catch (err: unknown) {\n    console.error(\"POST Patient error:\", err);\n    return res.status(500).json({\n      resourceType: \"OperationOutcome\",\n      issue: [\n        { severity: \"error\", code: \"exception\", diagnostics: String(err) },\n      ],\n    });\n  }\n}\n\nrouter.post(\"/\", createPatient);\n\n/** PUT /ehr/r4/Patient/:id — update: accepts Patient in body, persists via data store, returns 200. */\nexport async function updatePatient(\n  req: Request,\n  res: Response,\n): Promise<Response> {\n  const id = String(req.params.id);\n  const ctx = req.openhiContext!;\n  const { tenantId, workspaceId, date, userId, username } = ctx;\n  const body = req.body as Record<string, unknown>;\n  const patient = {\n    ...body,\n    resourceType: \"Patient\",\n    id,\n    meta: {\n      ...((body?.meta as object) ?? {}),\n      lastUpdated: date,\n      versionId: \"2\",\n    },\n  };\n  const service = getEhrR4DataService(TABLE_NAME);\n\n  try {\n    const existing = await service.entities.patient\n      .get({ tenantId, workspaceId, id, sk: SK })\n      .go();\n    if (!existing.data) {\n      return res.status(404).json({\n        resourceType: \"OperationOutcome\",\n        issue: [\n          {\n            severity: \"error\",\n            code: \"not-found\",\n            diagnostics: `Patient ${id} not found`,\n          },\n        ],\n      });\n    }\n    const existingMeta =\n      existing.data.resource != null\n        ? (\n            JSON.parse(decompressResource(existing.data.resource)) as {\n              meta?: Record<string, unknown>;\n            }\n          ).meta\n        : undefined;\n    const patientWithMeta = {\n      ...patient,\n      meta: mergeAuditIntoMeta(\n        (patient.meta as Record<string, unknown> | undefined) ?? existingMeta,\n        {\n          modifiedDate: date,\n          modifiedById: userId,\n          modifiedByName: username,\n        },\n      ),\n    };\n    await service.entities.patient\n      .patch({ tenantId, workspaceId, id, sk: SK })\n      .set({\n        resource: compressResource(JSON.stringify(patientWithMeta)),\n        lastUpdated: date,\n      })\n      .go();\n    return res.json(patientWithMeta);\n  } catch (err: unknown) {\n    console.error(\"PUT Patient error:\", err);\n    return res.status(500).json({\n      resourceType: \"OperationOutcome\",\n      issue: [\n        { severity: \"error\", code: \"exception\", diagnostics: String(err) },\n      ],\n    });\n  }\n}\n\nrouter.put(\"/:id\", updatePatient);\n\n/** DELETE /ehr/r4/Patient/:id — delete: removes from data store, returns 204. */\nexport async function deletePatient(\n  req: Request,\n  res: Response,\n): Promise<Response> {\n  const id = String(req.params.id);\n  const { tenantId, workspaceId } = req.openhiContext!;\n  const service = getEhrR4DataService(TABLE_NAME);\n\n  try {\n    await service.entities.patient\n      .delete({ tenantId, workspaceId, id, sk: SK })\n      .go();\n    return res.status(204).send();\n  } catch (err: unknown) {\n    console.error(\"DELETE Patient error:\", err);\n    return res.status(500).json({\n      resourceType: \"OperationOutcome\",\n      issue: [\n        { severity: \"error\", code: \"exception\", diagnostics: String(err) },\n      ],\n    });\n  }\n}\n\nrouter.delete(\"/:id\", deletePatient);\n\nexport { router as patientRouter };\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
const rest_api_1 = require("./rest-api");
|
|
4
|
+
const port = 3000;
|
|
5
|
+
rest_api_1.app.listen(port, () => {
|
|
6
|
+
console.log(`POC app listening at http://localhost:${port}`);
|
|
7
|
+
});
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVzdC1hcGktbG9jYWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZGF0YS9yZXN0LWFwaS9yZXN0LWFwaS1sb2NhbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQUFBLHlDQUFpQztBQUVqQyxNQUFNLElBQUksR0FBRyxJQUFJLENBQUM7QUFFbEIsY0FBRyxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsR0FBRyxFQUFFO0lBQ3BCLE9BQU8sQ0FBQyxHQUFHLENBQUMseUNBQXlDLElBQUksRUFBRSxDQUFDLENBQUM7QUFDL0QsQ0FBQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBhcHAgfSBmcm9tIFwiLi9yZXN0LWFwaVwiO1xuXG5jb25zdCBwb3J0ID0gMzAwMDtcblxuYXBwLmxpc3Rlbihwb3J0LCAoKSA9PiB7XG4gIGNvbnNvbGUubG9nKGBQT0MgYXBwIGxpc3RlbmluZyBhdCBodHRwOi8vbG9jYWxob3N0OiR7cG9ydH1gKTtcbn0pO1xuIl19
|