001package org.hl7.fhir.r4.hapi.ctx; 002 003import ca.uhn.fhir.context.FhirContext; 004import ca.uhn.fhir.context.support.ConceptValidationOptions; 005import ca.uhn.fhir.context.support.IValidationSupport; 006import ca.uhn.fhir.context.support.ValidationSupportContext; 007import ca.uhn.fhir.i18n.Msg; 008import ca.uhn.fhir.rest.api.Constants; 009import ca.uhn.fhir.sl.cache.Cache; 010import ca.uhn.fhir.sl.cache.CacheFactory; 011import ca.uhn.fhir.system.HapiSystemProperties; 012import ca.uhn.fhir.util.CoverageIgnore; 013import org.apache.commons.lang3.Validate; 014import org.fhir.ucum.UcumService; 015import org.hl7.fhir.exceptions.FHIRException; 016import org.hl7.fhir.exceptions.TerminologyServiceException; 017import org.hl7.fhir.r4.context.IWorkerContext; 018import org.hl7.fhir.r4.formats.IParser; 019import org.hl7.fhir.r4.formats.ParserType; 020import org.hl7.fhir.r4.model.CodeSystem; 021import org.hl7.fhir.r4.model.CodeSystem.ConceptDefinitionComponent; 022import org.hl7.fhir.r4.model.CodeableConcept; 023import org.hl7.fhir.r4.model.Coding; 024import org.hl7.fhir.r4.model.ConceptMap; 025import org.hl7.fhir.r4.model.ElementDefinition.ElementDefinitionBindingComponent; 026import org.hl7.fhir.r4.model.MetadataResource; 027import org.hl7.fhir.r4.model.Parameters; 028import org.hl7.fhir.r4.model.Resource; 029import org.hl7.fhir.r4.model.ResourceType; 030import org.hl7.fhir.r4.model.StructureDefinition; 031import org.hl7.fhir.r4.model.StructureMap; 032import org.hl7.fhir.r4.model.ValueSet; 033import org.hl7.fhir.r4.model.ValueSet.ConceptSetComponent; 034import org.hl7.fhir.r4.terminologies.ValueSetExpander; 035import org.hl7.fhir.r4.utils.validation.IResourceValidator; 036import org.hl7.fhir.utilities.TranslationServices; 037import org.hl7.fhir.utilities.i18n.I18nBase; 038import org.hl7.fhir.utilities.validation.ValidationMessage.IssueSeverity; 039import org.hl7.fhir.utilities.validation.ValidationOptions; 040 041import java.util.ArrayList; 042import java.util.Arrays; 043import java.util.Collections; 044import java.util.HashSet; 045import java.util.List; 046import java.util.Set; 047 048import static org.apache.commons.lang3.StringUtils.isNotBlank; 049 050public final class HapiWorkerContext extends I18nBase implements IWorkerContext { 051 private static final List<String> PRIMITIVE_TYPES = Arrays.asList( 052 "boolean", 053 "integer", 054 "integer64", 055 "string", 056 "decimal", 057 "uri", 058 "base64Binary", 059 "instant", 060 "date", 061 "dateTime", 062 "time", 063 "code", 064 "oid", 065 "id", 066 "markdown", 067 "unsignedInt", 068 "positiveInt", 069 "uuid", 070 "xhtml", 071 "url", 072 "canonical"); 073 074 private final FhirContext myCtx; 075 private final Cache<String, Resource> myFetchedResourceCache; 076 private IValidationSupport myValidationSupport; 077 private Parameters myExpansionProfile; 078 private String myOverrideVersionNs; 079 080 public HapiWorkerContext(FhirContext theCtx, IValidationSupport theValidationSupport) { 081 Validate.notNull(theCtx, "theCtx must not be null"); 082 Validate.notNull(theValidationSupport, "theValidationSupport must not be null"); 083 myCtx = theCtx; 084 myValidationSupport = theValidationSupport; 085 086 long timeoutMillis = HapiSystemProperties.getValidationResourceCacheTimeoutMillis(); 087 088 myFetchedResourceCache = CacheFactory.build(timeoutMillis); 089 090 // Set a default locale 091 setValidationMessageLanguage(getLocale()); 092 } 093 094 @Override 095 public List<StructureDefinition> allStructures() { 096 return myValidationSupport.fetchAllStructureDefinitions(); 097 } 098 099 @Override 100 public List<StructureDefinition> getStructures() { 101 return allStructures(); 102 } 103 104 @Override 105 public CodeSystem fetchCodeSystem(String theSystem) { 106 if (myValidationSupport == null) { 107 return null; 108 } else { 109 return (CodeSystem) myValidationSupport.fetchCodeSystem(theSystem); 110 } 111 } 112 113 @Override 114 public List<ConceptMap> findMapsForSource(String theUrl) { 115 throw new UnsupportedOperationException(Msg.code(258)); 116 } 117 118 @Override 119 public String getAbbreviation(String theName) { 120 throw new UnsupportedOperationException(Msg.code(259)); 121 } 122 123 @Override 124 public org.hl7.fhir.r4.utils.INarrativeGenerator getNarrativeGenerator(String thePrefix, String theBasePath) { 125 throw new UnsupportedOperationException(Msg.code(260)); 126 } 127 128 @Override 129 public IParser getParser(ParserType theType) { 130 throw new UnsupportedOperationException(Msg.code(261)); 131 } 132 133 @Override 134 public IParser getParser(String theType) { 135 throw new UnsupportedOperationException(Msg.code(262)); 136 } 137 138 @Override 139 public List<String> getResourceNames() { 140 List<String> result = new ArrayList<>(); 141 for (ResourceType next : ResourceType.values()) { 142 result.add(next.name()); 143 } 144 Collections.sort(result); 145 return result; 146 } 147 148 @Override 149 public IParser newJsonParser() { 150 throw new UnsupportedOperationException(Msg.code(263)); 151 } 152 153 @Override 154 public IResourceValidator newValidator() { 155 throw new UnsupportedOperationException(Msg.code(264)); 156 } 157 158 @Override 159 public IParser newXmlParser() { 160 throw new UnsupportedOperationException(Msg.code(265)); 161 } 162 163 @Override 164 public String oid2Uri(String theCode) { 165 throw new UnsupportedOperationException(Msg.code(266)); 166 } 167 168 @Override 169 public boolean supportsSystem(String theSystem) { 170 if (myValidationSupport == null) { 171 return false; 172 } else { 173 return myValidationSupport.isCodeSystemSupported( 174 new ValidationSupportContext(myValidationSupport), theSystem); 175 } 176 } 177 178 @Override 179 public Set<String> typeTails() { 180 return new HashSet<>(Arrays.asList( 181 "Integer", 182 "UnsignedInt", 183 "PositiveInt", 184 "Decimal", 185 "DateTime", 186 "Date", 187 "Time", 188 "Instant", 189 "String", 190 "Uri", 191 "Oid", 192 "Uuid", 193 "Id", 194 "Boolean", 195 "Code", 196 "Markdown", 197 "Base64Binary", 198 "Coding", 199 "CodeableConcept", 200 "Attachment", 201 "Identifier", 202 "Quantity", 203 "SampledData", 204 "Range", 205 "Period", 206 "Ratio", 207 "HumanName", 208 "Address", 209 "ContactPoint", 210 "Timing", 211 "Reference", 212 "Annotation", 213 "Signature", 214 "Meta")); 215 } 216 217 @Override 218 public ValidationResult validateCode(ValidationOptions theOptions, CodeableConcept theCode, ValueSet theVs) { 219 for (Coding next : theCode.getCoding()) { 220 ValidationResult retVal = validateCode(theOptions, next, theVs); 221 if (retVal.isOk()) { 222 return retVal; 223 } 224 } 225 226 return new ValidationResult(IssueSeverity.ERROR, null); 227 } 228 229 @Override 230 public ValidationResult validateCode(ValidationOptions theOptions, Coding theCode, ValueSet theVs) { 231 String system = theCode.getSystem(); 232 String code = theCode.getCode(); 233 String display = theCode.getDisplay(); 234 return validateCode(theOptions, system, code, display, theVs); 235 } 236 237 @Override 238 public ValidationResult validateCode( 239 ValidationOptions theOptions, String theSystem, String theCode, String theDisplay) { 240 IValidationSupport.CodeValidationResult result = myValidationSupport.validateCode( 241 new ValidationSupportContext(myValidationSupport), 242 convertConceptValidationOptions(theOptions), 243 theSystem, 244 theCode, 245 theDisplay, 246 null); 247 if (result == null) { 248 return null; 249 } 250 251 IssueSeverity severity = null; 252 if (result.getSeverity() != null) { 253 severity = IssueSeverity.fromCode(result.getSeverityCode()); 254 } 255 256 ConceptDefinitionComponent definition = new ConceptDefinitionComponent().setCode(result.getCode()); 257 return new ValidationResult(severity, result.getMessage(), definition); 258 } 259 260 @Override 261 public ValidationResult validateCode( 262 ValidationOptions theOptions, 263 String theSystem, 264 String theCode, 265 String theDisplay, 266 ConceptSetComponent theVsi) { 267 throw new UnsupportedOperationException(Msg.code(267)); 268 } 269 270 @Override 271 public ValidationResult validateCode( 272 ValidationOptions theOptions, String theSystem, String theCode, String theDisplay, ValueSet theVs) { 273 274 IValidationSupport.CodeValidationResult outcome; 275 if (isNotBlank(theVs.getUrl())) { 276 outcome = myValidationSupport.validateCode( 277 new ValidationSupportContext(myValidationSupport), 278 convertConceptValidationOptions(theOptions), 279 theSystem, 280 theCode, 281 theDisplay, 282 theVs.getUrl()); 283 } else { 284 outcome = myValidationSupport.validateCodeInValueSet( 285 new ValidationSupportContext(myValidationSupport), 286 convertConceptValidationOptions(theOptions), 287 theSystem, 288 theCode, 289 theDisplay, 290 theVs); 291 } 292 293 if (outcome != null && outcome.isOk()) { 294 ConceptDefinitionComponent definition = new ConceptDefinitionComponent(); 295 definition.setCode(theCode); 296 definition.setDisplay(outcome.getDisplay()); 297 return new ValidationResult(definition); 298 } 299 300 return new ValidationResult( 301 IssueSeverity.ERROR, 302 "Unknown code[" + theCode + "] in system[" + Constants.codeSystemWithDefaultDescription(theSystem) 303 + "]"); 304 } 305 306 @Override 307 public ValidationResult validateCode(ValidationOptions theOptions, String code, ValueSet vs) { 308 ValidationOptions options = theOptions.withGuessSystem(); 309 return validateCode(options, null, code, null, vs); 310 } 311 312 @Override 313 @CoverageIgnore 314 public List<MetadataResource> allConformanceResources() { 315 throw new UnsupportedOperationException(Msg.code(268)); 316 } 317 318 @Override 319 public void generateSnapshot(StructureDefinition p) throws FHIRException { 320 throw new UnsupportedOperationException(Msg.code(269)); 321 } 322 323 @Override 324 public Parameters getExpansionParameters() { 325 return myExpansionProfile; 326 } 327 328 @Override 329 public void setExpansionProfile(Parameters theExpParameters) { 330 myExpansionProfile = theExpParameters; 331 } 332 333 @Override 334 @CoverageIgnore 335 public boolean hasCache() { 336 throw new UnsupportedOperationException(Msg.code(270)); 337 } 338 339 @Override 340 public ValueSetExpander.ValueSetExpansionOutcome expandVS( 341 ValueSet theSource, boolean theCacheOk, boolean theHierarchical) { 342 throw new UnsupportedOperationException(Msg.code(271)); 343 } 344 345 @Override 346 public ValueSetExpander.ValueSetExpansionOutcome expandVS(ConceptSetComponent theInc, boolean theHierarchical) 347 throws TerminologyServiceException { 348 ValueSet input = new ValueSet(); 349 input.getCompose().addInclude(theInc); 350 IValidationSupport.ValueSetExpansionOutcome output = 351 myValidationSupport.expandValueSet(new ValidationSupportContext(myValidationSupport), null, input); 352 return new ValueSetExpander.ValueSetExpansionOutcome((ValueSet) output.getValueSet(), output.getError(), null); 353 } 354 355 @Override 356 public ILoggingService getLogger() { 357 throw new UnsupportedOperationException(Msg.code(272)); 358 } 359 360 @Override 361 public void setLogger(ILoggingService theLogger) { 362 throw new UnsupportedOperationException(Msg.code(273)); 363 } 364 365 @Override 366 public String getVersion() { 367 return myCtx.getVersion().getVersion().getFhirVersionString(); 368 } 369 370 @Override 371 public UcumService getUcumService() { 372 throw new UnsupportedOperationException(Msg.code(274)); 373 } 374 375 @Override 376 public void setUcumService(UcumService ucumService) { 377 throw new UnsupportedOperationException(Msg.code(275)); 378 } 379 380 @Override 381 public boolean isNoTerminologyServer() { 382 return false; 383 } 384 385 @Override 386 public TranslationServices translator() { 387 throw new UnsupportedOperationException(Msg.code(276)); 388 } 389 390 @Override 391 public List<StructureMap> listTransforms() { 392 throw new UnsupportedOperationException(Msg.code(277)); 393 } 394 395 @Override 396 public StructureMap getTransform(String url) { 397 throw new UnsupportedOperationException(Msg.code(278)); 398 } 399 400 @Override 401 public String getOverrideVersionNs() { 402 return myOverrideVersionNs; 403 } 404 405 @Override 406 public void setOverrideVersionNs(String value) { 407 myOverrideVersionNs = value; 408 } 409 410 @Override 411 public StructureDefinition fetchTypeDefinition(String theTypeName) { 412 return fetchResource(StructureDefinition.class, "http://hl7.org/fhir/StructureDefinition/" + theTypeName); 413 } 414 415 @Override 416 public String getLinkForUrl(String corePath, String url) { 417 throw new UnsupportedOperationException(Msg.code(279)); 418 } 419 420 @Override 421 public List<String> getTypeNames() { 422 throw new UnsupportedOperationException(Msg.code(280)); 423 } 424 425 @Override 426 public <T extends org.hl7.fhir.r4.model.Resource> T fetchResource(Class<T> theClass, String theUri) { 427 if (myValidationSupport == null || theUri == null) { 428 return null; 429 } else { 430 @SuppressWarnings("unchecked") 431 T retVal = (T) myFetchedResourceCache.get(theUri, t -> myValidationSupport.fetchResource(theClass, theUri)); 432 return retVal; 433 } 434 } 435 436 @Override 437 public <T extends org.hl7.fhir.r4.model.Resource> T fetchResource( 438 Class<T> theClass, String theUri, String theVersion) { 439 return fetchResource(theClass, theUri + "|" + theVersion); 440 } 441 442 @Override 443 public <T extends org.hl7.fhir.r4.model.Resource> T fetchResourceWithException(Class<T> theClass, String theUri) 444 throws FHIRException { 445 T retVal = fetchResource(theClass, theUri); 446 if (retVal == null) { 447 throw new FHIRException(Msg.code(281) + "Could not find resource: " + theUri); 448 } 449 return retVal; 450 } 451 452 @Override 453 public org.hl7.fhir.r4.model.Resource fetchResourceById(String theType, String theUri) { 454 throw new UnsupportedOperationException(Msg.code(282)); 455 } 456 457 @Override 458 public <T extends org.hl7.fhir.r4.model.Resource> boolean hasResource(Class<T> theClass_, String theUri) { 459 throw new UnsupportedOperationException(Msg.code(283)); 460 } 461 462 @Override 463 public void cacheResource(org.hl7.fhir.r4.model.Resource theRes) throws FHIRException { 464 throw new UnsupportedOperationException(Msg.code(284)); 465 } 466 467 @Override 468 public Set<String> getResourceNamesAsSet() { 469 return myCtx.getResourceTypes(); 470 } 471 472 @Override 473 public ValueSetExpander.ValueSetExpansionOutcome expandVS( 474 ElementDefinitionBindingComponent theBinding, boolean theCacheOk, boolean theHierarchical) 475 throws FHIRException { 476 throw new UnsupportedOperationException(Msg.code(285)); 477 } 478 479 public static ConceptValidationOptions convertConceptValidationOptions(ValidationOptions theOptions) { 480 ConceptValidationOptions retVal = new ConceptValidationOptions(); 481 if (theOptions.isGuessSystem()) { 482 retVal = retVal.setInferSystem(true); 483 } 484 return retVal; 485 } 486 487 @Override 488 public boolean isPrimitiveType(String theType) { 489 return PRIMITIVE_TYPES.contains(theType); 490 } 491}