forked from openemr/openemr
-
Notifications
You must be signed in to change notification settings - Fork 0
/
RestControllerHelper.php
444 lines (399 loc) · 19.9 KB
/
RestControllerHelper.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
<?php
/**
* RestControllerHelper
*
* @package OpenEMR
* @link http://www.open-emr.org
* @author Matthew Vita <[email protected]>
* @copyright Copyright (c) 2018 Matthew Vita <[email protected]>
* @license https://github.com/openemr/openemr/blob/master/LICENSE GNU General Public License 3
*/
namespace OpenEMR\RestControllers;
use OpenEMR\Common\Logging\SystemLogger;
use OpenEMR\Events\RestApiExtend\RestApiCreateEvent;
use OpenEMR\Events\RestApiExtend\RestApiResourceServiceEvent;
use OpenEMR\Events\RestApiExtend\RestApiScopeEvent;
use OpenEMR\FHIR\R4\FHIRDomainResource\FHIROperationDefinition;
use OpenEMR\FHIR\R4\FHIRElement\FHIROperationKind;
use OpenEMR\FHIR\R4\FHIRElement\FHIROperationParameterUse;
use OpenEMR\Services\FHIR\IResourceSearchableService;
use OpenEMR\Services\FHIR\UtilsService;
use OpenEMR\Services\Search\FhirSearchParameterDefinition;
use OpenEMR\Services\Search\SearchFieldType;
use OpenEMR\FHIR\R4\FHIRDomainResource\FHIRPatient;
use OpenEMR\FHIR\R4\FHIRElement\FHIRCanonical;
use OpenEMR\FHIR\R4\FHIRElement\FHIRCode;
use OpenEMR\FHIR\R4\FHIRElement\FHIRExtension;
use OpenEMR\FHIR\R4\FHIRElement\FHIRRestfulCapabilityMode;
use OpenEMR\FHIR\R4\FHIRElement\FHIRTypeRestfulInteraction;
use OpenEMR\FHIR\R4\FHIRResource;
use OpenEMR\FHIR\R4\FHIRResource\FHIRCapabilityStatement\FHIRCapabilityStatementInteraction;
use OpenEMR\FHIR\R4\FHIRResource\FHIRCapabilityStatement\FHIRCapabilityStatementOperation;
use OpenEMR\FHIR\R4\FHIRResource\FHIRCapabilityStatement\FHIRCapabilityStatementResource;
use OpenEMR\FHIR\R4\FHIRResource\FHIRCapabilityStatement\FHIRCapabilityStatementRest;
use OpenEMR\Services\FHIR\IResourceUSCIGProfileService;
use OpenEMR\Validators\ProcessingResult;
class RestControllerHelper
{
/**
* The resource endpoint names we want to skip over.
*/
const IGNORE_ENDPOINT_RESOURCES = ['.well-known', 'metadata'];
/**
* The default FHIR services class namespace
* TODO: should we build a fhir service locator class? There are two places this is now used, in this class and
* in the FhirProvenanceService...
*/
const FHIR_SERVICES_NAMESPACE = "OpenEMR\\Services\\FHIR\\Fhir";
const DEFAULT_STRUCTURE_DEFINITION = "http://hl7.org/fhir/StructureDefinition/";
// @see https://www.hl7.org/fhir/search.html#table
const FHIR_SEARCH_CONTROL_PARAM_REV_INCLUDE_PROVENANCE = "Provenance:target";
private $restURL = "";
public function __construct($restAPIUrl = "")
{
$this->restURL = $restAPIUrl;
}
/**
* Configures the HTTP status code and payload returned within a response.
*
* @param $serviceResult
* @param $customRespPayload
* @param $idealStatusCode
* @return null
*/
public static function responseHandler($serviceResult, $customRespPayload, $idealStatusCode)
{
if ($serviceResult) {
http_response_code($idealStatusCode);
if ($customRespPayload) {
return $customRespPayload;
}
return $serviceResult;
}
// if no result is present return a 404 with a null response
http_response_code(404);
return null;
}
public static function validationHandler($validationResult)
{
if (property_exists($validationResult, 'isValid') && !$validationResult->isValid()) {
http_response_code(400);
$validationMessages = null;
if (property_exists($validationResult, 'getValidationMessages')) {
$validationMessages = $validationResult->getValidationMessages();
} else {
$validationMessages = $validationResult->getMessages();
}
return $validationMessages;
}
return null;
}
/**
* Parses a service processing result for standard Apis to determine the appropriate HTTP status code and response format
* for a request.
*
* The response body has a uniform structure with the following top level keys:
* - validationErrors
* - internalErrors
* - data
*
* The response data key conveys the data payload for a response. The payload is either a "top level" array for a
* single result, or an array for multiple results.
*
* @param $processingResult - The service processing result.
* @param $successStatusCode - The HTTP status code to return for a successful operation that completes without error.
* @param $isMultipleResultResponse - Indicates if the response contains multiple results.
* @return array[]
*/
public static function handleProcessingResult(ProcessingResult $processingResult, $successStatusCode, $isMultipleResultResponse = false): array
{
$httpResponseBody = [
"validationErrors" => [],
"internalErrors" => [],
"data" => [],
"links" => []
];
if (!$processingResult->isValid()) {
http_response_code(400);
$httpResponseBody["validationErrors"] = $processingResult->getValidationMessages();
(new SystemLogger())->debug("RestControllerHelper::handleProcessingResult() 400 error", ['validationErrors' => $processingResult->getValidationMessages()]);
} elseif ($processingResult->hasInternalErrors()) {
http_response_code(500);
$httpResponseBody["internalErrors"] = $processingResult->getInternalErrors();
(new SystemLogger())->debug("RestControllerHelper::handleProcessingResult() 500 error", ['internalErrors' => $processingResult->getValidationMessages()]);
} else {
http_response_code($successStatusCode ?? 0);
$dataResult = $processingResult->getData();
$recordsCount = count($dataResult);
(new SystemLogger())->debug("RestControllerHelper::handleFhirProcessingResult() Records found", ['count' => $recordsCount]);
if (!$isMultipleResultResponse) {
$dataResult = ($recordsCount === 0) ? [] : $dataResult[0];
} else {
$pagination = $processingResult->getPagination();
// if site_addr_oauth is not set then we set it to be empty so we can handle relative urls
$bundleUrl = ($GLOBALS['site_addr_oath'] ?? '') . ($_SERVER['REDIRECT_URL'] ?? '');
$getParams = $_GET;
// cleanup _limit and _offset
unset($getParams['_limit']);
unset($getParams['_offset']);
$queryParams = http_build_query($getParams);
$pagination->setSearchUri($bundleUrl . '?' . $queryParams);
$httpResponseBody['links'] = $processingResult->getPagination()->getLinks();
}
$httpResponseBody["data"] = $dataResult;
}
return $httpResponseBody;
}
/**
* Parses a service processing result for FHIR endpoints to determine the appropriate HTTP status code and response format
* for a request.
*
* The response body has a normal Fhir Resource json:
*
* @param $processingResult - The service processing result.
* @param $successStatusCode - The HTTP status code to return for a successful operation that completes without error.
* @return array|mixed
*/
public static function handleFhirProcessingResult(ProcessingResult $processingResult, $successStatusCode)
{
$httpResponseBody = [];
if (!$processingResult->isValid()) {
http_response_code(400);
$httpResponseBody["validationErrors"] = $processingResult->getValidationMessages();
(new SystemLogger())->debug("RestControllerHelper::handleFhirProcessingResult() 400 error", ['validationErrors' => $processingResult->getValidationMessages()]);
} elseif (count($processingResult->getData()) <= 0) {
http_response_code(404);
(new SystemLogger())->debug("RestControllerHelper::handleFhirProcessingResult() 404 records not found");
} elseif ($processingResult->hasInternalErrors()) {
http_response_code(500);
(new SystemLogger())->debug("RestControllerHelper::handleFhirProcessingResult() 500 error", ['internalErrors' => $processingResult->getValidationMessages()]);
$httpResponseBody["internalErrors"] = $processingResult->getInternalErrors();
} else {
http_response_code($successStatusCode);
$dataResult = $processingResult->getData();
(new SystemLogger())->debug("RestControllerHelper::handleFhirProcessingResult() Records found", ['count' => count($dataResult)]);
$httpResponseBody = $dataResult[0];
}
return $httpResponseBody;
}
public function setSearchParams($resource, FHIRCapabilityStatementResource $capResource, $service)
{
if (empty($service)) {
return; // nothing to do here as the service isn't defined.
}
if (!$service instanceof IResourceSearchableService) {
return; // nothing to do here as the source is not searchable.
}
if (empty($capResource->getSearchInclude())) {
$capResource->addSearchInclude('*');
}
if ($service instanceof IResourceUSCIGProfileService && empty($capResource->getSearchRevInclude())) {
$capResource->addSearchRevInclude(self::FHIR_SEARCH_CONTROL_PARAM_REV_INCLUDE_PROVENANCE);
}
$searchParams = $service->getSearchParams();
$searchParams = is_array($searchParams) ? $searchParams : [];
foreach ($searchParams as $fhirSearchField => $searchDefinition) {
/**
* @var FhirSearchParameterDefinition $searchDefinition
*/
$paramExists = false;
$type = $searchDefinition->getType();
if ($type == SearchFieldType::DATETIME) {
$type = 'date'; // fhir merges date and datetime into a single date for capability statement purposes.
}
foreach ($capResource->getSearchParam() as $searchParam) {
if (strcmp($searchParam->getName(), $fhirSearchField) == 0) {
$paramExists = true;
}
}
if (!$paramExists) {
$param = new FHIRResource\FHIRCapabilityStatement\FHIRCapabilityStatementSearchParam();
$param->setName($fhirSearchField);
$param->setType($type);
$capResource->addSearchParam($param);
}
}
}
/**
* Retrieves the fully qualified service class name for a given FHIR resource. It will only return a class that
* actually exists.
* @param $resource The name of the FHIR resource that we attempt to find the service class for.
* @param string $serviceClassNameSpace The namespace to find the class in. Defaults to self::FHIR_SERVICES_NAMESPACE
* @return string|null Returns the fully qualified name if the class is found, otherwise it returns null.
*/
public function getFullyQualifiedServiceClassForResource($resource, $serviceClassNameSpace = self::FHIR_SERVICES_NAMESPACE)
{
$serviceClass = $serviceClassNameSpace . $resource . "Service";
if (class_exists($serviceClass)) {
return $serviceClass;
}
return null;
}
public function addOperations($resource, $items, FHIRCapabilityStatementResource $capResource)
{
// TODO: @adunsulag we need to architect a more generic way of adding operations like we do with resources
$operation = end($items);
// we want to skip over anything that's not a resource $operation
// first check to make sure the operation is not already defined
// such as $bulkdata-status when we have both a POST and a DELETE rest route to the same operation
if (!empty($capResource->getOperation())) {
foreach ($capResource->getOperation() as $existingOperation) {
// this doesn't handle the $export operations
// TODO: is there a better way to handle all operations and not just things such as $bulkdata-status?
if ($existingOperation->getName() == $operation) {
return; // already exists so let's skip adding this operation
}
}
}
if ($operation == '$export') {
// operation definition must use the operation 'name'
// rest.resource.operation.name must come from the OperationDefinition's code attribute which in this case is 'export'
$definitionName = 'export';
$operationName = 'export';
if ($resource != '$export') {
$definitionName = strtolower($resource) . '-export';
}
// define export operation
$fhirOperation = new FHIRCapabilityStatementOperation();
$fhirOperation->setName($operationName);
$fhirOperation->setDefinition(new FHIRCanonical('http://hl7.org/fhir/uv/bulkdata/OperationDefinition/' . $definitionName));
$capResource->addOperation($fhirOperation);
} elseif ($operation === '$bulkdata-status') {
$fhirOperation = new FHIRCapabilityStatementOperation();
$fhirOperation->setName($operation);
$fhirOperation->setDefinition($this->restURL . '/OperationDefinition/$bulkdata-status');
$capResource->addOperation($fhirOperation);
// TODO: @adunsulag we should document in our capability statement how to use the bulkdata-status operation
} elseif ($operation === '$docref') {
$fhirOperation = new FHIRCapabilityStatementOperation();
$fhirOperation->setName($operation);
$fhirOperation->setDefinition(new FHIRCanonical('http://hl7.org/fhir/us/core/OperationDefinition/docref'));
$capResource->addOperation($fhirOperation);
} elseif (is_string($operation) && strpos($operation, '$') === 0) {
(new SystemLogger())->debug("Found operation that is not supported in system", ['resource' => $resource, 'operation' => $operation, 'items' => $items]);
}
}
public function addRequestMethods($items, FHIRCapabilityStatementResource $capResource)
{
$reqMethod = trim($items[0], " ");
$numberItems = count($items);
$code = "";
// we want to skip over $export operations.
if (end($items) === '$export') {
return;
}
// now setup our interaction types
if (strcmp($reqMethod, "GET") == 0) {
if (!empty(preg_match('/:/', $items[$numberItems - 1]))) {
$code = "read";
} else {
$code = "search-type";
}
} elseif (strcmp($reqMethod, "POST") == 0) {
$code = "create";
} elseif (strcmp($reqMethod, "PUT") == 0) {
$code = "update";
} elseif (strcmp($reqMethod, "DELETE") == 0) {
$code = "delete";
}
if (!empty($code)) {
$interaction = new FHIRCapabilityStatementInteraction();
$restfulInteraction = new FHIRTypeRestfulInteraction();
$restfulInteraction->setValue($code);
$interaction->setCode($restfulInteraction);
$capResource->addInteraction($interaction);
}
}
public function getCapabilityRESTObject($routes, $serviceClassNameSpace = self::FHIR_SERVICES_NAMESPACE, $structureDefinition = self::DEFAULT_STRUCTURE_DEFINITION): FHIRCapabilityStatementRest
{
$restItem = new FHIRCapabilityStatementRest();
$mode = new FHIRRestfulCapabilityMode();
$mode->setValue('server');
$restItem->setMode($mode);
$resourcesHash = array();
foreach ($routes as $key => $function) {
$items = explode("/", $key);
if ($serviceClassNameSpace == self::FHIR_SERVICES_NAMESPACE) {
// FHIR routes always have the resource at $items[2]
$resource = $items[2];
} else {
// API routes do not always have the resource at $items[2]
if (count($items) < 5) {
$resource = $items[2];
} elseif (count($items) < 7) {
$resource = $items[4];
if (substr($resource, 0, 1) === ':') {
// special behavior needed for the API portal route
$resource = $items[3];
}
} else { // count($items) < 9
$resource = $items[6];
}
}
if (!in_array($resource, self::IGNORE_ENDPOINT_RESOURCES)) {
$service = null;
$serviceClass = $this->getFullyQualifiedServiceClassForResource($resource, $serviceClassNameSpace);
$serviceClass = self::filterServiceClassForResource($resource, $serviceClass);
if (!empty($serviceClass)) {
$service = new $serviceClass();
}
// typically the type is the same as the resource, but for operations it will be our OperationDefinition
$type = self::getResourceTypeForResource($resource);
$capResource = $resourcesHash[$type] ?? null;
if (empty($capResource)) {
$capResource = new FHIRCapabilityStatementResource();
// make it explicit that we do not let the user use their own resource ids to create a new resource
// in the PUT/update operation.
$capResource->setUpdateCreate(false);
$capResource->setType(new FHIRCode($type));
$capResource->setProfile(new FHIRCanonical($structureDefinition . $type));
if ($service instanceof IResourceUSCIGProfileService) {
$profileUris = $service->getProfileURIs();
foreach ($profileUris as $uri) {
$capResource->addSupportedProfile(new FHIRCanonical($uri));
}
}
// per the specification type must be unique in the capability statement
$resourcesHash[$type] = $capResource;
}
$this->setSearchParams($resource, $capResource, $service);
$this->addRequestMethods($items, $capResource);
$this->addOperations($resource, $items, $capResource);
}
}
foreach ($resourcesHash as $resource => $capResource) {
$restItem->addResource($capResource);
}
return $restItem;
}
/**
* Given a resource we've pulled from our rest route definitions figure out the type from our valueset
* for the resource type: http://hl7.org/fhir/2021Mar/valueset-resource-types.html
* @param string $resource
* @return string
*/
private static function getResourceTypeForResource(string $resource)
{
$firstChar = $resource[0] ?? '';
if ($firstChar == '$') {
return 'OperationDefinition';
}
return $resource;
}
/**
* Fires off a system event for the given API resource to filter the serviceClass. This gives module writers
* the opportunity to extend the api, add / remove Implementation Guide profiles and declare different API conformance
* @param $resource The api resource that was parsed
* @param $serviceClass The service class that was found by default in the system or null if none was found
* @return string|null The filtered service class property
*/
private static function filterServiceClassForResource(string $resource, ?string $serviceClass)
{
if (!empty($GLOBALS['kernel'])) {
$dispatcher = $GLOBALS['kernel']->getEventDispatcher();
$event = $dispatcher->dispatch(new RestApiResourceServiceEvent($resource, $serviceClass), RestApiResourceServiceEvent::EVENT_HANDLE);
return $event->getServiceClass();
}
return $serviceClass;
}
}