Merge remote-tracking branch 'origin/master' into do-20241009-documentbuilderfactory-refactor
This commit is contained in:
commit
2d80a09deb
Binary file not shown.
Before Width: | Height: | Size: 30 KiB After Width: | Height: | Size: 30 KiB |
|
@ -2,4 +2,4 @@ Locale,Coverage #,Coverage %
|
|||
de,831,40%
|
||||
es,714,34%
|
||||
ja,902,44%
|
||||
nl,1988,97%
|
||||
nl,1988,96%
|
||||
|
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -0,0 +1,37 @@
|
|||
package org.hl7.fhir.convertors.conv40_50;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStream;
|
||||
|
||||
import org.hl7.fhir.convertors.factory.VersionConvertorFactory_40_50;
|
||||
import org.hl7.fhir.utilities.TextFile;
|
||||
import org.junit.jupiter.api.DisplayName;
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
public class Questionnaire40_50Test {
|
||||
|
||||
@Test
|
||||
@DisplayName("Test r5 -> r4 Questionnaire conversion.")
|
||||
public void testR5_R4() throws IOException {
|
||||
InputStream r4_input = this.getClass().getResourceAsStream("/q_open_40.json");
|
||||
String source = TextFile.streamToString(r4_input);
|
||||
System.out.println(source);
|
||||
|
||||
org.hl7.fhir.r4.model.Questionnaire r4_actual = (org.hl7.fhir.r4.model.Questionnaire) new org.hl7.fhir.r4.formats.JsonParser().parse(source);
|
||||
org.hl7.fhir.r5.model.Resource r5_conv = VersionConvertorFactory_40_50.convertResource(r4_actual);
|
||||
|
||||
org.hl7.fhir.r5.formats.JsonParser r5_parser = new org.hl7.fhir.r5.formats.JsonParser();
|
||||
|
||||
ByteArrayOutputStream stream = new ByteArrayOutputStream();
|
||||
r5_parser.compose(stream, r5_conv);
|
||||
|
||||
org.hl7.fhir.r5.model.Resource r5_streamed = (org.hl7.fhir.r5.model.Questionnaire) new org.hl7.fhir.r5.formats.JsonParser().parse(new ByteArrayInputStream(stream.toByteArray()));
|
||||
org.hl7.fhir.r4.model.Resource r4_conv = VersionConvertorFactory_40_50.convertResource(r5_streamed);
|
||||
|
||||
assertTrue(r4_actual.equalsDeep(r4_conv), "should be the same");
|
||||
}
|
||||
}
|
|
@ -0,0 +1,53 @@
|
|||
{"resourceType": "Questionnaire",
|
||||
"id": "ed364266b937bb3bd73082b1",
|
||||
"item": [
|
||||
{
|
||||
"extension": [
|
||||
{
|
||||
"url": "http://hl7.org/fhir/StructureDefinition/questionnaire-itemControl",
|
||||
"valueCodeableConcept": {
|
||||
"coding": [
|
||||
{
|
||||
"code": "editableDropdown"
|
||||
}
|
||||
]
|
||||
}
|
||||
}
|
||||
],
|
||||
"id": "specimen-source",
|
||||
"answerOption": [
|
||||
{
|
||||
"valueCoding": {
|
||||
"code": "U",
|
||||
"display": "Urine"
|
||||
}
|
||||
},
|
||||
{
|
||||
"valueCoding": {
|
||||
"code": "B",
|
||||
"display": "Blood"
|
||||
}
|
||||
},
|
||||
{
|
||||
"valueCoding": {
|
||||
"code": "S",
|
||||
"display": "Saliva"
|
||||
}
|
||||
}
|
||||
],
|
||||
"code": [
|
||||
{
|
||||
"code": "specimen-source"
|
||||
}
|
||||
],
|
||||
"linkId": "specimen-source",
|
||||
"text": "Source of specimen",
|
||||
"type": "open-choice"
|
||||
}
|
||||
],
|
||||
"name": "Test Open Choice question",
|
||||
"status": "active",
|
||||
"subjectType": [
|
||||
"Patient"
|
||||
]
|
||||
}
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -31,11 +31,14 @@ package org.hl7.fhir.r4.utils;
|
|||
|
||||
import org.hl7.fhir.r4.model.CodeableConcept;
|
||||
import org.hl7.fhir.r4.model.IntegerType;
|
||||
import org.hl7.fhir.r4.model.Narrative.NarrativeStatus;
|
||||
import org.hl7.fhir.r4.model.OperationOutcome;
|
||||
import org.hl7.fhir.r4.model.OperationOutcome.IssueSeverity;
|
||||
import org.hl7.fhir.r4.model.OperationOutcome.IssueType;
|
||||
import org.hl7.fhir.r4.model.OperationOutcome.OperationOutcomeIssueComponent;
|
||||
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
||||
import org.hl7.fhir.utilities.xhtml.NodeType;
|
||||
import org.hl7.fhir.utilities.xhtml.XhtmlNode;
|
||||
|
||||
public class OperationOutcomeUtilities {
|
||||
|
||||
|
@ -142,4 +145,18 @@ public class OperationOutcomeUtilities {
|
|||
}
|
||||
return IssueType.NULL;
|
||||
}
|
||||
|
||||
|
||||
public static OperationOutcome outcomeFromTextError(String text) {
|
||||
OperationOutcome oo = new OperationOutcome();
|
||||
oo.getText().setStatus(NarrativeStatus.GENERATED);
|
||||
oo.getText().setDiv(new XhtmlNode(NodeType.Element, "div"));
|
||||
oo.getText().getDiv().tx(text);
|
||||
OperationOutcomeIssueComponent issue = oo.addIssue();
|
||||
issue.setSeverity(IssueSeverity.ERROR);
|
||||
issue.setCode(IssueType.EXCEPTION);
|
||||
issue.getDetails().setText(text);
|
||||
return oo;
|
||||
}
|
||||
|
||||
}
|
|
@ -1,5 +1,7 @@
|
|||
package org.hl7.fhir.r4.utils.client.network;
|
||||
|
||||
import static org.hl7.fhir.r4.utils.OperationOutcomeUtilities.outcomeFromTextError;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
@ -15,20 +17,19 @@ import org.hl7.fhir.r4.formats.XmlParser;
|
|||
import org.hl7.fhir.r4.model.Bundle;
|
||||
import org.hl7.fhir.r4.model.OperationOutcome;
|
||||
import org.hl7.fhir.r4.model.Resource;
|
||||
import org.hl7.fhir.r4.utils.OperationOutcomeUtilities;
|
||||
import org.hl7.fhir.r4.utils.ResourceUtilities;
|
||||
import org.hl7.fhir.r4.utils.client.EFhirClientException;
|
||||
import org.hl7.fhir.r4.utils.client.ResourceFormat;
|
||||
import org.hl7.fhir.utilities.MimeType;
|
||||
import org.hl7.fhir.utilities.TextFile;
|
||||
import org.hl7.fhir.utilities.settings.FhirSettings;
|
||||
import org.hl7.fhir.utilities.xhtml.XhtmlUtils;
|
||||
|
||||
import okhttp3.Authenticator;
|
||||
import okhttp3.Credentials;
|
||||
import okhttp3.Headers;
|
||||
import okhttp3.Headers.Builder;
|
||||
import okhttp3.OkHttpClient;
|
||||
import okhttp3.Request;
|
||||
import okhttp3.RequestBody;
|
||||
import okhttp3.Response;
|
||||
|
||||
public class FhirRequestBuilder {
|
||||
|
@ -242,7 +243,7 @@ public class FhirRequestBuilder {
|
|||
public <T extends Resource> ResourceRequest<T> execute() throws IOException {
|
||||
formatHeaders(httpRequest, resourceFormat, headers);
|
||||
Response response = getHttpClient().newCall(httpRequest.build()).execute();
|
||||
T resource = unmarshalReference(response, resourceFormat);
|
||||
T resource = unmarshalReference(response, resourceFormat, null);
|
||||
return new ResourceRequest<T>(resource, response.code(), getLocationHeader(response.headers()));
|
||||
}
|
||||
|
||||
|
@ -256,80 +257,94 @@ public class FhirRequestBuilder {
|
|||
* Unmarshalls a resource from the response stream.
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
protected <T extends Resource> T unmarshalReference(Response response, String format) {
|
||||
T resource = null;
|
||||
OperationOutcome error = null;
|
||||
byte[] body = null;
|
||||
protected <T extends Resource> T unmarshalReference(Response response, String format, String resourceType) {
|
||||
int code = response.code();
|
||||
boolean ok = code >= 200 && code < 300;
|
||||
if (response.body() == null) {
|
||||
if (!ok) {
|
||||
throw new EFhirClientException(response.message());
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
String body;
|
||||
|
||||
if (response.body() != null) {
|
||||
try {
|
||||
body = response.body().bytes();
|
||||
resource = (T) getParser(format).parse(body);
|
||||
if (resource instanceof OperationOutcome && hasError((OperationOutcome) resource)) {
|
||||
error = (OperationOutcome) resource;
|
||||
Resource resource = null;
|
||||
try {
|
||||
body = response.body().string();
|
||||
String ct = response.header("Content-Type");
|
||||
if (ct == null) {
|
||||
if (ok) {
|
||||
resource = getParser(format).parse(body);
|
||||
} else {
|
||||
System.out.println("Got error response with no Content-Type from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
}
|
||||
} catch (IOException ioe) {
|
||||
throw new EFhirClientException("Error reading Http Response from "+source+": " + ioe.getMessage(), ioe);
|
||||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": " + e.getMessage(), e);
|
||||
} else {
|
||||
if (ct.contains(";")) {
|
||||
ct = ct.substring(0, ct.indexOf(";"));
|
||||
}
|
||||
switch (ct) {
|
||||
case "application/json":
|
||||
case "application/fhir+json":
|
||||
if (!format.contains("json")) {
|
||||
System.out.println("Got json response expecting "+format+" from "+source+" with status "+code);
|
||||
}
|
||||
resource = getParser(ResourceFormat.RESOURCE_JSON.getHeader()).parse(body);
|
||||
break;
|
||||
case "application/xml":
|
||||
case "application/fhir+xml":
|
||||
case "text/xml":
|
||||
if (!format.contains("xml")) {
|
||||
System.out.println("Got xml response expecting "+format+" from "+source+" with status "+code);
|
||||
}
|
||||
resource = getParser(ResourceFormat.RESOURCE_XML.getHeader()).parse(response.body().bytes());
|
||||
break;
|
||||
case "text/plain":
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
break;
|
||||
case "text/html" :
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.body().string(), source));
|
||||
break;
|
||||
default: // not sure what else to do?
|
||||
System.out.println("Got content-type '"+ct+"' from "+source);
|
||||
System.out.println(body);
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
}
|
||||
}
|
||||
} catch (IOException ioe) {
|
||||
throw new EFhirClientException("Error reading Http Response from "+source+":"+ioe.getMessage(), ioe);
|
||||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": "+e.getMessage(), e);
|
||||
}
|
||||
if (resource instanceof OperationOutcome && (!"OperationOutcome".equals(resourceType) || !ok)) {
|
||||
OperationOutcome error = (OperationOutcome) resource;
|
||||
if (hasError((OperationOutcome) resource)) {
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
} else {
|
||||
// umm, weird...
|
||||
System.out.println("Got OperationOutcome with no error from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
if (error != null) {
|
||||
String s = ResourceUtilities.getErrorDescription(error);
|
||||
String reqid = response.header("x-request-id");
|
||||
if (reqid == null) {
|
||||
reqid = response.header("X-Request-Id");
|
||||
}
|
||||
if (reqid != null) {
|
||||
s = s + " [x-request-id: "+reqid+"]";
|
||||
}
|
||||
System.out.println("Error from "+source+": " + s);
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
if (resource == null) {
|
||||
System.out.println("No resource from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
return null; // shouldn't get here?
|
||||
}
|
||||
|
||||
return resource;
|
||||
if (resourceType != null && !resource.fhirType().equals(resourceType)) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": Found an "+resource.fhirType()+" looking for a "+resourceType);
|
||||
}
|
||||
return (T) resource;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Unmarshalls Bundle from response stream.
|
||||
*/
|
||||
protected Bundle unmarshalFeed(Response response, String format) {
|
||||
Bundle feed = null;
|
||||
OperationOutcome error = null;
|
||||
try {
|
||||
byte[] body = response.body().bytes();
|
||||
String contentType = response.header("Content-Type");
|
||||
if (body != null) {
|
||||
if (contentType.contains(ResourceFormat.RESOURCE_XML.getHeader())
|
||||
|| contentType.contains(ResourceFormat.RESOURCE_JSON.getHeader())
|
||||
|| contentType.contains("text/xml+fhir")) {
|
||||
Resource rf = getParser(format).parse(body);
|
||||
if (rf instanceof Bundle)
|
||||
feed = (Bundle) rf;
|
||||
else if (rf instanceof OperationOutcome && hasError((OperationOutcome) rf)) {
|
||||
error = (OperationOutcome) rf;
|
||||
} else {
|
||||
throw new EFhirClientException("Error reading server response from "+source+": a resource was returned instead");
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!response.isSuccessful() && feed == null && error == null) {
|
||||
String text = TextFile.bytesToString(body);
|
||||
throw new EFhirClientException("Error from "+source+": " + text);
|
||||
}
|
||||
} catch (EFhirClientException e) {
|
||||
throw e;
|
||||
} catch (IOException ioe) {
|
||||
throw new EFhirClientException("Error reading Http Response from "+source+":"+ioe.getMessage(), ioe);
|
||||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+":"+e.getMessage(), e);
|
||||
}
|
||||
if (error != null) {
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
}
|
||||
return feed;
|
||||
return unmarshalReference(response, format, "Bundle");
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -33,11 +33,14 @@ import java.util.List;
|
|||
|
||||
import org.hl7.fhir.r4b.model.CodeableConcept;
|
||||
import org.hl7.fhir.r4b.model.IntegerType;
|
||||
import org.hl7.fhir.r4b.model.Narrative.NarrativeStatus;
|
||||
import org.hl7.fhir.r4b.model.OperationOutcome;
|
||||
import org.hl7.fhir.r4b.model.OperationOutcome.IssueSeverity;
|
||||
import org.hl7.fhir.r4b.model.OperationOutcome.IssueType;
|
||||
import org.hl7.fhir.r4b.model.OperationOutcome.OperationOutcomeIssueComponent;
|
||||
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
||||
import org.hl7.fhir.utilities.xhtml.NodeType;
|
||||
import org.hl7.fhir.utilities.xhtml.XhtmlNode;
|
||||
|
||||
public class OperationOutcomeUtilities {
|
||||
|
||||
|
@ -154,4 +157,16 @@ public class OperationOutcomeUtilities {
|
|||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
public static OperationOutcome outcomeFromTextError(String text) {
|
||||
OperationOutcome oo = new OperationOutcome();
|
||||
oo.getText().setStatus(NarrativeStatus.GENERATED);
|
||||
oo.getText().setDiv(new XhtmlNode(NodeType.Element, "div"));
|
||||
oo.getText().getDiv().tx(text);
|
||||
OperationOutcomeIssueComponent issue = oo.addIssue();
|
||||
issue.setSeverity(IssueSeverity.ERROR);
|
||||
issue.setCode(IssueType.EXCEPTION);
|
||||
issue.getDetails().setText(text);
|
||||
return oo;
|
||||
}
|
||||
}
|
|
@ -1,6 +1,12 @@
|
|||
package org.hl7.fhir.r4b.utils.client.network;
|
||||
|
||||
import okhttp3.*;
|
||||
import java.io.IOException;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.hl7.fhir.r4b.formats.IParser;
|
||||
import org.hl7.fhir.r4b.formats.JsonParser;
|
||||
|
@ -8,18 +14,18 @@ import org.hl7.fhir.r4b.formats.XmlParser;
|
|||
import org.hl7.fhir.r4b.model.Bundle;
|
||||
import org.hl7.fhir.r4b.model.OperationOutcome;
|
||||
import org.hl7.fhir.r4b.model.Resource;
|
||||
import org.hl7.fhir.r4b.utils.OperationOutcomeUtilities;
|
||||
import org.hl7.fhir.r4b.utils.ResourceUtilities;
|
||||
import org.hl7.fhir.r4b.utils.client.EFhirClientException;
|
||||
import org.hl7.fhir.r4b.utils.client.ResourceFormat;
|
||||
import org.hl7.fhir.utilities.ToolingClientLogger;
|
||||
import org.hl7.fhir.utilities.xhtml.XhtmlUtils;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collections;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import okhttp3.Authenticator;
|
||||
import okhttp3.Credentials;
|
||||
import okhttp3.Headers;
|
||||
import okhttp3.OkHttpClient;
|
||||
import okhttp3.Request;
|
||||
import okhttp3.Response;
|
||||
|
||||
public class FhirRequestBuilder {
|
||||
|
||||
|
@ -227,7 +233,7 @@ public class FhirRequestBuilder {
|
|||
public <T extends Resource> ResourceRequest<T> execute() throws IOException {
|
||||
formatHeaders(httpRequest, resourceFormat, headers);
|
||||
Response response = getHttpClient().newCall(httpRequest.build()).execute();
|
||||
T resource = unmarshalReference(response, resourceFormat);
|
||||
T resource = unmarshalReference(response, resourceFormat, null);
|
||||
return new ResourceRequest<T>(resource, response.code(), getLocationHeader(response.headers()));
|
||||
}
|
||||
|
||||
|
@ -241,63 +247,94 @@ public class FhirRequestBuilder {
|
|||
* Unmarshalls a resource from the response stream.
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
protected <T extends Resource> T unmarshalReference(Response response, String format) {
|
||||
T resource = null;
|
||||
OperationOutcome error = null;
|
||||
|
||||
if (response.body() != null) {
|
||||
try {
|
||||
byte[] body = response.body().bytes();
|
||||
resource = (T) getParser(format).parse(body);
|
||||
if (resource instanceof OperationOutcome && hasError((OperationOutcome) resource)) {
|
||||
error = (OperationOutcome) resource;
|
||||
}
|
||||
} catch (IOException ioe) {
|
||||
throw new EFhirClientException("Error reading Http Response from "+source+": " + ioe.getMessage(), ioe);
|
||||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": " + e.getMessage(), e);
|
||||
protected <T extends Resource> T unmarshalReference(Response response, String format, String resourceType) {
|
||||
int code = response.code();
|
||||
boolean ok = code >= 200 && code < 300;
|
||||
if (response.body() == null) {
|
||||
if (!ok) {
|
||||
throw new EFhirClientException(response.message());
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
if (error != null) {
|
||||
throw new EFhirClientException("Error from server: " + ResourceUtilities.getErrorDescription(error), error);
|
||||
String body;
|
||||
|
||||
Resource resource = null;
|
||||
try {
|
||||
body = response.body().string();
|
||||
String ct = response.header("Content-Type");
|
||||
if (ct == null) {
|
||||
if (ok) {
|
||||
resource = getParser(format).parse(body);
|
||||
} else {
|
||||
System.out.println("Got error response with no Content-Type from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
}
|
||||
} else {
|
||||
if (ct.contains(";")) {
|
||||
ct = ct.substring(0, ct.indexOf(";"));
|
||||
}
|
||||
switch (ct) {
|
||||
case "application/json":
|
||||
case "application/fhir+json":
|
||||
if (!format.contains("json")) {
|
||||
System.out.println("Got json response expecting "+format+" from "+source+" with status "+code);
|
||||
}
|
||||
resource = getParser(ResourceFormat.RESOURCE_JSON.getHeader()).parse(body);
|
||||
break;
|
||||
case "application/xml":
|
||||
case "application/fhir+xml":
|
||||
case "text/xml":
|
||||
if (!format.contains("xml")) {
|
||||
System.out.println("Got xml response expecting "+format+" from "+source+" with status "+code);
|
||||
}
|
||||
resource = getParser(ResourceFormat.RESOURCE_XML.getHeader()).parse(body);
|
||||
break;
|
||||
case "text/plain":
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
break;
|
||||
case "text/html" :
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.body().string(), source));
|
||||
break;
|
||||
default: // not sure what else to do?
|
||||
System.out.println("Got content-type '"+ct+"' from "+source);
|
||||
System.out.println(body);
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
}
|
||||
}
|
||||
} catch (IOException ioe) {
|
||||
throw new EFhirClientException("Error reading Http Response from "+source+":"+ioe.getMessage(), ioe);
|
||||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": "+e.getMessage(), e);
|
||||
}
|
||||
|
||||
return resource;
|
||||
if (resource instanceof OperationOutcome && (!"OperationOutcome".equals(resourceType) || !ok)) {
|
||||
OperationOutcome error = (OperationOutcome) resource;
|
||||
if (hasError((OperationOutcome) resource)) {
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
} else {
|
||||
// umm, weird...
|
||||
System.out.println("Got OperationOutcome with no error from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
if (resource == null) {
|
||||
System.out.println("No resource from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
return null; // shouldn't get here?
|
||||
}
|
||||
if (resourceType != null && !resource.fhirType().equals(resourceType)) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": Found an "+resource.fhirType()+" looking for a "+resourceType);
|
||||
}
|
||||
return (T) resource;
|
||||
}
|
||||
|
||||
/**
|
||||
* Unmarshalls Bundle from response stream.
|
||||
*/
|
||||
protected Bundle unmarshalFeed(Response response, String format) {
|
||||
Bundle feed = null;
|
||||
OperationOutcome error = null;
|
||||
try {
|
||||
byte[] body = response.body().bytes();
|
||||
String contentType = response.header("Content-Type");
|
||||
if (body != null) {
|
||||
if (contentType.contains(ResourceFormat.RESOURCE_XML.getHeader())
|
||||
|| contentType.contains(ResourceFormat.RESOURCE_JSON.getHeader())
|
||||
|| contentType.contains("text/xml+fhir")) {
|
||||
Resource rf = getParser(format).parse(body);
|
||||
if (rf instanceof Bundle)
|
||||
feed = (Bundle) rf;
|
||||
else if (rf instanceof OperationOutcome && hasError((OperationOutcome) rf)) {
|
||||
error = (OperationOutcome) rf;
|
||||
} else {
|
||||
throw new EFhirClientException("Error reading server response: a resource was returned instead");
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (IOException ioe) {
|
||||
throw new EFhirClientException("Error reading Http Response from "+source+": "+ioe.getMessage(), ioe);
|
||||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": "+e.getMessage(), e);
|
||||
}
|
||||
if (error != null) {
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
}
|
||||
return feed;
|
||||
return unmarshalReference(response, format, "Bundle");
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -294,17 +294,28 @@ public class ProfileUtilities {
|
|||
public static class SourcedChildDefinitions {
|
||||
private StructureDefinition source;
|
||||
private List<ElementDefinition> list;
|
||||
private String path;
|
||||
public SourcedChildDefinitions(StructureDefinition source, List<ElementDefinition> list) {
|
||||
super();
|
||||
this.source = source;
|
||||
this.list = list;
|
||||
}
|
||||
public SourcedChildDefinitions(StructureDefinition source, List<ElementDefinition> list, String path) {
|
||||
super();
|
||||
this.source = source;
|
||||
this.list = list;
|
||||
this.path = path;
|
||||
}
|
||||
public StructureDefinition getSource() {
|
||||
return source;
|
||||
}
|
||||
public List<ElementDefinition> getList() {
|
||||
return list;
|
||||
}
|
||||
public String getPath() {
|
||||
return path;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public class ElementDefinitionResolution {
|
||||
|
|
|
@ -1764,6 +1764,10 @@ public abstract class BaseWorkerContext extends I18nBase implements IWorkerConte
|
|||
} else if (options.getVsAsUrl()){
|
||||
pin.addParameter().setName("url").setValue(new UriType(vs.getUrl()));
|
||||
} else {
|
||||
if (vs.hasCompose() && vs.hasExpansion()) {
|
||||
vs = vs.copy();
|
||||
vs.setExpansion(null);
|
||||
}
|
||||
pin.addParameter().setName("valueSet").setResource(vs);
|
||||
if (vs.getUrl() != null) {
|
||||
terminologyClientContext.getCached().add(vs.getUrl()+"|"+ vs.getVersion());
|
||||
|
|
|
@ -41,6 +41,7 @@ import org.hl7.fhir.r5.model.StructureDefinition;
|
|||
import org.hl7.fhir.r5.model.ValueSet;
|
||||
import org.hl7.fhir.r5.model.ValueSet.ConceptReferenceComponent;
|
||||
import org.hl7.fhir.r5.model.ValueSet.ConceptReferenceDesignationComponent;
|
||||
import org.hl7.fhir.r5.renderers.Renderer.RenderingStatus;
|
||||
import org.hl7.fhir.r5.renderers.utils.RenderingContext;
|
||||
import org.hl7.fhir.r5.renderers.utils.RenderingContext.GenerationRules;
|
||||
import org.hl7.fhir.r5.renderers.utils.RenderingContext.ResourceRendererMode;
|
||||
|
@ -486,24 +487,45 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
// }
|
||||
// }
|
||||
//
|
||||
public void renderExtensionsInText(RenderingStatus status, XhtmlNode div, ResourceWrapper element, String sep) throws FHIRFormatError, DefinitionException, IOException {
|
||||
public void renderExtensionsInText(RenderingStatus status, XhtmlNode x, ResourceWrapper element, String sep) throws FHIRFormatError, DefinitionException, IOException {
|
||||
boolean first = true;
|
||||
for (ResourceWrapper ext : element.extensions()) {
|
||||
if (canRender(ext)) {
|
||||
status.setExtensions(true);
|
||||
if (first) {
|
||||
first = false;
|
||||
} else {
|
||||
div.tx(sep);
|
||||
div.tx(" ");
|
||||
x.tx(sep);
|
||||
x.tx(" ");
|
||||
}
|
||||
|
||||
String lbl = getExtensionLabel(ext);
|
||||
div.tx(lbl);
|
||||
div.tx(": ");
|
||||
renderDataType(status, div, ext.child("value"));
|
||||
x.tx(lbl);
|
||||
x.tx(": ");
|
||||
renderDataType(status, x, ext.child("value"));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
protected void checkRenderExtensions(RenderingStatus status, XhtmlNode x, ResourceWrapper element) throws FHIRFormatError, DefinitionException, IOException {
|
||||
if (element.has("extension")) {
|
||||
boolean someCanRender = false;
|
||||
for (ResourceWrapper ext : element.children("extension")) {
|
||||
ResourceWrapper value = ext.child("value");
|
||||
if (canRender(ext) && value.isPrimitive()) {
|
||||
someCanRender = true;
|
||||
}
|
||||
}
|
||||
if (someCanRender) {
|
||||
status.setExtensions(true);
|
||||
x.tx(" (");
|
||||
renderExtensionsInText(status, x, element, ", ");
|
||||
x.tx(")");
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// public void renderExtensionsInList(XhtmlNode div, BackboneType element, String sep) throws FHIRFormatError, DefinitionException, IOException {
|
||||
// boolean first = true;
|
||||
|
@ -742,13 +764,14 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
}
|
||||
|
||||
public boolean canRenderDataType(String type) {
|
||||
return context.getContextUtilities().isPrimitiveType(type) || Utilities.existsInList(type, "Annotation", "Coding", "CodeableConcept", "Identifier", "HumanName", "Address",
|
||||
return context.getContextUtilities().isPrimitiveType(type) || Utilities.existsInList(type, "Annotation", "Coding", "CodeableConcept", "Identifier", "HumanName", "Address", "Dosage",
|
||||
"Expression", "Money", "ContactPoint", "Quantity", "Range", "Period", "Timing", "SampledData", "Reference", "UsageContext", "ContactDetail", "Ratio", "Attachment", "CodeableReference");
|
||||
}
|
||||
|
||||
public boolean renderDataType(RenderingStatus status, XhtmlNode x, ResourceWrapper type) throws FHIRFormatError, DefinitionException, IOException {
|
||||
return renderDataType(status, null, x, type);
|
||||
}
|
||||
|
||||
public boolean renderDataType(RenderingStatus status, XhtmlNode parent, XhtmlNode x, ResourceWrapper type) throws FHIRFormatError, DefinitionException, IOException {
|
||||
if (type == null) {
|
||||
return false;
|
||||
|
@ -863,10 +886,11 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
renderUri(status, x, type);
|
||||
}
|
||||
|
||||
private void renderRatio(RenderingStatus status, XhtmlNode x, ResourceWrapper type) {
|
||||
private void renderRatio(RenderingStatus status, XhtmlNode x, ResourceWrapper type) throws FHIRFormatError, DefinitionException, IOException {
|
||||
renderQuantity(status, x, type.child("numerator"));
|
||||
x.tx("/");
|
||||
renderQuantity(status, x, type.child("denominator"));
|
||||
renderQuantity(status, x, type.child("denominator"));
|
||||
checkRenderExtensions(status, x, type);
|
||||
}
|
||||
|
||||
private void renderAttachment(RenderingStatus status, XhtmlNode x, ResourceWrapper att) {
|
||||
|
@ -892,6 +916,7 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
private void renderDateTime(RenderingStatus status, XhtmlNode x, ResourceWrapper type) throws FHIRFormatError, DefinitionException, IOException {
|
||||
if (!renderPrimitiveWithNoValue(status, x, type)) {
|
||||
x.tx(displayDateTime(type));
|
||||
checkRenderExtensions(status, x, type);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1002,7 +1027,11 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
return true;
|
||||
}
|
||||
|
||||
private String tail(String url) {
|
||||
protected String tail(String url) {
|
||||
return url.substring(url.lastIndexOf(".")+1);
|
||||
}
|
||||
|
||||
protected String utail(String url) {
|
||||
return url.contains("/") ? url.substring(url.lastIndexOf("/")+1) : url;
|
||||
}
|
||||
|
||||
|
@ -1038,7 +1067,8 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
checkRenderExtensions(status, x, uri);
|
||||
}
|
||||
|
||||
protected void renderAnnotation(RenderingStatus status, XhtmlNode x, ResourceWrapper a) throws FHIRException {
|
||||
|
@ -1270,7 +1300,7 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
return new CodeResolution(null, null, null, code.getText(), code.getText());
|
||||
}
|
||||
}
|
||||
protected void renderCodingWithDetails(RenderingStatus status, XhtmlNode x, ResourceWrapper c) {
|
||||
protected void renderCodingWithDetails(RenderingStatus status, XhtmlNode x, ResourceWrapper c) throws FHIRFormatError, DefinitionException, IOException {
|
||||
String s = "";
|
||||
if (c.has("display"))
|
||||
s = context.getTranslated(c.child("display"));
|
||||
|
@ -1296,6 +1326,7 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
if (c.has("version")) {
|
||||
x.tx(" "+context.formatPhrase(RenderingContext.DATA_REND_VERSION, c.primitiveValue("version"), ")"));
|
||||
}
|
||||
checkRenderExtensions(status, x, c);
|
||||
}
|
||||
|
||||
protected void renderCoding(RenderingStatus status, XhtmlNode x, ResourceWrapper c) {
|
||||
|
@ -1434,7 +1465,8 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
}
|
||||
|
||||
x.span(null, context.formatPhrase(RenderingContext.DATA_REND_CODES) +b.toString()).addText(s);
|
||||
}
|
||||
}
|
||||
checkRenderExtensions(status, x, cc);
|
||||
}
|
||||
|
||||
protected String displayIdentifier(ResourceWrapper ii) {
|
||||
|
@ -1477,7 +1509,7 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
return s;
|
||||
}
|
||||
|
||||
protected void renderIdentifier(RenderingStatus status, XhtmlNode x, ResourceWrapper ii) {
|
||||
protected void renderIdentifier(RenderingStatus status, XhtmlNode x, ResourceWrapper ii) throws FHIRFormatError, DefinitionException, IOException {
|
||||
if (ii.has("type")) {
|
||||
ResourceWrapper type = ii.child("type");
|
||||
if (type.has("text")) {
|
||||
|
@ -1527,7 +1559,8 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
x.tx(displayPeriod(ii.child("period")));
|
||||
}
|
||||
x.tx(")");
|
||||
}
|
||||
}
|
||||
checkRenderExtensions(status, x, ii);
|
||||
}
|
||||
|
||||
public static String displayHumanName(ResourceWrapper name) {
|
||||
|
@ -1550,7 +1583,7 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
}
|
||||
|
||||
|
||||
protected void renderHumanName(RenderingStatus status, XhtmlNode x, ResourceWrapper name) {
|
||||
protected void renderHumanName(RenderingStatus status, XhtmlNode x, ResourceWrapper name) throws FHIRFormatError, DefinitionException, IOException {
|
||||
StringBuilder s = new StringBuilder();
|
||||
if (name.has("text"))
|
||||
s.append(context.getTranslated(name.child("text")));
|
||||
|
@ -1567,7 +1600,8 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
if (name.has("use") && !name.primitiveValue("use").equals("usual")) {
|
||||
s.append("("+context.getTranslatedCode(name.primitiveValue("use"), "http://hl7.org/fhir/name-use")+")");
|
||||
}
|
||||
x.addText(s.toString());
|
||||
x.addText(s.toString());
|
||||
checkRenderExtensions(status, x, name);
|
||||
}
|
||||
|
||||
private String displayAddress(ResourceWrapper address) {
|
||||
|
@ -1604,8 +1638,9 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
return s.toString();
|
||||
}
|
||||
|
||||
protected void renderAddress(RenderingStatus status, XhtmlNode x, ResourceWrapper address) {
|
||||
x.addText(displayAddress(address));
|
||||
protected void renderAddress(RenderingStatus status, XhtmlNode x, ResourceWrapper address) throws FHIRFormatError, DefinitionException, IOException {
|
||||
x.addText(displayAddress(address));
|
||||
checkRenderExtensions(status, x, address);
|
||||
}
|
||||
|
||||
|
||||
|
@ -1788,7 +1823,7 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
return s.toString();
|
||||
}
|
||||
|
||||
protected void renderQuantity(RenderingStatus status, XhtmlNode x, ResourceWrapper q) {
|
||||
protected void renderQuantity(RenderingStatus status, XhtmlNode x, ResourceWrapper q) throws FHIRFormatError, DefinitionException, IOException {
|
||||
if (q.has("comparator"))
|
||||
x.addText(q.primitiveValue("comparator"));
|
||||
if (q.has("value")) {
|
||||
|
@ -1805,7 +1840,8 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
}
|
||||
if (context.isTechnicalMode() && q.has("code")) {
|
||||
x.span("background: LightGoldenRodYellow", null).tx(" "+ (context.formatPhrase(RenderingContext.DATA_REND_DETAILS, displaySystem(q.primitiveValue("system")))) +q.primitiveValue("code")+" = '"+lookupCode(q.primitiveValue("system"), null, q.primitiveValue("code"))+"')");
|
||||
}
|
||||
}
|
||||
checkRenderExtensions(status, x, q);
|
||||
}
|
||||
|
||||
|
||||
|
@ -2073,7 +2109,7 @@ public class DataRenderer extends Renderer implements CodeResolver {
|
|||
st = st + "-"+rep.primitiveValue("frequencyMax");
|
||||
}
|
||||
if (rep.has("period")) {
|
||||
st = st + " "+ (context.formatPhrase(RenderingContext.DATA_REND_PER))+rep.primitiveValue("period");
|
||||
st = st + " "+ (context.formatPhrase(RenderingContext.DATA_REND_PER))+" "+rep.primitiveValue("period");
|
||||
if (rep.has("periodMax"))
|
||||
st = st + "-"+rep.primitiveValue("periodMax");
|
||||
st = st + " "+displayTimeUnits(rep.primitiveValue("periodUnit"));
|
||||
|
|
|
@ -3,6 +3,7 @@ package org.hl7.fhir.r5.renderers;
|
|||
import java.io.IOException;
|
||||
import java.io.UnsupportedEncodingException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.HashMap;
|
||||
import java.util.HashSet;
|
||||
import java.util.List;
|
||||
|
@ -19,6 +20,7 @@ import org.hl7.fhir.r5.model.ElementDefinition;
|
|||
import org.hl7.fhir.r5.model.Resource;
|
||||
import org.hl7.fhir.r5.model.StructureDefinition;
|
||||
import org.hl7.fhir.r5.model.StructureDefinition.StructureDefinitionKind;
|
||||
import org.hl7.fhir.r5.renderers.StructureDefinitionRenderer.SourcedElementDefinition;
|
||||
import org.hl7.fhir.r5.renderers.utils.RenderingContext;
|
||||
import org.hl7.fhir.r5.renderers.utils.ResourceWrapper;
|
||||
import org.hl7.fhir.r5.renderers.utils.ResourceWrapper.NamedResourceWrapperList;
|
||||
|
@ -49,7 +51,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
} else {
|
||||
ElementDefinition ed = sd.getSnapshot().getElement().get(0);
|
||||
containedIds.clear();
|
||||
generateByProfile(status, r, sd, r, sd.getSnapshot().getElement(), ed, context.getProfileUtilities().getChildList(sd, ed), x, r.fhirType(), context.isTechnicalMode(), 0);
|
||||
generateByProfile(status, r, sd, r, ed, context.getProfileUtilities().getChildList(sd, ed), x, r.fhirType(), context.isTechnicalMode(), 0);
|
||||
}
|
||||
} catch (Exception e) {
|
||||
if (DEBUG) {
|
||||
|
@ -159,7 +161,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
return null;
|
||||
}
|
||||
|
||||
private void renderLeaf(RenderingStatus status, ResourceWrapper res, ResourceWrapper ew, StructureDefinition sd, ElementDefinition defn, XhtmlNode parent, XhtmlNode x, boolean title, boolean showCodeDetails, Map<String, String> displayHints, String path, int indent) throws FHIRException, UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
private void renderLeaf(RenderingStatus status, ResourceWrapper res, ResourceWrapper ew, StructureDefinition sd, ElementDefinition defn, XhtmlNode parent, XhtmlNode x, boolean title, boolean showCodeDetails, Map<String, String> displayHints, int indent) throws FHIRException, UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
if (ew == null)
|
||||
return;
|
||||
|
||||
|
@ -173,7 +175,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
} else if (!renderDataType(status, parent, x, ew)) {
|
||||
// well, we have a cell (x) to render this thing, whatever it is
|
||||
// it's not a data type for which we have a built rendering, so we're going to get a list of it's renderable datatype properties, and render them in a list
|
||||
SourcedChildDefinitions childDefs = context.getProfileUtilities().getChildMap(sd, defn);
|
||||
// SourcedChildDefinitions childDefs = context.getProfileUtilities().getChildMap(sd, defn);
|
||||
boolean first = true;
|
||||
x.tx(" (");
|
||||
for (ResourceWrapper child : ew.children()) {
|
||||
|
@ -199,24 +201,24 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
}
|
||||
}
|
||||
|
||||
private boolean displayLeaf(ResourceWrapper res, ResourceWrapper ew, ElementDefinition defn, XhtmlNode x, String name, boolean showCodeDetails, boolean allowLinks) throws FHIRException, UnsupportedEncodingException, IOException {
|
||||
if (ew == null)
|
||||
return false;
|
||||
|
||||
Map<String, String> displayHints = readDisplayHints(defn);
|
||||
|
||||
if (name.endsWith("[x]"))
|
||||
name = name.substring(0, name.length() - 3);
|
||||
|
||||
if (!showCodeDetails && ew.isPrimitive() && isDefault(displayHints, ew)) {
|
||||
return false;
|
||||
} else if (Utilities.existsInList(ew.fhirType(), "Extension")) {
|
||||
return false;
|
||||
} else {
|
||||
x.addText(name+": "+ displayDataType(ew));
|
||||
return true;
|
||||
}
|
||||
}
|
||||
// private boolean displayLeaf(ResourceWrapper res, ResourceWrapper ew, ElementDefinition defn, XhtmlNode x, String name, boolean showCodeDetails, boolean allowLinks) throws FHIRException, UnsupportedEncodingException, IOException {
|
||||
// if (ew == null)
|
||||
// return false;
|
||||
//
|
||||
// Map<String, String> displayHints = readDisplayHints(defn);
|
||||
//
|
||||
// if (name.endsWith("[x]"))
|
||||
// name = name.substring(0, name.length() - 3);
|
||||
//
|
||||
// if (!showCodeDetails && ew.isPrimitive() && isDefault(displayHints, ew)) {
|
||||
// return false;
|
||||
// } else if (Utilities.existsInList(ew.fhirType(), "Extension")) {
|
||||
// return false;
|
||||
// } else {
|
||||
// x.addText(name+": "+ displayDataType(ew));
|
||||
// return true;
|
||||
// }
|
||||
// }
|
||||
|
||||
|
||||
|
||||
|
@ -250,7 +252,8 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
return code != null && (code.equals("Element") || code.equals("BackboneElement"));
|
||||
}
|
||||
|
||||
private List<ElementDefinition> getChildrenForPath(StructureDefinition profile, List<ElementDefinition> elements, String path) throws DefinitionException {
|
||||
private SourcedChildDefinitions getChildrenForPath(StructureDefinition profile, String path) throws DefinitionException {
|
||||
var elements = profile.getSnapshot().getElement();
|
||||
// do we need to do a name reference substitution?
|
||||
for (ElementDefinition e : elements) {
|
||||
if (e.getPath().equals(path) && e.hasContentReference()) {
|
||||
|
@ -279,45 +282,45 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
}
|
||||
if (results.isEmpty() && t != null && t.getType().size() == 1) {
|
||||
StructureDefinition tsd = context.getWorker().fetchTypeDefinition(t.getTypeFirstRep().getWorkingCode());
|
||||
return getChildrenForPath(tsd, tsd.getSnapshot().getElement(), tsd.getType());
|
||||
return getChildrenForPath(tsd, tsd.getType());
|
||||
}
|
||||
return results;
|
||||
return new SourcedChildDefinitions(profile, results, path);
|
||||
}
|
||||
|
||||
private void generateByProfile(RenderingStatus status, ResourceWrapper res, StructureDefinition profile, ResourceWrapper e, List<ElementDefinition> allElements, ElementDefinition defn, List<ElementDefinition> children, XhtmlNode x, String path, boolean showCodeDetails, int indent) throws FHIRException, UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
private void generateByProfile(RenderingStatus status, ResourceWrapper res, StructureDefinition profile, ResourceWrapper e, ElementDefinition defn, List<ElementDefinition> children, XhtmlNode x, String path, boolean showCodeDetails, int indent) throws FHIRException, UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
if (children.isEmpty()) {
|
||||
StructureDefinition sdt = context.getWorker().fetchTypeDefinition(e.fhirType());
|
||||
if (sdt != null && (sdt.getKind() == StructureDefinitionKind.COMPLEXTYPE || sdt.getKind() == StructureDefinitionKind.PRIMITIVETYPE)) {
|
||||
renderLeaf(status, res, e, profile, defn, x, x, false, showCodeDetails, readDisplayHints(defn), path, indent);
|
||||
renderLeaf(status, res, e, profile, defn, x, x, false, showCodeDetails, readDisplayHints(defn), indent);
|
||||
} else {
|
||||
// we don't have anything to render?
|
||||
}
|
||||
} else {
|
||||
List<NamedResourceWrapperList> pl = splitExtensions(profile, e.childrenInGroups());
|
||||
for (NamedResourceWrapperList p : pl) {
|
||||
generateForProperty(status, res, profile, allElements, children, x, path, showCodeDetails, indent, false, p);
|
||||
generateForProperty(status, res, profile, children, x, path, showCodeDetails, indent, false, p);
|
||||
}
|
||||
for (NamedResourceWrapperList p : pl) {
|
||||
generateForProperty(status, res, profile, allElements, children, x, path, showCodeDetails, indent, true, p);
|
||||
generateForProperty(status, res, profile, children, x, path, showCodeDetails, indent, true, p);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private void generateForProperty(RenderingStatus status, ResourceWrapper res, StructureDefinition profile,
|
||||
List<ElementDefinition> allElements, List<ElementDefinition> children, XhtmlNode x, String path,
|
||||
List<ElementDefinition> children, XhtmlNode x, String path,
|
||||
boolean showCodeDetails, int indent, boolean round2, NamedResourceWrapperList p)
|
||||
throws UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
if (!p.getValues().isEmpty()) {
|
||||
ElementDefinition child = getElementDefinition(children, path+"."+p.getName());
|
||||
if (child != null) {
|
||||
if (!child.getBase().hasPath() || !child.getBase().getPath().startsWith("Resource.")) {
|
||||
generateElementByProfile(status, res, profile, allElements, x, path, showCodeDetails, indent, p, child, round2);
|
||||
generateElementByProfile(status, res, profile, x, path, showCodeDetails, indent, p, child, round2);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void generateElementByProfile(RenderingStatus status, ResourceWrapper res, StructureDefinition profile, List<ElementDefinition> allElements, XhtmlNode x, String path,
|
||||
public void generateElementByProfile(RenderingStatus status, ResourceWrapper res, StructureDefinition profile, XhtmlNode x, String path,
|
||||
boolean showCodeDetails, int indent, NamedResourceWrapperList p, ElementDefinition child, boolean round2) throws UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
Map<String, String> displayHints = readDisplayHints(child);
|
||||
if ("DomainResource.contained".equals(child.getBase().getPath())) {
|
||||
|
@ -336,8 +339,8 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
if (isExt) {
|
||||
status.setExtensions(true);
|
||||
}
|
||||
List<ElementDefinition> grandChildren = getChildrenForPath(profile, allElements, path+"."+p.getName());
|
||||
filterGrandChildren(grandChildren, path+"."+p.getName(), p);
|
||||
SourcedChildDefinitions grandChildren = getChildrenForPath(profile, path+"."+p.getName());
|
||||
filterGrandChildren(grandChildren.getList(), path+"."+p.getName(), p);
|
||||
if (p.getValues().size() > 0) {
|
||||
if (isSimple(child) && !isExt) {
|
||||
XhtmlNode para = x.isPara() ? para = x : x.para();
|
||||
|
@ -351,7 +354,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
if (renderAsList(child) && p.getValues().size() > 1) {
|
||||
XhtmlNode list = x.ul();
|
||||
for (ResourceWrapper v : p.getValues())
|
||||
renderLeaf(status, res, v, profile, child, x, list.li(), false, showCodeDetails, displayHints, path, indent);
|
||||
renderLeaf(status, res, v, profile, child, x, list.li(), false, showCodeDetails, displayHints, indent);
|
||||
} else {
|
||||
boolean first = true;
|
||||
for (ResourceWrapper v : p.getValues()) {
|
||||
|
@ -360,23 +363,23 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
} else {
|
||||
para.tx(", ");
|
||||
}
|
||||
renderLeaf(status, res, v, profile, child, x, para, false, showCodeDetails, displayHints, path, indent);
|
||||
renderLeaf(status, res, v, profile, child, x, para, false, showCodeDetails, displayHints, indent);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (canDoTable(path, p, grandChildren, x)) {
|
||||
} else if (canDoTable(path, p, grandChildren.getList(), x)) {
|
||||
XhtmlNode xn = new XhtmlNode(NodeType.Element, getHeader());
|
||||
xn.addText(Utilities.capitalize(Utilities.camelCase(Utilities.pluralizeMe(p.getName()))));
|
||||
XhtmlNode tbl = new XhtmlNode(NodeType.Element, "table");
|
||||
tbl.setAttribute("class", "grid");
|
||||
XhtmlNode tr = tbl.tr();
|
||||
tr.td().style("display: none").tx("-"); // work around problem with empty table rows
|
||||
boolean add = addColumnHeadings(tr, grandChildren);
|
||||
boolean add = addColumnHeadings(tr, grandChildren.getList());
|
||||
for (ResourceWrapper v : p.getValues()) {
|
||||
if (v != null) {
|
||||
tr = tbl.tr();
|
||||
tr.td().style("display: none").tx("*"); // work around problem with empty table rows
|
||||
add = addColumnValues(status, res, tr, profile, grandChildren, v, showCodeDetails, displayHints, path, indent) || add;
|
||||
add = addColumnValues(status, res, tr, profile, grandChildren.getList(), v, showCodeDetails, displayHints, indent) || add;
|
||||
}
|
||||
}
|
||||
if (add) {
|
||||
|
@ -393,7 +396,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
XhtmlNode para = x.para();
|
||||
para.b().addText(labelforExtension(sd, p.getUrl()));
|
||||
para.tx(": ");
|
||||
renderLeaf(status, res, vp, profile, child, x, para, false, showCodeDetails, displayHints, path, indent);
|
||||
renderLeaf(status, res, vp, profile, child, x, para, false, showCodeDetails, displayHints, indent);
|
||||
} else if (!ev.isEmpty()) {
|
||||
XhtmlNode bq = x.addTag("blockquote");
|
||||
bq.para().b().addText(labelforExtension(sd, p.getUrl()));
|
||||
|
@ -410,13 +413,13 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
XhtmlNode li = ul.li();
|
||||
li.tx(labelForSubExtension(vv.primitiveValue("url"), sd));
|
||||
li.tx(": ");
|
||||
renderLeaf(status, res, vv.child("value"), sd, child, x, li, isExt, showCodeDetails, displayHints, path, indent);
|
||||
renderLeaf(status, res, vv.child("value"), sd, child, x, li, isExt, showCodeDetails, displayHints, indent);
|
||||
}
|
||||
} else {
|
||||
for (ResourceWrapper vv : ev) {
|
||||
StructureDefinition ex = context.getWorker().fetchTypeDefinition("Extension");
|
||||
List<ElementDefinition> children = getChildrenForPath(profile, ex.getSnapshot().getElement(), "Extension");
|
||||
generateByProfile(status, res, ex, vv, allElements, child, children, bq, "Extension", showCodeDetails, indent+1);
|
||||
SourcedChildDefinitions children = getChildrenForPath(ex, "Extension");
|
||||
generateByProfile(status, res, ex, vv, child, children.getList(), bq, "Extension", showCodeDetails, indent+1);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -427,7 +430,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
if (v != null) {
|
||||
XhtmlNode bq = x.addTag("blockquote");
|
||||
bq.para().b().addText(p.getName());
|
||||
generateByProfile(status, res, profile, v, allElements, child, grandChildren, bq, path+"."+p.getName(), showCodeDetails, indent+1);
|
||||
generateByProfile(status, res, grandChildren.getSource(), v, child, grandChildren.getList(), bq, grandChildren.getPath(), showCodeDetails, indent+1);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -435,6 +438,29 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
}
|
||||
}
|
||||
|
||||
//
|
||||
// private String getGrandChildBase(List<ElementDefinition> grandChildren) {
|
||||
// if (grandChildren == null || grandChildren.isEmpty()) {
|
||||
// return null;
|
||||
// }
|
||||
// String[] path = grandChildren.get(0).getPath().split("\\.");
|
||||
// for (int i = 1; i < grandChildren.size(); i++) {
|
||||
// path = getSharedString(path, grandChildren.get(1).getPath().split("\\."));
|
||||
// }
|
||||
// return CommaSeparatedStringBuilder.join(".", path);
|
||||
// }
|
||||
//
|
||||
// private String[] getSharedString(String[] path, String[] path2) {
|
||||
// int m = -1;
|
||||
// for (int i = 0; i < Integer.min(path.length, path2.length); i++) {
|
||||
// if (path[i].equals(path2[i])) {
|
||||
// m = i;
|
||||
// } else {
|
||||
// break;
|
||||
// }
|
||||
// }
|
||||
// return m == -1 ? new String[0] : Arrays.copyOfRange(path, 0, m+1);
|
||||
// }
|
||||
|
||||
private String labelForSubExtension(String url, StructureDefinition sd) {
|
||||
return url;
|
||||
|
@ -530,7 +556,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
return b;
|
||||
}
|
||||
|
||||
private boolean addColumnValues(RenderingStatus status, ResourceWrapper res, XhtmlNode tr, StructureDefinition profile, List<ElementDefinition> grandChildren, ResourceWrapper v, boolean showCodeDetails, Map<String, String> displayHints, String path, int indent) throws FHIRException, UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
private boolean addColumnValues(RenderingStatus status, ResourceWrapper res, XhtmlNode tr, StructureDefinition profile, List<ElementDefinition> grandChildren, ResourceWrapper v, boolean showCodeDetails, Map<String, String> displayHints, int indent) throws FHIRException, UnsupportedEncodingException, IOException, EOperationOutcome {
|
||||
boolean b = false;
|
||||
for (ElementDefinition e : grandChildren) {
|
||||
List<ResourceWrapper> p = v.children(e.getPath().substring(e.getPath().lastIndexOf(".")+1));
|
||||
|
@ -542,7 +568,7 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
for (ResourceWrapper vv : p) {
|
||||
b = true;
|
||||
td.sep(", ");
|
||||
renderLeaf(status, res, vv, profile, e, td, td, false, showCodeDetails, displayHints, path, indent);
|
||||
renderLeaf(status, res, vv, profile, e, td, td, false, showCodeDetails, displayHints, indent);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -656,6 +682,10 @@ public class ProfileDrivenRenderer extends ResourceRenderer {
|
|||
return path.substring(path.lastIndexOf(".")+1);
|
||||
}
|
||||
|
||||
protected String utail(String path) {
|
||||
return path.contains("/") ? path.substring(path.lastIndexOf("/")+1) : path;
|
||||
}
|
||||
|
||||
public boolean canRender(Resource resource) {
|
||||
return context.getWorker().getResourceNames().contains(resource.fhirType());
|
||||
}
|
||||
|
|
|
@ -35,7 +35,6 @@ import org.hl7.fhir.r5.terminologies.CodeSystemUtilities;
|
|||
import org.hl7.fhir.r5.utils.EOperationOutcome;
|
||||
import org.hl7.fhir.r5.utils.ToolingExtensions;
|
||||
import org.hl7.fhir.r5.utils.XVerExtensionManager;
|
||||
import org.hl7.fhir.utilities.CommaSeparatedStringBuilder;
|
||||
import org.hl7.fhir.utilities.Utilities;
|
||||
import org.hl7.fhir.utilities.xhtml.HierarchicalTableGenerator;
|
||||
import org.hl7.fhir.utilities.xhtml.HierarchicalTableGenerator.Piece;
|
||||
|
@ -369,8 +368,10 @@ public abstract class ResourceRenderer extends DataRenderer {
|
|||
} else {
|
||||
x.tx("??");
|
||||
}
|
||||
checkRenderExtensions(status, x, type);
|
||||
}
|
||||
|
||||
|
||||
public void renderReference(ResourceWrapper res, HierarchicalTableGenerator gen, List<Piece> pieces, Reference r, boolean allowLinks) throws UnsupportedEncodingException, IOException {
|
||||
if (r == null) {
|
||||
pieces.add(gen.new Piece(null, "null!", null));
|
||||
|
|
|
@ -83,6 +83,42 @@ public class TerminologyClientManager {
|
|||
ITerminologyClient makeClient(String id, String url, String userAgent, ToolingClientLogger logger) throws URISyntaxException;
|
||||
String getVersion();
|
||||
}
|
||||
|
||||
public class InternalLogEvent {
|
||||
private String message;
|
||||
private String server;
|
||||
private String vs;
|
||||
private String systems;
|
||||
private String choices;
|
||||
protected InternalLogEvent(String message, String server, String vs, String systems, String choices) {
|
||||
super();
|
||||
this.message = message;
|
||||
this.server = server;
|
||||
this.vs = vs;
|
||||
this.systems = systems;
|
||||
this.choices = choices;
|
||||
}
|
||||
protected InternalLogEvent(String message) {
|
||||
super();
|
||||
this.message = message;
|
||||
}
|
||||
public String getMessage() {
|
||||
return message;
|
||||
}
|
||||
public String getVs() {
|
||||
return vs;
|
||||
}
|
||||
public String getSystems() {
|
||||
return systems;
|
||||
}
|
||||
public String getChoices() {
|
||||
return choices;
|
||||
}
|
||||
public String getServer() {
|
||||
return server;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public static final String UNRESOLVED_VALUESET = "--unknown--";
|
||||
|
||||
|
@ -93,7 +129,7 @@ public class TerminologyClientManager {
|
|||
private List<TerminologyClientContext> serverList = new ArrayList<>(); // clients by server address
|
||||
private Map<String, TerminologyClientContext> serverMap = new HashMap<>(); // clients by server address
|
||||
private Map<String, ServerOptionList> resMap = new HashMap<>(); // client resolution list
|
||||
private List<String> internalLog = new ArrayList<>();
|
||||
private List<InternalLogEvent> internalLog = new ArrayList<>();
|
||||
protected Parameters expParameters;
|
||||
|
||||
private TerminologyCache cache;
|
||||
|
@ -124,6 +160,7 @@ public class TerminologyClientManager {
|
|||
monitorServiceURL = other.monitorServiceURL;
|
||||
factory = other.factory;
|
||||
usage = other.usage;
|
||||
internalLog = other.internalLog;
|
||||
}
|
||||
|
||||
|
||||
|
@ -150,6 +187,7 @@ public class TerminologyClientManager {
|
|||
}
|
||||
}
|
||||
if (ok) {
|
||||
log(vs, s, systems, choices, "Found authoritative server "+s);
|
||||
return findClient(s, systems, expand);
|
||||
}
|
||||
}
|
||||
|
@ -165,6 +203,7 @@ public class TerminologyClientManager {
|
|||
}
|
||||
}
|
||||
if (ok) {
|
||||
log(vs, s, systems, choices, "Found partially authoritative server "+s);
|
||||
return findClient(s, systems, expand);
|
||||
}
|
||||
}
|
||||
|
@ -180,6 +219,7 @@ public class TerminologyClientManager {
|
|||
}
|
||||
}
|
||||
if (ok) {
|
||||
log(vs, s, systems, choices, "Found candidate server "+s);
|
||||
return findClient(s, systems, expand);
|
||||
}
|
||||
}
|
||||
|
@ -194,6 +234,7 @@ public class TerminologyClientManager {
|
|||
"urn:ietf:rfc:3986", "http://www.ama-assn.org/go/cpt", "urn:oid:1.2.36.1.2001.1005.17", "urn:iso:std:iso:3166",
|
||||
"http://varnomen.hgvs.org", "http://unstats.un.org/unsd/methods/m49/m49.htm", "urn:iso:std:iso:4217",
|
||||
"http://hl7.org/fhir/sid/ndc", "http://fhir.ohdsi.org/CodeSystem/concepts", "http://fdasis.nlm.nih.gov", "https://www.usps.com/")) {
|
||||
log(vs, serverList.get(0).getAddress(), systems, choices, "Use primary server for "+uri);
|
||||
return serverList.get(0);
|
||||
}
|
||||
}
|
||||
|
@ -201,30 +242,42 @@ public class TerminologyClientManager {
|
|||
// no agreement? Then what we do depends
|
||||
if (vs != null) {
|
||||
if (vs.hasUserData("External.Link")) {
|
||||
if (systems.size() == 1) {
|
||||
internalLog.add(vs.getVersionedUrl()+" uses the system "+systems.toString()+" not handled by any servers. Using source @ '"+vs.getUserString("External.Link")+"'");
|
||||
} else {
|
||||
internalLog.add(vs.getVersionedUrl()+" includes multiple systems "+systems.toString()+" best handled by multiple servers: "+choices.toString()+". Using source @ '"+vs.getUserString("External.Link")+"'");
|
||||
String el = vs.getUserString("External.Link");
|
||||
if ("https://vsac.nlm.nih.gov".equals(el)) {
|
||||
el = getMaster().getAddress();
|
||||
}
|
||||
return findClient(vs.getUserString("External.Link"), systems, expand);
|
||||
if (systems.size() == 1) {
|
||||
log(vs, el, systems, choices, "Not handled by any servers. Using source @ '"+el+"'");
|
||||
} else {
|
||||
log(vs, el, systems, choices, "Handled by multiple servers. Using source @ '"+el+"'");
|
||||
}
|
||||
return findClient(el, systems, expand);
|
||||
} else {
|
||||
if (systems.size() == 1) {
|
||||
internalLog.add(vs.getVersionedUrl()+" uses the system "+systems.toString()+" not handled by any servers. Using master @ '"+serverList.get(0)+"'");
|
||||
log(vs, serverList.get(0).getAddress(), systems, choices, "System not handled by any servers. Using primary server");
|
||||
} else {
|
||||
internalLog.add(vs.getVersionedUrl()+" includes multiple systems "+systems.toString()+" best handled by multiple servers: "+choices.toString()+". Using master @ '"+serverList.get(0)+"'");
|
||||
log(vs, serverList.get(0).getAddress(), systems, choices, "Systems handled by multiple servers. Using primary server");
|
||||
}
|
||||
return findClient(serverList.get(0).getAddress(), systems, expand);
|
||||
}
|
||||
} else {
|
||||
if (systems.size() == 1) {
|
||||
internalLog.add("Request for system "+systems.toString()+" not handled by any servers. Using master @ '"+serverList.get(0)+"'");
|
||||
log(vs, serverList.get(0).getAddress(), systems, choices, "System not handled by any servers. Using primary server");
|
||||
} else {
|
||||
internalLog.add("Request for multiple systems "+systems.toString()+" best handled by multiple servers: "+choices.toString()+". Using master @ '"+serverList.get(0)+"'");
|
||||
log(vs, serverList.get(0).getAddress(), systems, choices, "Systems handled by multiple servers. Using primary server");
|
||||
}
|
||||
log(vs, serverList.get(0).getAddress(), systems, choices, "Fallback: primary server");
|
||||
return findClient(serverList.get(0).getAddress(), systems, expand);
|
||||
}
|
||||
}
|
||||
|
||||
private void log(ValueSet vs, String server, Set<String> systems, List<ServerOptionList> choices, String message) {
|
||||
String svs = (vs == null ? "null" : vs.getVersionedUrl());
|
||||
String sys = systems.isEmpty() ? "--" : systems.size() == 1 ? systems.iterator().next() : systems.toString();
|
||||
String sch = choices.isEmpty() ? "--" : choices.size() == 1 ? choices.iterator().next().toString() : choices.toString();
|
||||
internalLog.add(new InternalLogEvent(message, server, svs, sys, sch));
|
||||
}
|
||||
|
||||
private TerminologyClientContext findClient(String server, Set<String> systems, boolean expand) {
|
||||
TerminologyClientContext client = serverMap.get(server);
|
||||
if (client == null) {
|
||||
|
@ -293,8 +346,8 @@ public class TerminologyClientManager {
|
|||
return ret;
|
||||
} catch (Exception e) {
|
||||
String msg = "Error resolving system "+url+": "+e.getMessage()+" ("+request+")";
|
||||
if (!internalLog.contains(msg)) {
|
||||
internalLog.add(msg);
|
||||
if (!hasMessage(msg)) {
|
||||
internalLog.add(new InternalLogEvent(msg));
|
||||
}
|
||||
if (!monitorServiceURL.contains("tx.fhir.org")) {
|
||||
e.printStackTrace();
|
||||
|
@ -304,6 +357,15 @@ public class TerminologyClientManager {
|
|||
|
||||
}
|
||||
|
||||
private boolean hasMessage(String msg) {
|
||||
for (InternalLogEvent log : internalLog) {
|
||||
if (msg.equals(log.message)) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public List<TerminologyClientContext> serverList() {
|
||||
return serverList;
|
||||
}
|
||||
|
@ -406,10 +468,6 @@ public class TerminologyClientManager {
|
|||
}
|
||||
}
|
||||
|
||||
public List<String> getInternalLog() {
|
||||
return internalLog;
|
||||
}
|
||||
|
||||
public List<TerminologyClientContext> getServerList() {
|
||||
return serverList;
|
||||
}
|
||||
|
@ -514,8 +572,8 @@ public class TerminologyClientManager {
|
|||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
String msg = "Error resolving valueSet "+canonical+": "+e.getMessage()+" ("+request+")";
|
||||
if (!internalLog.contains(msg)) {
|
||||
internalLog.add(msg);
|
||||
if (!hasMessage(msg)) {
|
||||
internalLog.add(new InternalLogEvent(msg));
|
||||
}
|
||||
e.printStackTrace();
|
||||
return null;
|
||||
|
@ -594,8 +652,8 @@ public class TerminologyClientManager {
|
|||
} catch (Exception e) {
|
||||
e.printStackTrace();
|
||||
String msg = "Error resolving valueSet "+canonical+": "+e.getMessage()+" ("+request+")";
|
||||
if (!internalLog.contains(msg)) {
|
||||
internalLog.add(msg);
|
||||
if (!hasMessage(msg)) {
|
||||
internalLog.add(new InternalLogEvent(msg));
|
||||
}
|
||||
e.printStackTrace();
|
||||
return null;
|
||||
|
@ -610,5 +668,8 @@ public class TerminologyClientManager {
|
|||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
|
||||
public List<InternalLogEvent> getInternalLog() {
|
||||
return internalLog;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -364,7 +364,7 @@ public class TerminologyCache {
|
|||
csCache.clear();
|
||||
}
|
||||
|
||||
private void clear() throws IOException {
|
||||
public void clear() throws IOException {
|
||||
if (folder != null) {
|
||||
Utilities.clearDirectory(folder);
|
||||
}
|
||||
|
|
|
@ -65,6 +65,7 @@ public class CompareUtilities extends BaseTestingUtilities {
|
|||
case "$$" : return "$$";
|
||||
case "$instant$": return "\"An Instant\"";
|
||||
case "$uuid$": return "\"A Uuid\"";
|
||||
case "$id$": return "\"An Id\"";
|
||||
default: return "Unhandled template: "+expected;
|
||||
}
|
||||
}
|
||||
|
@ -466,6 +467,7 @@ public class CompareUtilities extends BaseTestingUtilities {
|
|||
case "$$" : return true;
|
||||
case "$instant$": return actualJsonString.matches("([0-9]([0-9]([0-9][1-9]|[1-9]0)|[1-9]00)|[1-9]000)-(0[1-9]|1[0-2])-(0[1-9]|[1-2][0-9]|3[0-1])T([01][0-9]|2[0-3]):[0-5][0-9]:([0-5][0-9]|60)(\\.[0-9]{1,9})?(Z|(\\+|-)((0[0-9]|1[0-3]):[0-5][0-9]|14:00))");
|
||||
case "$uuid$": return actualJsonString.matches("urn:uuid:[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}");
|
||||
case "$id$": return actualJsonString.matches("[A-Za-z0-9\\-\\.]{1,64}");
|
||||
default:
|
||||
throw new Error("Unhandled template: "+expectedJsonString);
|
||||
}
|
||||
|
|
|
@ -35,6 +35,7 @@ import java.util.List;
|
|||
|
||||
import org.hl7.fhir.r5.model.CodeableConcept;
|
||||
import org.hl7.fhir.r5.model.IntegerType;
|
||||
import org.hl7.fhir.r5.model.Narrative.NarrativeStatus;
|
||||
import org.hl7.fhir.r5.model.OperationOutcome;
|
||||
import org.hl7.fhir.r5.model.OperationOutcome.IssueSeverity;
|
||||
import org.hl7.fhir.r5.model.OperationOutcome.IssueType;
|
||||
|
@ -43,6 +44,8 @@ import org.hl7.fhir.r5.model.StringType;
|
|||
import org.hl7.fhir.r5.model.UrlType;
|
||||
import org.hl7.fhir.utilities.CommaSeparatedStringBuilder;
|
||||
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
||||
import org.hl7.fhir.utilities.xhtml.NodeType;
|
||||
import org.hl7.fhir.utilities.xhtml.XhtmlNode;
|
||||
|
||||
public class OperationOutcomeUtilities {
|
||||
|
||||
|
@ -180,4 +183,16 @@ public class OperationOutcomeUtilities {
|
|||
return res;
|
||||
}
|
||||
|
||||
public static OperationOutcome outcomeFromTextError(String text) {
|
||||
OperationOutcome oo = new OperationOutcome();
|
||||
oo.getText().setStatus(NarrativeStatus.GENERATED);
|
||||
oo.getText().setDiv(new XhtmlNode(NodeType.Element, "div"));
|
||||
oo.getText().getDiv().tx(text);
|
||||
OperationOutcomeIssueComponent issue = oo.addIssue();
|
||||
issue.setSeverity(IssueSeverity.ERROR);
|
||||
issue.setCode(IssueType.EXCEPTION);
|
||||
issue.getDetails().setText(text);
|
||||
return oo;
|
||||
}
|
||||
|
||||
}
|
|
@ -18,8 +18,8 @@ public class ByteUtils {
|
|||
public static <T extends Resource> byte[] resourceToByteArray(T resource, boolean pretty, boolean isJson, boolean noXhtml) {
|
||||
ByteArrayOutputStream baos = null;
|
||||
byte[] byteArray = null;
|
||||
baos = new ByteArrayOutputStream();
|
||||
try {
|
||||
baos = new ByteArrayOutputStream();
|
||||
IParser parser = null;
|
||||
if (isJson) {
|
||||
parser = new JsonParser();
|
||||
|
|
|
@ -1,6 +1,12 @@
|
|||
package org.hl7.fhir.r5.utils.client.network;
|
||||
|
||||
import okhttp3.*;
|
||||
import java.io.IOException;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
|
||||
import org.apache.commons.lang3.StringUtils;
|
||||
import org.hl7.fhir.exceptions.FHIRException;
|
||||
import org.hl7.fhir.r5.formats.IParser;
|
||||
|
@ -9,18 +15,21 @@ import org.hl7.fhir.r5.formats.XmlParser;
|
|||
import org.hl7.fhir.r5.model.Bundle;
|
||||
import org.hl7.fhir.r5.model.OperationOutcome;
|
||||
import org.hl7.fhir.r5.model.Resource;
|
||||
import org.hl7.fhir.r5.utils.OperationOutcomeUtilities;
|
||||
import org.hl7.fhir.r5.utils.ResourceUtilities;
|
||||
import org.hl7.fhir.r5.utils.client.EFhirClientException;
|
||||
import org.hl7.fhir.r5.utils.client.ResourceFormat;
|
||||
import org.hl7.fhir.utilities.MimeType;
|
||||
import org.hl7.fhir.utilities.Utilities;
|
||||
import org.hl7.fhir.utilities.settings.FhirSettings;
|
||||
import org.hl7.fhir.utilities.xhtml.XhtmlUtils;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
import java.io.IOException;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import okhttp3.Authenticator;
|
||||
import okhttp3.Credentials;
|
||||
import okhttp3.Headers;
|
||||
import okhttp3.OkHttpClient;
|
||||
import okhttp3.Request;
|
||||
import okhttp3.Response;
|
||||
|
||||
public class FhirRequestBuilder {
|
||||
|
||||
|
@ -227,7 +236,7 @@ public class FhirRequestBuilder {
|
|||
public <T extends Resource> ResourceRequest<T> execute() throws IOException {
|
||||
formatHeaders(httpRequest, resourceFormat, headers);
|
||||
Response response = getHttpClient().newCall(httpRequest.build()).execute();
|
||||
T resource = unmarshalReference(response, resourceFormat);
|
||||
T resource = unmarshalReference(response, resourceFormat, null);
|
||||
return new ResourceRequest<T>(resource, response.code(), getLocationHeader(response.headers()));
|
||||
}
|
||||
|
||||
|
@ -241,50 +250,60 @@ public class FhirRequestBuilder {
|
|||
* Unmarshalls a resource from the response stream.
|
||||
*/
|
||||
@SuppressWarnings("unchecked")
|
||||
protected <T extends Resource> T unmarshalReference(Response response, String format) {
|
||||
T resource = null;
|
||||
OperationOutcome error = null;
|
||||
|
||||
if (response.body() != null) {
|
||||
try {
|
||||
byte[] body = response.body().bytes();
|
||||
resource = (T) getParser(format).parse(body);
|
||||
if (resource instanceof OperationOutcome && hasError((OperationOutcome) resource)) {
|
||||
error = (OperationOutcome) resource;
|
||||
}
|
||||
} catch (IOException ioe) {
|
||||
throw new EFhirClientException("Error reading Http Response from "+source+": " + ioe.getMessage(), ioe);
|
||||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": " + e.getMessage(), e);
|
||||
protected <T extends Resource> T unmarshalReference(Response response, String format, String resourceType) {
|
||||
int code = response.code();
|
||||
boolean ok = code >= 200 && code < 300;
|
||||
if (response.body() == null) {
|
||||
if (!ok) {
|
||||
throw new EFhirClientException(response.message());
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
if (error != null) {
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
}
|
||||
|
||||
return resource;
|
||||
}
|
||||
|
||||
/**
|
||||
* Unmarshalls Bundle from response stream.
|
||||
*/
|
||||
protected Bundle unmarshalFeed(Response response, String format) {
|
||||
Bundle feed = null;
|
||||
OperationOutcome error = null;
|
||||
String body;
|
||||
|
||||
Resource resource = null;
|
||||
try {
|
||||
byte[] body = response.body().bytes();
|
||||
String contentType = response.header("Content-Type");
|
||||
if (body != null) {
|
||||
if (contentType.contains(ResourceFormat.RESOURCE_XML.getHeader()) || contentType.contains(ResourceFormat.RESOURCE_JSON.getHeader()) || contentType.contains("text/xml+fhir")) {
|
||||
Resource rf = getParser(format).parse(body);
|
||||
if (rf instanceof Bundle)
|
||||
feed = (Bundle) rf;
|
||||
else if (rf instanceof OperationOutcome && hasError((OperationOutcome) rf)) {
|
||||
error = (OperationOutcome) rf;
|
||||
} else {
|
||||
throw new EFhirClientException("Error reading server response: a resource was returned instead");
|
||||
body = response.body().string();
|
||||
String ct = response.header("Content-Type");
|
||||
if (ct == null) {
|
||||
if (ok) {
|
||||
resource = getParser(format).parse(body);
|
||||
} else {
|
||||
System.out.println("Got error response with no Content-Type from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
}
|
||||
} else {
|
||||
if (ct.contains(";")) {
|
||||
ct = ct.substring(0, ct.indexOf(";"));
|
||||
}
|
||||
switch (ct) {
|
||||
case "application/json":
|
||||
case "application/fhir+json":
|
||||
if (!format.contains("json")) {
|
||||
System.out.println("Got json response expecting "+format+" from "+source+" with status "+code);
|
||||
}
|
||||
resource = getParser(ResourceFormat.RESOURCE_JSON.getHeader()).parse(body);
|
||||
break;
|
||||
case "application/xml":
|
||||
case "application/fhir+xml":
|
||||
case "text/xml":
|
||||
if (!format.contains("xml")) {
|
||||
System.out.println("Got xml response expecting "+format+" from "+source+" with status "+code);
|
||||
}
|
||||
resource = getParser(ResourceFormat.RESOURCE_XML.getHeader()).parse(body);
|
||||
break;
|
||||
case "text/plain":
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
break;
|
||||
case "text/html" :
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.body().string(), source));
|
||||
break;
|
||||
default: // not sure what else to do?
|
||||
System.out.println("Got content-type '"+ct+"' from "+source);
|
||||
System.out.println(body);
|
||||
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
|
||||
}
|
||||
}
|
||||
} catch (IOException ioe) {
|
||||
|
@ -292,10 +311,33 @@ public class FhirRequestBuilder {
|
|||
} catch (Exception e) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": "+e.getMessage(), e);
|
||||
}
|
||||
if (error != null) {
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
if (resource instanceof OperationOutcome && (!"OperationOutcome".equals(resourceType) || !ok)) {
|
||||
OperationOutcome error = (OperationOutcome) resource;
|
||||
if (hasError((OperationOutcome) resource)) {
|
||||
throw new EFhirClientException("Error from "+source+": " + ResourceUtilities.getErrorDescription(error), error);
|
||||
} else {
|
||||
// umm, weird...
|
||||
System.out.println("Got OperationOutcome with no error from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
return feed;
|
||||
if (resource == null) {
|
||||
System.out.println("No resource from "+source+" with status "+code);
|
||||
System.out.println(body);
|
||||
return null; // shouldn't get here?
|
||||
}
|
||||
if (resourceType != null && !resource.fhirType().equals(resourceType)) {
|
||||
throw new EFhirClientException("Error parsing response message from "+source+": Found an "+resource.fhirType()+" looking for a "+resourceType);
|
||||
}
|
||||
return (T) resource;
|
||||
}
|
||||
|
||||
/**
|
||||
* Unmarshalls Bundle from response stream.
|
||||
*/
|
||||
protected Bundle unmarshalFeed(Response response, String format) {
|
||||
return unmarshalReference(response, format, "Bundle");
|
||||
}
|
||||
|
||||
/**
|
||||
|
|
|
@ -80,7 +80,7 @@ public interface IResourceValidator {
|
|||
IResourceValidator setFetcher(IValidatorResourceFetcher value);
|
||||
|
||||
IValidationPolicyAdvisor getPolicyAdvisor();
|
||||
IResourceValidator setPolicyAdvisor(IValidationPolicyAdvisor advisor);
|
||||
void setPolicyAdvisor(IValidationPolicyAdvisor advisor);
|
||||
|
||||
IValidationProfileUsageTracker getTracker();
|
||||
IResourceValidator setTracker(IValidationProfileUsageTracker value);
|
||||
|
|
|
@ -16,6 +16,21 @@ import org.hl7.fhir.r5.utils.validation.constants.BindingKind;
|
|||
|
||||
public interface IValidationPolicyAdvisor {
|
||||
|
||||
/**
|
||||
* Internal use, for chaining advisors
|
||||
*
|
||||
* @return
|
||||
*/
|
||||
ReferenceValidationPolicy getReferencePolicy();
|
||||
|
||||
/**
|
||||
*
|
||||
* @param path - the current path of the element
|
||||
* @param messageId - the message id (from messages.properties)
|
||||
* @return true if the validator should ignore the message
|
||||
*/
|
||||
boolean isSuppressMessageId(String path, String messageId);
|
||||
|
||||
/**
|
||||
*
|
||||
* @param validator
|
||||
|
@ -169,5 +184,6 @@ public interface IValidationPolicyAdvisor {
|
|||
boolean valid,
|
||||
IMessagingServices msgServices,
|
||||
List<ValidationMessage> messages);
|
||||
|
||||
|
||||
}
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -514,6 +514,7 @@ public class I18nConstants {
|
|||
public static final String QUESTIONNAIRE_QR_ITEM_ONLYONEI = "Questionnaire_QR_Item_OnlyOneI";
|
||||
public static final String QUESTIONNAIRE_QR_ITEM_ORDER = "Questionnaire_QR_Item_Order";
|
||||
public static final String QUESTIONNAIRE_QR_ITEM_STRINGNOOPTIONS = "Questionnaire_QR_Item_StringNoOptions";
|
||||
public static final String QUESTIONNAIRE_QR_ITEM_STRING_IN_CODING = "QUESTIONNAIRE_QR_ITEM_STRING_IN_CODING";
|
||||
public static final String QUESTIONNAIRE_QR_ITEM_TEXT = "Questionnaire_QR_Item_Text";
|
||||
public static final String QUESTIONNAIRE_QR_ITEM_TIMENOOPTIONS = "Questionnaire_QR_Item_TimeNoOptions";
|
||||
public static final String QUESTIONNAIRE_QR_ITEM_WRONGTYPE = "Questionnaire_QR_Item_WrongType";
|
||||
|
|
|
@ -348,7 +348,13 @@ public class XhtmlNode extends XhtmlFluent implements IBaseXhtml {
|
|||
}
|
||||
}
|
||||
if (n.getNodeType() == NodeType.Element) {
|
||||
b.append(n.allText());
|
||||
if (!Utilities.existsInList(n.getName(), "img")) {
|
||||
b.append(n.allText());
|
||||
} else if (n.hasAttribute("alt")) {
|
||||
b.append(n.getAttribute("alt"));
|
||||
} else {
|
||||
b.append("[image]");
|
||||
}
|
||||
if (Utilities.existsInList(n.getName(), "p", "div", "tr", "th", "ul", "ol", "li", "h1", "h2", "h3", "h4", "h5", "h6")) {
|
||||
b.append("\r\n");
|
||||
} else if (Utilities.existsInList(n.getName(), "th", "td", "span")) {
|
||||
|
|
|
@ -0,0 +1,18 @@
|
|||
package org.hl7.fhir.utilities.xhtml;
|
||||
|
||||
public class XhtmlUtils {
|
||||
|
||||
public static String convertHtmlToText(String source, String desc) {
|
||||
try {
|
||||
XhtmlDocument doc = new XhtmlParser().parse(source, "html");
|
||||
return doc.getDocumentElement().allText();
|
||||
} catch (Exception e) {
|
||||
// todo - should we try another way?
|
||||
System.err.println("XHTML content could not be parsed from "+desc);
|
||||
e.printStackTrace();
|
||||
System.err.println(source);
|
||||
return "Unparseable HTML";
|
||||
}
|
||||
}
|
||||
|
||||
}
|
|
@ -524,6 +524,7 @@ Questionnaire_QR_Item_NoOptionsCoding = Option list has no option values of type
|
|||
Questionnaire_QR_Item_NoOptionsDate = Option list has no option values of type date
|
||||
Questionnaire_QR_Item_NoOptionsInteger = Option list has no option values of type integer
|
||||
Questionnaire_QR_Item_NoOptionsString = Option list has no option values of type string
|
||||
QUESTIONNAIRE_QR_ITEM_STRING_IN_CODING = The string value ''{0}'' matches an entry in the list of valid Codings, so this is probably an error
|
||||
Questionnaire_QR_Item_NoOptionsTime = Option list has no option values of type time
|
||||
Questionnaire_QR_Item_NoString = The string {0} is not a valid option
|
||||
Questionnaire_QR_Item_NoTime = The time {0} is not a valid option
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -5,7 +5,7 @@
|
|||
<parent>
|
||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<relativePath>../pom.xml</relativePath>
|
||||
</parent>
|
||||
|
||||
|
|
|
@ -69,9 +69,11 @@ import org.hl7.fhir.r5.utils.XVerExtensionManager;
|
|||
import org.hl7.fhir.r5.utils.XVerExtensionManager.XVerExtensionStatus;
|
||||
import org.hl7.fhir.r5.utils.validation.IMessagingServices;
|
||||
import org.hl7.fhir.r5.utils.validation.IResourceValidator;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidatorResourceFetcher;
|
||||
import org.hl7.fhir.r5.utils.validation.ValidationContextCarrier.IValidationContextResourceLoader;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.BestPracticeWarningLevel;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ReferenceValidationPolicy;
|
||||
import org.hl7.fhir.utilities.CommaSeparatedStringBuilder;
|
||||
import org.hl7.fhir.utilities.FhirPublication;
|
||||
import org.hl7.fhir.utilities.StandardsStatus;
|
||||
|
@ -85,6 +87,7 @@ import org.hl7.fhir.utilities.validation.ValidationMessage.IssueType;
|
|||
import org.hl7.fhir.utilities.validation.ValidationMessage.Source;
|
||||
import org.hl7.fhir.utilities.xhtml.XhtmlNode;
|
||||
import org.hl7.fhir.validation.cli.utils.ValidationLevel;
|
||||
import org.hl7.fhir.validation.instance.advisor.BasePolicyAdvisorForFullValidation;
|
||||
import org.hl7.fhir.validation.instance.utils.IndexedElement;
|
||||
import org.hl7.fhir.validation.instance.utils.NodeStack;
|
||||
|
||||
|
@ -165,25 +168,31 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
protected final String BUNDLE = "Bundle";
|
||||
protected final String LAST_UPDATED = "lastUpdated";
|
||||
|
||||
protected String sessionId;
|
||||
protected BaseValidator parent;
|
||||
protected Source source;
|
||||
protected IWorkerContext context;
|
||||
protected ValidationTimeTracker timeTracker = new ValidationTimeTracker();
|
||||
protected XVerExtensionManager xverManager;
|
||||
protected List<TrackedLocationRelatedMessage> trackedMessages = new ArrayList<>();
|
||||
protected List<ValidationMessage> messagesToRemove = new ArrayList<>();
|
||||
protected ValidationLevel level = ValidationLevel.HINTS;
|
||||
protected Coding jurisdiction;
|
||||
protected boolean allowExamples;
|
||||
protected boolean forPublication;
|
||||
protected boolean debug;
|
||||
protected boolean warnOnDraftOrExperimental;
|
||||
protected Set<String> statusWarnings = new HashSet<>();
|
||||
protected BestPracticeWarningLevel bpWarnings = BestPracticeWarningLevel.Warning;
|
||||
protected String sessionId = Utilities.makeUuidLC();
|
||||
protected List<UsageContext> usageContexts = new ArrayList<UsageContext>();
|
||||
protected ValidationOptions baseOptions = new ValidationOptions(FhirPublication.R5);
|
||||
protected IValidatorResourceFetcher fetcher;
|
||||
protected IValidationPolicyAdvisor policyAdvisor;
|
||||
|
||||
// these two related to removing warnings on extensible bindings in structures that have derivatives that replace their bindings
|
||||
protected List<TrackedLocationRelatedMessage> trackedMessages = new ArrayList<>();
|
||||
protected List<ValidationMessage> messagesToRemove = new ArrayList<>();
|
||||
|
||||
// don't repeatedly raise the same warnings all the time
|
||||
protected Set<String> statusWarnings = new HashSet<>();
|
||||
|
||||
protected Source source; // @configuration
|
||||
protected ValidationLevel level = ValidationLevel.HINTS; // @configuration
|
||||
protected Coding jurisdiction; // @configuration
|
||||
protected boolean allowExamples; // @configuration
|
||||
protected boolean forPublication; // @configuration
|
||||
protected boolean debug; // @configuration
|
||||
protected boolean warnOnDraftOrExperimental; // @configuration
|
||||
protected BestPracticeWarningLevel bpWarnings = BestPracticeWarningLevel.Warning; // @configuration
|
||||
protected List<UsageContext> usageContexts = new ArrayList<UsageContext>(); // @configuration
|
||||
protected ValidationOptions baseOptions = new ValidationOptions(FhirPublication.R5); // @configuration
|
||||
|
||||
public BaseValidator(IWorkerContext context, XVerExtensionManager xverManager, boolean debug) {
|
||||
super();
|
||||
|
@ -193,6 +202,8 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
this.xverManager = new XVerExtensionManager(context);
|
||||
}
|
||||
this.debug = debug;
|
||||
sessionId = Utilities.makeUuidLC();
|
||||
policyAdvisor = new BasePolicyAdvisorForFullValidation(ReferenceValidationPolicy.CHECK_VALID);
|
||||
urlRegex = Constants.URI_REGEX_XVER.replace("$$", CommaSeparatedStringBuilder.join("|", context.getResourceNames()));
|
||||
}
|
||||
|
||||
|
@ -217,6 +228,8 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
this.usageContexts.addAll(parent.usageContexts);
|
||||
this.baseOptions = parent.baseOptions;
|
||||
this.fetcher = parent.fetcher;
|
||||
this.sessionId = parent.sessionId;
|
||||
this.policyAdvisor = parent.policyAdvisor;
|
||||
}
|
||||
|
||||
private boolean doingLevel(IssueSeverity error) {
|
||||
|
@ -261,75 +274,22 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
|
||||
protected String urlRegex;
|
||||
|
||||
/**
|
||||
* Test a rule and add a {@link IssueSeverity#FATAL} validation message if the validation fails
|
||||
*
|
||||
* @param thePass
|
||||
* Set this parameter to <code>false</code> if the validation does not pass
|
||||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
@Deprecated
|
||||
protected boolean fail(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg) {
|
||||
if (!thePass && doingErrors()) {
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, msg, IssueSeverity.FATAL, null);
|
||||
private boolean suppressMsg(String path, String theMessage) {
|
||||
if (policyAdvisor == null) {
|
||||
return false;
|
||||
} else {
|
||||
return policyAdvisor.isSuppressMessageId(path, theMessage);
|
||||
}
|
||||
return thePass;
|
||||
}
|
||||
|
||||
protected boolean fail(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(path, theMessage)) {
|
||||
String msg = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, msg, IssueSeverity.FATAL, theMessage);
|
||||
}
|
||||
return thePass;
|
||||
}
|
||||
|
||||
/**
|
||||
* Test a rule and add a {@link IssueSeverity#FATAL} validation message if the validation fails
|
||||
*
|
||||
* @param thePass
|
||||
* Set this parameter to <code>false</code> if the validation does not pass
|
||||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
@Deprecated
|
||||
protected boolean fail(List<ValidationMessage> errors, String ruleDate, IssueType type, List<String> pathParts, boolean thePass, String msg) {
|
||||
if (!thePass && doingErrors()) {
|
||||
String path = toPath(pathParts);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, msg, IssueSeverity.FATAL, null);
|
||||
}
|
||||
return thePass;
|
||||
}
|
||||
|
||||
/**
|
||||
* Test a rule and add a {@link IssueSeverity#FATAL} validation message if the validation fails
|
||||
*
|
||||
* @param thePass
|
||||
* Set this parameter to <code>false</code> if the validation does not pass
|
||||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
@Deprecated
|
||||
protected boolean fail(List<ValidationMessage> errors, String ruleDate, IssueType type, List<String> pathParts, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
String path = toPath(pathParts);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, context.formatMessage(theMessage, theMessageArguments), IssueSeverity.FATAL, theMessage);
|
||||
}
|
||||
return thePass;
|
||||
}
|
||||
|
||||
/**
|
||||
* Test a rule and add a {@link IssueSeverity#FATAL} validation message if the validation fails
|
||||
*
|
||||
* @param thePass
|
||||
* Set this parameter to <code>false</code> if the validation does not pass
|
||||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
@Deprecated
|
||||
protected boolean fail(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, String msg) {
|
||||
if (!thePass && doingErrors()) {
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, msg, IssueSeverity.FATAL, null);
|
||||
}
|
||||
return thePass;
|
||||
}
|
||||
//TODO: i18n
|
||||
protected boolean grammarWord(String w) {
|
||||
return w.equals("and") || w.equals("or") || w.equals("a") || w.equals("the") || w.equals("for") || w.equals("this") || w.equals("that") || w.equals("of");
|
||||
|
@ -343,7 +303,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean hint(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg) {
|
||||
if (!thePass && doingHints()) {
|
||||
if (!thePass && doingHints() && !suppressMsg(path, msg)) {
|
||||
String message = context.formatMessage(msg);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.INFORMATION, msg);
|
||||
}
|
||||
|
@ -358,7 +318,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean hintInv(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg, String invId) {
|
||||
if (!thePass && doingHints()) {
|
||||
if (!thePass && doingHints() && !suppressMsg(path, invId)) {
|
||||
String message = context.formatMessage(msg);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.INFORMATION, msg).setInvId(invId);
|
||||
}
|
||||
|
@ -397,7 +357,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean hint(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingHints()) {
|
||||
if (!thePass && doingHints() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.INFORMATION, theMessage);
|
||||
}
|
||||
|
@ -405,7 +365,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean hintPlural(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, int num, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingHints()) {
|
||||
if (!thePass && doingHints() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessagePlural(num, theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.INFORMATION, theMessage);
|
||||
}
|
||||
|
@ -418,7 +378,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean txHint(List<ValidationMessage> errors, String ruleDate, String txLink, IssueType type, int line, int col, String path, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingHints()) {
|
||||
if (!thePass && doingHints() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.INFORMATION, Source.TerminologyEngine, theMessage).setTxLink(txLink);
|
||||
}
|
||||
|
@ -433,7 +393,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean hint(List<ValidationMessage> errors, String ruleDate, IssueType type, List<String> pathParts, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingHints()) {
|
||||
if (!thePass && doingHints() && !suppressMsg(CommaSeparatedStringBuilder.join(".", pathParts), theMessage)) {
|
||||
String path = toPath(pathParts);
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.INFORMATION, theMessage);
|
||||
|
@ -449,7 +409,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean hint(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingHints()) {
|
||||
if (!thePass && doingHints() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.INFORMATION, null);
|
||||
}
|
||||
|
@ -464,7 +424,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean rule(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.ERROR, theMessage);
|
||||
}
|
||||
|
@ -472,7 +432,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean ruleInv(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String theMessage, String invId, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.ERROR, invId).setInvId(invId);
|
||||
}
|
||||
|
@ -480,7 +440,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean rule(List<ValidationMessage> errors, String ruleDate, IssueType type, NodeStack stack, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(stack.getLiteralPath(), theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, stack.line(), stack.col(), stack.getLiteralPath(), message, IssueSeverity.ERROR, theMessage);
|
||||
}
|
||||
|
@ -492,7 +452,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean rulePlural(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, int num, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessagePlural(num, theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, message, IssueSeverity.ERROR, theMessage);
|
||||
}
|
||||
|
@ -500,7 +460,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean txRule(List<ValidationMessage> errors, String ruleDate, String txLink, IssueType type, int line, int col, String path, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
ValidationMessage vm = new ValidationMessage(Source.TerminologyEngine, type, line, col, path, message, IssueSeverity.ERROR).setMessageId(idForMessage(theMessage, message));
|
||||
vm.setRuleDate(ruleDate);
|
||||
|
@ -538,7 +498,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean rule(List<ValidationMessage> errors, String ruleDate, IssueType type, List<String> pathParts, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(CommaSeparatedStringBuilder.join(".", pathParts), theMessage)) {
|
||||
String path = toPath(pathParts);
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.ERROR, theMessage);
|
||||
|
@ -556,7 +516,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
|
||||
|
||||
protected boolean rule(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.ERROR, theMessage);
|
||||
}
|
||||
|
@ -564,7 +524,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean rulePlural(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, int num, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingErrors()) {
|
||||
if (!thePass && doingErrors() && !suppressMsg(path, theMessage)) {
|
||||
String message = context.formatMessagePlural(num, theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.ERROR, theMessage);
|
||||
}
|
||||
|
@ -630,7 +590,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean warning(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
IssueSeverity severity = IssueSeverity.WARNING;
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, nmsg, severity, msg);
|
||||
|
@ -640,7 +600,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean warning(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, String id, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
IssueSeverity severity = IssueSeverity.WARNING;
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, nmsg, severity, id);
|
||||
|
@ -650,7 +610,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean warningInv(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg, String invId, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, invId)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
IssueSeverity severity = IssueSeverity.WARNING;
|
||||
String id = idForMessage(msg, nmsg);
|
||||
|
@ -669,7 +629,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean warningPlural(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, int num, String msg, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessagePlural(num, msg, theMessageArguments);
|
||||
IssueSeverity severity = IssueSeverity.WARNING;
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, nmsg, severity, msg);
|
||||
|
@ -711,7 +671,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean txWarning(List<ValidationMessage> errors, String ruleDate, String txLink, IssueType type, int line, int col, String path, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
ValidationMessage vmsg = new ValidationMessage(Source.TerminologyEngine, type, line, col, path, nmsg, IssueSeverity.WARNING).setTxLink(txLink).setMessageId(idForMessage(msg, nmsg));
|
||||
vmsg.setRuleDate(ruleDate);
|
||||
|
@ -744,7 +704,9 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
vmsg.setMessageId(issue.getExtensionString(ToolingExtensions.EXT_ISSUE_MSG_ID));
|
||||
}
|
||||
|
||||
errors.add(vmsg);
|
||||
if (!suppressMsg(path, vmsg.getMessageId())) {
|
||||
errors.add(vmsg);
|
||||
}
|
||||
return vmsg;
|
||||
}
|
||||
|
||||
|
@ -760,7 +722,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean txWarningForLaterRemoval(Object location, List<ValidationMessage> errors, String ruleDate, String txLink, IssueType type, int line, int col, String path, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
ValidationMessage vmsg = new ValidationMessage(Source.TerminologyEngine, type, line, col, path, nmsg, IssueSeverity.WARNING).setTxLink(txLink).setMessageId(msg);
|
||||
vmsg.setRuleDate(ruleDate);
|
||||
|
@ -789,7 +751,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean warningOrError(boolean isError, List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass) {
|
||||
if (!thePass && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
IssueSeverity lvl = isError ? IssueSeverity.ERROR : IssueSeverity.WARNING;
|
||||
if (doingLevel(lvl)) {
|
||||
|
@ -805,7 +767,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
}
|
||||
|
||||
protected boolean hintOrError(boolean isError, List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass) {
|
||||
if (!thePass && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
IssueSeverity lvl = isError ? IssueSeverity.ERROR : IssueSeverity.INFORMATION;
|
||||
if (doingLevel(lvl)) {
|
||||
|
@ -824,7 +786,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean warning(List<ValidationMessage> errors, String ruleDate, IssueType type, List<String> pathParts, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(CommaSeparatedStringBuilder.join(".", pathParts), theMessage)) {
|
||||
String path = toPath(pathParts);
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.WARNING, theMessage);
|
||||
|
@ -840,7 +802,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean warning(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String message = context.formatMessage(msg, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.WARNING, null);
|
||||
}
|
||||
|
@ -855,7 +817,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean warningOrHint(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, boolean warning, String msg, Object... theMessageArguments) {
|
||||
if (!thePass) {
|
||||
if (!thePass && !suppressMsg(path, msg)) {
|
||||
String message = context.formatMessage(msg, theMessageArguments);
|
||||
IssueSeverity lvl = warning ? IssueSeverity.WARNING : IssueSeverity.INFORMATION;
|
||||
if (doingLevel(lvl)) {
|
||||
|
@ -873,7 +835,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean warningHtml(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, String msg, String html) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
addValidationMessage(errors, ruleDate, type, path, msg, html, IssueSeverity.WARNING, null);
|
||||
}
|
||||
return thePass;
|
||||
|
@ -887,7 +849,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean warningHtml(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, String msg, String html, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, path, nmsg, html, IssueSeverity.WARNING, msg);
|
||||
}
|
||||
|
@ -903,7 +865,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean suppressedwarning(List<ValidationMessage> errors, String ruleDate, IssueType type, int line, int col, String path, boolean thePass, String msg, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, line, col, path, nmsg, IssueSeverity.INFORMATION, msg);
|
||||
}
|
||||
|
@ -919,7 +881,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean suppressedwarning(List<ValidationMessage> errors, String ruleDate, IssueType type, List<String> pathParts, boolean thePass, String theMessage, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(CommaSeparatedStringBuilder.join(".", pathParts), theMessage)) {
|
||||
String path = toPath(pathParts);
|
||||
String message = context.formatMessage(theMessage, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, -1, -1, path, message, IssueSeverity.INFORMATION, theMessage);
|
||||
|
@ -974,7 +936,7 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
* @return Returns <code>thePass</code> (in other words, returns <code>true</code> if the rule did not fail validation)
|
||||
*/
|
||||
protected boolean suppressedwarning(List<ValidationMessage> errors, String ruleDate, IssueType type, String path, boolean thePass, String msg, String html, Object... theMessageArguments) {
|
||||
if (!thePass && doingWarnings()) {
|
||||
if (!thePass && doingWarnings() && !suppressMsg(path, msg)) {
|
||||
String nmsg = context.formatMessage(msg, theMessageArguments);
|
||||
addValidationMessage(errors, ruleDate, type, path, nmsg, html, IssueSeverity.INFORMATION, msg);
|
||||
}
|
||||
|
@ -1681,4 +1643,17 @@ public class BaseValidator implements IValidationContextResourceLoader, IMessagi
|
|||
return false;
|
||||
}
|
||||
|
||||
public IValidationPolicyAdvisor getPolicyAdvisor() {
|
||||
return policyAdvisor;
|
||||
}
|
||||
|
||||
public void setPolicyAdvisor(IValidationPolicyAdvisor advisor) {
|
||||
if (advisor == null) {
|
||||
throw new Error("Cannot set advisor to null");
|
||||
}
|
||||
this.policyAdvisor = advisor;
|
||||
}
|
||||
|
||||
|
||||
|
||||
}
|
|
@ -1288,4 +1288,14 @@ public class ValidationEngine implements IValidatorResourceFetcher, IValidationP
|
|||
definition, structure, resource, valid, msgServices, messages);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isSuppressMessageId(String path, String messageId) {
|
||||
return policyAdvisor.isSuppressMessageId(path, messageId);
|
||||
}
|
||||
|
||||
@Override
|
||||
public ReferenceValidationPolicy getReferencePolicy() {
|
||||
return ReferenceValidationPolicy.IGNORE;
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -376,6 +376,9 @@ public class ValidatorCli {
|
|||
((StandaloneTask) cliTask).executeTask(cliContext,params,tt,tts);
|
||||
}
|
||||
|
||||
if (cliContext.getAdvisorFile() != null) {
|
||||
System.out.println("Note: Some validation issues might be hidden by the advisor settings in the file "+cliContext.getAdvisorFile());
|
||||
}
|
||||
System.out.println("Done. " + tt.report()+". Max Memory = "+Utilities.describeSize(Runtime.getRuntime().maxMemory()));
|
||||
SystemExitManager.finish();
|
||||
}
|
||||
|
|
|
@ -132,6 +132,12 @@ public class CliContext {
|
|||
@JsonProperty("showTimes")
|
||||
private boolean showTimes = false;
|
||||
|
||||
@JsonProperty("showTerminologyRouting")
|
||||
private boolean showTerminologyRouting = false;
|
||||
|
||||
@JsonProperty("clearTxCache")
|
||||
private boolean clearTxCache = false;
|
||||
|
||||
@JsonProperty("locale")
|
||||
private String locale = Locale.ENGLISH.toLanguageTag();
|
||||
|
||||
|
@ -172,6 +178,9 @@ public class CliContext {
|
|||
@JsonProperty("noExperimentalContent")
|
||||
private boolean noExperimentalContent;
|
||||
|
||||
@JsonProperty("advisorFile")
|
||||
private String advisorFile;
|
||||
|
||||
@JsonProperty("baseEngine")
|
||||
public String getBaseEngine() {
|
||||
return baseEngine;
|
||||
|
@ -754,6 +763,22 @@ public class CliContext {
|
|||
this.showTimes = showTimes;
|
||||
}
|
||||
|
||||
public boolean isShowTerminologyRouting() {
|
||||
return showTerminologyRouting;
|
||||
}
|
||||
|
||||
public void setShowTerminologyRouting(boolean showTerminologyRouting) {
|
||||
this.showTerminologyRouting = showTerminologyRouting;
|
||||
}
|
||||
|
||||
public boolean isClearTxCache() {
|
||||
return clearTxCache;
|
||||
}
|
||||
|
||||
public void setClearTxCache(boolean clearTxCache) {
|
||||
this.clearTxCache = clearTxCache;
|
||||
}
|
||||
|
||||
public String getOutputStyle() {
|
||||
return outputStyle;
|
||||
}
|
||||
|
@ -852,6 +877,7 @@ public class CliContext {
|
|||
Objects.equals(watchScanDelay, that.watchScanDelay) &&
|
||||
Objects.equals(unknownCodeSystemsCauseErrors, that.unknownCodeSystemsCauseErrors) &&
|
||||
Objects.equals(noExperimentalContent, that.noExperimentalContent) &&
|
||||
Objects.equals(advisorFile, that.advisorFile) &&
|
||||
Objects.equals(watchSettleTime, that.watchSettleTime) ;
|
||||
}
|
||||
|
||||
|
@ -860,7 +886,7 @@ public class CliContext {
|
|||
return Objects.hash(baseEngine, doNative, extensions, hintAboutNonMustSupport, recursive, doDebug, assumeValidRestReferences, canDoNative, noInternalCaching,
|
||||
noExtensibleBindingMessages, noInvariants, displayWarnings, wantInvariantsInMessages, map, output, outputSuffix, htmlOutput, txServer, sv, txLog, txCache, mapLog, lang, srcLang, tgtLang, fhirpath, snomedCT,
|
||||
targetVer, igs, questionnaireMode, level, profiles, sources, inputs, mode, locale, locations, crumbTrails, showMessageIds, forPublication, showTimes, allowExampleUrls, outputStyle, jurisdiction, noUnicodeBiDiControlChars,
|
||||
watchMode, watchScanDelay, watchSettleTime, bestPracticeLevel, unknownCodeSystemsCauseErrors, noExperimentalContent, htmlInMarkdownCheck, allowDoubleQuotesInFHIRPath, checkIPSCodes);
|
||||
watchMode, watchScanDelay, watchSettleTime, bestPracticeLevel, unknownCodeSystemsCauseErrors, noExperimentalContent, advisorFile, htmlInMarkdownCheck, allowDoubleQuotesInFHIRPath, checkIPSCodes);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -922,6 +948,7 @@ public class CliContext {
|
|||
", watchScanDelay=" + watchScanDelay +
|
||||
", unknownCodeSystemsCauseErrors=" + unknownCodeSystemsCauseErrors +
|
||||
", noExperimentalContent=" + noExperimentalContent +
|
||||
", advisorFile=" + advisorFile +
|
||||
'}';
|
||||
}
|
||||
|
||||
|
@ -1002,5 +1029,15 @@ public class CliContext {
|
|||
this.noExperimentalContent = noExperimentalContent;
|
||||
}
|
||||
|
||||
@JsonProperty("advisorFile")
|
||||
public String getAdvisorFile() {
|
||||
return advisorFile;
|
||||
}
|
||||
|
||||
@JsonProperty("advisorFile")
|
||||
public void setAdvisorFile(String advisorFile) {
|
||||
this.advisorFile = advisorFile;
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -17,12 +17,14 @@ import org.hl7.fhir.exceptions.FHIRException;
|
|||
import org.hl7.fhir.r5.context.IWorkerContext;
|
||||
import org.hl7.fhir.r5.context.IWorkerContextManager;
|
||||
import org.hl7.fhir.r5.elementmodel.Element;
|
||||
import org.hl7.fhir.r5.elementmodel.Element.SpecialElement;
|
||||
import org.hl7.fhir.r5.model.CanonicalResource;
|
||||
import org.hl7.fhir.r5.model.ElementDefinition;
|
||||
import org.hl7.fhir.r5.model.Resource;
|
||||
import org.hl7.fhir.r5.model.StructureDefinition;
|
||||
import org.hl7.fhir.r5.model.ValueSet;
|
||||
import org.hl7.fhir.r5.terminologies.client.ITerminologyClient;
|
||||
import org.hl7.fhir.r5.utils.validation.IMessagingServices;
|
||||
import org.hl7.fhir.r5.utils.validation.IResourceValidator;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidatorResourceFetcher;
|
||||
|
@ -41,13 +43,14 @@ import org.hl7.fhir.utilities.json.model.JsonObject;
|
|||
import org.hl7.fhir.utilities.json.parser.JsonParser;
|
||||
import org.hl7.fhir.utilities.npm.FilesystemPackageCacheManager;
|
||||
import org.hl7.fhir.utilities.npm.NpmPackage;
|
||||
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
||||
import org.hl7.fhir.validation.cli.utils.Common;
|
||||
import org.hl7.fhir.validation.instance.advisor.BasePolicyAdvisorForFullValidation;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
|
||||
|
||||
public class StandAloneValidatorFetcher extends BasePolicyAdvisorForFullValidation implements IValidatorResourceFetcher, IValidationPolicyAdvisor, IWorkerContextManager.ICanonicalResourceLocator {
|
||||
public class StandAloneValidatorFetcher implements IValidatorResourceFetcher, IValidationPolicyAdvisor, IWorkerContextManager.ICanonicalResourceLocator {
|
||||
|
||||
List<String> mappingsUris = new ArrayList<>();
|
||||
private FilesystemPackageCacheManager pcm;
|
||||
|
@ -56,12 +59,13 @@ public class StandAloneValidatorFetcher extends BasePolicyAdvisorForFullValidati
|
|||
private Map<String, Boolean> urlList = new HashMap<>();
|
||||
private Map<String, String> pidList = new HashMap<>();
|
||||
private Map<String, NpmPackage> pidMap = new HashMap<>();
|
||||
|
||||
private IValidationPolicyAdvisor policyAdvisor;
|
||||
|
||||
public StandAloneValidatorFetcher(FilesystemPackageCacheManager pcm, IWorkerContext context, IPackageInstaller installer) {
|
||||
super(ReferenceValidationPolicy.IGNORE);
|
||||
this.pcm = pcm;
|
||||
this.context = context;
|
||||
this.installer = installer;
|
||||
this.policyAdvisor = new BasePolicyAdvisorForFullValidation(ReferenceValidationPolicy.IGNORE);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -297,4 +301,55 @@ public class StandAloneValidatorFetcher extends BasePolicyAdvisorForFullValidati
|
|||
return new HashSet<>();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isSuppressMessageId(String path, String messageId) {
|
||||
return policyAdvisor.isSuppressMessageId(path, messageId);
|
||||
}
|
||||
|
||||
@Override
|
||||
public ContainedReferenceValidationPolicy policyForContained(IResourceValidator validator, Object appContext,
|
||||
StructureDefinition structure, ElementDefinition element, String containerType, String containerId,
|
||||
SpecialElement containingResourceType, String path, String url) {
|
||||
return policyAdvisor.policyForContained(validator, appContext, structure, element, containerType, containerId, containingResourceType, path, url);
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<ResourceValidationAction> policyForResource(IResourceValidator validator, Object appContext,
|
||||
StructureDefinition type, String path) {
|
||||
return policyAdvisor.policyForResource(validator, appContext, type, path);
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<ElementValidationAction> policyForElement(IResourceValidator validator, Object appContext,
|
||||
StructureDefinition structure, ElementDefinition element, String path) {
|
||||
return policyAdvisor.policyForElement(validator, appContext, structure, element, path);
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<CodedContentValidationAction> policyForCodedContent(IResourceValidator validator, Object appContext,
|
||||
String stackPath, ElementDefinition definition, StructureDefinition structure, BindingKind kind,
|
||||
AdditionalBindingPurpose purpose, ValueSet valueSet, List<String> systems) {
|
||||
return policyAdvisor.policyForCodedContent(validator, appContext, stackPath, definition, structure, kind, purpose, valueSet, systems);
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<StructureDefinition> getImpliedProfilesForResource(IResourceValidator validator, Object appContext,
|
||||
String stackPath, ElementDefinition definition, StructureDefinition structure, Element resource, boolean valid,
|
||||
IMessagingServices msgServices, List<ValidationMessage> messages) {
|
||||
return policyAdvisor.getImpliedProfilesForResource(validator, appContext, stackPath, definition, structure, resource, valid, msgServices, messages);
|
||||
}
|
||||
|
||||
@Override
|
||||
public ReferenceValidationPolicy getReferencePolicy() {
|
||||
return policyAdvisor.getReferencePolicy();
|
||||
}
|
||||
|
||||
public IValidationPolicyAdvisor getPolicyAdvisor() {
|
||||
return policyAdvisor;
|
||||
}
|
||||
|
||||
public void setPolicyAdvisor(IValidationPolicyAdvisor policyAdvisor) {
|
||||
this.policyAdvisor = policyAdvisor;
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -40,6 +40,7 @@ import org.hl7.fhir.r5.renderers.spreadsheets.ConceptMapSpreadsheetGenerator;
|
|||
import org.hl7.fhir.r5.renderers.spreadsheets.StructureDefinitionSpreadsheetGenerator;
|
||||
import org.hl7.fhir.r5.renderers.spreadsheets.ValueSetSpreadsheetGenerator;
|
||||
import org.hl7.fhir.r5.terminologies.CodeSystemUtilities;
|
||||
import org.hl7.fhir.r5.terminologies.client.TerminologyClientManager.InternalLogEvent;
|
||||
import org.hl7.fhir.r5.terminologies.utilities.TerminologyCache;
|
||||
import org.hl7.fhir.utilities.FhirPublication;
|
||||
import org.hl7.fhir.utilities.SystemExitManager;
|
||||
|
@ -69,6 +70,8 @@ import org.hl7.fhir.validation.cli.renderers.ValidationOutputRenderer;
|
|||
import org.hl7.fhir.validation.cli.utils.Common;
|
||||
import org.hl7.fhir.validation.cli.utils.EngineMode;
|
||||
import org.hl7.fhir.validation.cli.utils.VersionSourceInformation;
|
||||
import org.hl7.fhir.validation.instance.advisor.JsonDrivenPolicyAdvisor;
|
||||
import org.hl7.fhir.validation.instance.advisor.TextDrivenPolicyAdvisor;
|
||||
|
||||
public class ValidationService {
|
||||
|
||||
|
@ -155,6 +158,7 @@ public class ValidationService {
|
|||
if (request.getCliContext().isShowTimes()) {
|
||||
response.getValidationTimes().put(fileToValidate.getFileName(), validatedFragments.getValidationTime());
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -289,6 +293,19 @@ public class ValidationService {
|
|||
TextFile.stringToFile(html, cliContext.getHtmlOutput());
|
||||
System.out.println("HTML Summary in " + cliContext.getHtmlOutput());
|
||||
}
|
||||
|
||||
if (cliContext.isShowTerminologyRouting()) {
|
||||
System.out.println("");
|
||||
System.out.println("Terminology Routing Dump ---------------------------------------");
|
||||
if (validator.getContext().getTxClientManager().getInternalLog().isEmpty()) {
|
||||
System.out.println("(nothing happened)");
|
||||
} else {
|
||||
for (InternalLogEvent log : validator.getContext().getTxClientManager().getInternalLog()) {
|
||||
System.out.println(log.getMessage()+" -> "+log.getServer()+" (for VS "+log.getVs()+" with systems '"+log.getSystems()+"', choices = '"+log.getChoices()+"')");
|
||||
}
|
||||
}
|
||||
validator.getContext().getTxClientManager().getInternalLog().clear();
|
||||
}
|
||||
}
|
||||
if (watch != ValidatorWatchMode.NONE) {
|
||||
if (statusNeeded) {
|
||||
|
@ -545,6 +562,10 @@ public class ValidationService {
|
|||
System.out.println(" No Terminology Cache");
|
||||
} else {
|
||||
System.out.println(" Terminology Cache at "+validationEngine.getContext().getTxCache().getFolder());
|
||||
if (cliContext.isClearTxCache()) {
|
||||
System.out.println(" Terminology Cache Entries Cleaned out");
|
||||
validationEngine.getContext().getTxCache().clear();
|
||||
}
|
||||
}
|
||||
System.out.print(" Get set... ");
|
||||
validationEngine.setQuestionnaireMode(cliContext.getQuestionnaireMode());
|
||||
|
@ -584,6 +605,13 @@ public class ValidationService {
|
|||
validationEngine.setFetcher(fetcher);
|
||||
validationEngine.getContext().setLocator(fetcher);
|
||||
validationEngine.setPolicyAdvisor(fetcher);
|
||||
if (cliContext.getAdvisorFile() != null) {
|
||||
if (cliContext.getAdvisorFile().endsWith(".json")) {
|
||||
fetcher.setPolicyAdvisor(new JsonDrivenPolicyAdvisor(fetcher.getPolicyAdvisor(), new File(cliContext.getAdvisorFile())));
|
||||
} else {
|
||||
fetcher.setPolicyAdvisor(new TextDrivenPolicyAdvisor(fetcher.getPolicyAdvisor(), new File(cliContext.getAdvisorFile())));
|
||||
}
|
||||
}
|
||||
}
|
||||
validationEngine.getBundleValidationRules().addAll(cliContext.getBundleValidationRules());
|
||||
validationEngine.setJurisdiction(CodeSystemUtilities.readCoding(cliContext.getJurisdiction()));
|
||||
|
|
|
@ -54,6 +54,8 @@ public class Params {
|
|||
public static final String TERMINOLOGY = "-tx";
|
||||
public static final String TERMINOLOGY_LOG = "-txLog";
|
||||
public static final String TERMINOLOGY_CACHE = "-txCache";
|
||||
public static final String TERMINOLOGY_ROUTING = "-tx-routing";
|
||||
public static final String TERMINOLOGY_CACHE_CLEAR = "-clear-tx-cache";
|
||||
public static final String LOG = "-log";
|
||||
public static final String LANGUAGE = "-language";
|
||||
public static final String IMPLEMENTATION_GUIDE = "-ig";
|
||||
|
@ -86,6 +88,7 @@ public class Params {
|
|||
public static final String SHOW_TIMES = "-show-times";
|
||||
public static final String ALLOW_EXAMPLE_URLS = "-allow-example-urls";
|
||||
public static final String OUTPUT_STYLE = "-output-style";
|
||||
public static final String ADVSIOR_FILE = "-advisor-file";
|
||||
public static final String DO_IMPLICIT_FHIRPATH_STRING_CONVERSION = "-implicit-fhirpath-string-conversions";
|
||||
public static final String JURISDICTION = "-jurisdiction";
|
||||
public static final String HTML_IN_MARKDOWN = "-html-in-markdown";
|
||||
|
@ -339,10 +342,22 @@ public class Params {
|
|||
} else {
|
||||
throw new Error("Value for "+ALLOW_EXAMPLE_URLS+" not understood: "+bl);
|
||||
}
|
||||
} else if (args[i].equals(TERMINOLOGY_ROUTING)) {
|
||||
cliContext.setShowTerminologyRouting(true);
|
||||
} else if (args[i].equals(TERMINOLOGY_CACHE_CLEAR)) {
|
||||
cliContext.setClearTxCache(true);
|
||||
} else if (args[i].equals(SHOW_TIMES)) {
|
||||
cliContext.setShowTimes(true);
|
||||
} else if (args[i].equals(OUTPUT_STYLE)) {
|
||||
cliContext.setOutputStyle(args[++i]);
|
||||
} else if (args[i].equals(ADVSIOR_FILE)) {
|
||||
cliContext.setAdvisorFile(args[++i]);
|
||||
File f = new File(cliContext.getAdvisorFile());
|
||||
if (!f.exists()) {
|
||||
throw new Error("Cannot find advisor file "+cliContext.getAdvisorFile());
|
||||
} else if (!Utilities.existsInList(Utilities.getFileExtension(f.getName()), "json", "txt")) {
|
||||
throw new Error("Advisor file "+cliContext.getAdvisorFile()+" must be a .json or a .txt file");
|
||||
}
|
||||
} else if (args[i].equals(SCAN)) {
|
||||
cliContext.setMode(EngineMode.SCAN);
|
||||
} else if (args[i].equals(TERMINOLOGY)) {
|
||||
|
|
|
@ -574,7 +574,6 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
|
|||
private boolean noBindingMsgSuppressed;
|
||||
private Map<String, Element> fetchCache = new HashMap<>();
|
||||
private HashMap<Element, ResourceValidationTracker> resourceTracker = new HashMap<>();
|
||||
private IValidationPolicyAdvisor policyAdvisor = new BasePolicyAdvisorForFullValidation(ReferenceValidationPolicy.CHECK_VALID);
|
||||
long time = 0;
|
||||
long start = 0;
|
||||
long lastlog = 0;
|
||||
|
@ -664,21 +663,6 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
|
|||
return this;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public IValidationPolicyAdvisor getPolicyAdvisor() {
|
||||
return policyAdvisor;
|
||||
}
|
||||
|
||||
@Override
|
||||
public IResourceValidator setPolicyAdvisor(IValidationPolicyAdvisor advisor) {
|
||||
if (advisor == null) {
|
||||
throw new Error("Cannot set advisor to null");
|
||||
}
|
||||
this.policyAdvisor = advisor;
|
||||
return this;
|
||||
}
|
||||
|
||||
public IValidationProfileUsageTracker getTracker() {
|
||||
return this.tracker;
|
||||
}
|
||||
|
|
|
@ -13,10 +13,6 @@ import org.hl7.fhir.r5.model.ValueSet;
|
|||
import org.hl7.fhir.r5.utils.validation.IMessagingServices;
|
||||
import org.hl7.fhir.r5.utils.validation.IResourceValidator;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor.AdditionalBindingPurpose;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor.CodedContentValidationAction;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor.ElementValidationAction;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor.ResourceValidationAction;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.BindingKind;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ContainedReferenceValidationPolicy;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ReferenceValidationPolicy;
|
||||
|
@ -169,5 +165,15 @@ public class BasePolicyAdvisorForFullValidation implements IValidationPolicyAdvi
|
|||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isSuppressMessageId(String path, String messageId) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ReferenceValidationPolicy getReferencePolicy() {
|
||||
return refpol;
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
|
|
@ -1,77 +1,46 @@
|
|||
package org.hl7.fhir.validation.instance.advisor;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.IOException;
|
||||
import java.util.EnumSet;
|
||||
import java.util.List;
|
||||
|
||||
import org.hl7.fhir.r5.elementmodel.Element;
|
||||
import org.hl7.fhir.r5.elementmodel.Element.SpecialElement;
|
||||
import org.hl7.fhir.r5.model.ElementDefinition;
|
||||
import org.hl7.fhir.r5.model.StructureDefinition;
|
||||
import org.hl7.fhir.r5.model.ValueSet;
|
||||
import org.hl7.fhir.r5.utils.validation.IMessagingServices;
|
||||
import org.hl7.fhir.r5.utils.validation.IResourceValidator;
|
||||
import javax.annotation.Nonnull;
|
||||
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.BindingKind;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ContainedReferenceValidationPolicy;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ReferenceValidationPolicy;
|
||||
import org.hl7.fhir.utilities.json.JsonException;
|
||||
import org.hl7.fhir.utilities.json.model.JsonElement;
|
||||
import org.hl7.fhir.utilities.json.model.JsonObject;
|
||||
import org.hl7.fhir.utilities.json.parser.JsonParser;
|
||||
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
||||
|
||||
public class JsonDrivenPolicyAdvisor implements IValidationPolicyAdvisor {
|
||||
public class JsonDrivenPolicyAdvisor extends RulesDrivenPolicyAdvisor {
|
||||
|
||||
private JsonObject config;
|
||||
private IValidationPolicyAdvisor base;
|
||||
|
||||
public JsonDrivenPolicyAdvisor(IValidationPolicyAdvisor base, JsonObject config) {
|
||||
this.base = base;
|
||||
this.config = config;
|
||||
public JsonDrivenPolicyAdvisor(IValidationPolicyAdvisor base, File source) throws JsonException, IOException {
|
||||
super(base);
|
||||
load(source);
|
||||
}
|
||||
|
||||
public JsonDrivenPolicyAdvisor(String config) throws JsonException, IOException {
|
||||
this.config = JsonParser.parseObject(config, true);
|
||||
public JsonDrivenPolicyAdvisor(ReferenceValidationPolicy refpol, File source) throws JsonException, IOException {
|
||||
super(refpol);
|
||||
load(source);
|
||||
}
|
||||
|
||||
public JsonDrivenPolicyAdvisor(File config) throws JsonException, IOException {
|
||||
this.config = JsonParser.parseObject(new FileInputStream(config), true);
|
||||
}
|
||||
|
||||
public JsonDrivenPolicyAdvisor(FileInputStream config) throws JsonException, IOException {
|
||||
this.config = JsonParser.parseObject(config, true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public ReferenceValidationPolicy policyForReference(IResourceValidator validator, Object appContext, String path, String url) {
|
||||
return base.policyForReference(validator, appContext, path, url);
|
||||
}
|
||||
|
||||
@Override
|
||||
public ContainedReferenceValidationPolicy policyForContained(IResourceValidator validator, Object appContext, StructureDefinition structure, ElementDefinition element, String containerType, String containerId, SpecialElement containingResourceType, String path, String url) {
|
||||
return base.policyForContained(validator, appContext, structure, element, containerType, containerId, containingResourceType, path, url);
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<ResourceValidationAction> policyForResource(IResourceValidator validator, Object appContext, StructureDefinition type, String path) {
|
||||
return base.policyForResource(validator, appContext, type, path);
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<ElementValidationAction> policyForElement(IResourceValidator validator, Object appContext, StructureDefinition structure, ElementDefinition element, String path) {
|
||||
return base.policyForElement(validator, appContext, structure, element, path);
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<CodedContentValidationAction> policyForCodedContent(IResourceValidator validator, Object appContext, String stackPath, ElementDefinition definition, StructureDefinition structure, BindingKind kind, AdditionalBindingPurpose purpose, ValueSet valueSet, List<String> systems) {
|
||||
return base.policyForCodedContent(validator, appContext, stackPath, definition, structure, kind, purpose, valueSet, systems);
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<StructureDefinition> getImpliedProfilesForResource(IResourceValidator validator, Object appContext, String stackPath, ElementDefinition definition, StructureDefinition structure, Element resource, boolean valid, IMessagingServices msgServices, List<ValidationMessage> messages) {
|
||||
return base.getImpliedProfilesForResource(validator, appContext, stackPath, definition, structure, resource, valid, msgServices, messages);
|
||||
private void load(File source) throws JsonException, IOException {
|
||||
JsonObject json = JsonParser.parseObject(source);
|
||||
for (JsonElement e : json.forceArray("suppress").getItems()) {
|
||||
@Nonnull String s = e.asString();
|
||||
if (s.contains("@")) {
|
||||
String id = s.substring(0, s.indexOf("@"));
|
||||
String path = s.substring(s.indexOf("@")+1);
|
||||
boolean regex = false;
|
||||
if (path.startsWith("^")) {
|
||||
regex = true;
|
||||
path = path.substring(1);
|
||||
}
|
||||
addSuppressMessageRule(id, path, regex);
|
||||
} else {
|
||||
addSuppressMessageRule(s);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -0,0 +1,230 @@
|
|||
package org.hl7.fhir.validation.instance.advisor;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.EnumSet;
|
||||
import java.util.List;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
|
||||
import org.hl7.fhir.r5.elementmodel.Element;
|
||||
import org.hl7.fhir.r5.model.ElementDefinition;
|
||||
import org.hl7.fhir.r5.model.StructureDefinition;
|
||||
import org.hl7.fhir.r5.model.ValueSet;
|
||||
import org.hl7.fhir.r5.utils.validation.IMessagingServices;
|
||||
import org.hl7.fhir.r5.utils.validation.IResourceValidator;
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.BindingKind;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ContainedReferenceValidationPolicy;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ReferenceValidationPolicy;
|
||||
import org.hl7.fhir.utilities.validation.ValidationMessage;
|
||||
|
||||
public class RulesDrivenPolicyAdvisor extends BasePolicyAdvisorForFullValidation {
|
||||
|
||||
private IValidationPolicyAdvisor base;
|
||||
|
||||
public RulesDrivenPolicyAdvisor(ReferenceValidationPolicy refpol) {
|
||||
super(refpol);
|
||||
base = null;
|
||||
}
|
||||
|
||||
public RulesDrivenPolicyAdvisor(IValidationPolicyAdvisor base) {
|
||||
super(base.getReferencePolicy());
|
||||
this.base = base;
|
||||
}
|
||||
|
||||
private class SuppressMessageRule {
|
||||
private String id;
|
||||
private String path;
|
||||
private String[] pathSegments;
|
||||
private boolean regex;
|
||||
|
||||
protected SuppressMessageRule(@Nonnull String id, String path, boolean regex) {
|
||||
super();
|
||||
this.id = id;
|
||||
this.path = path;
|
||||
this.pathSegments = path.split("\\.");
|
||||
this.regex = regex;
|
||||
}
|
||||
|
||||
protected SuppressMessageRule(@Nonnull String id) {
|
||||
super();
|
||||
this.id = id;
|
||||
}
|
||||
|
||||
public boolean matches(@Nonnull String mid, @Nonnull String path, String[] p) {
|
||||
if (regex) {
|
||||
return stringMatches(id, mid) && regexMatches(path, path);
|
||||
} else {
|
||||
return stringMatches(id, mid) && pathMatches(pathSegments, p);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// string matching
|
||||
|
||||
boolean pathMatches(String[] specifier, String[] actual) {
|
||||
if (specifier == null) {
|
||||
return true;
|
||||
}
|
||||
for (int i = 0; i < specifier.length; i++) {
|
||||
if (i == actual.length) {
|
||||
return false;
|
||||
} else if (!pathSegmentMatches(specifier[i], actual[i])) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (actual.length > specifier.length) {
|
||||
return specifier[specifier.length-1].equals("*");
|
||||
} else {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
boolean pathSegmentMatches(String specifier, String actual) {
|
||||
if ("*".equals(specifier)) {
|
||||
return true;
|
||||
} else if (!specifier.contains("[")) {
|
||||
if (actual.contains("[")) {
|
||||
actual = actual.substring(0, actual.indexOf("["));
|
||||
}
|
||||
return specifier.equals(actual);
|
||||
} else {
|
||||
return specifier.equals(actual);
|
||||
}
|
||||
}
|
||||
|
||||
boolean stringMatches(String specifier, @Nonnull String actual) {
|
||||
if (specifier == null) {
|
||||
return true;
|
||||
} else if (specifier.endsWith("*")) {
|
||||
return specifier.substring(0, specifier.length()-1).equalsIgnoreCase(actual.substring(0, specifier.length()-1));
|
||||
} else {
|
||||
return specifier.equalsIgnoreCase(actual);
|
||||
}
|
||||
}
|
||||
|
||||
boolean regexMatches(String specifier, @Nonnull String actual) {
|
||||
if (specifier == null) {
|
||||
return true;
|
||||
} else {
|
||||
return actual.matches(specifier);
|
||||
}
|
||||
}
|
||||
|
||||
private List<SuppressMessageRule> suppressMessageRules = new ArrayList<>();
|
||||
private int suppressed = 0;
|
||||
|
||||
protected void addSuppressMessageRule(@Nonnull String id, String path, boolean regex) {
|
||||
suppressMessageRules.add(new SuppressMessageRule(id, path, regex));
|
||||
}
|
||||
|
||||
protected void addSuppressMessageRule(@Nonnull String id) {
|
||||
suppressMessageRules.add(new SuppressMessageRule(id));
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isSuppressMessageId(String path, String messageId) {
|
||||
String[] p = path.split("\\.");
|
||||
for (SuppressMessageRule rule : suppressMessageRules) {
|
||||
if (rule.matches(messageId, path, p)) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
if (base != null) {
|
||||
return base.isSuppressMessageId(path, messageId);
|
||||
} else {
|
||||
return super.isSuppressMessageId(path, messageId);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public ReferenceValidationPolicy policyForReference(IResourceValidator validator,
|
||||
Object appContext,
|
||||
String path,
|
||||
String url) {
|
||||
if (base != null) {
|
||||
return base.policyForReference(validator, appContext, path, url);
|
||||
} else {
|
||||
return super.policyForReference(validator, appContext, path, url);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public ContainedReferenceValidationPolicy policyForContained(IResourceValidator validator,
|
||||
Object appContext,
|
||||
StructureDefinition structure,
|
||||
ElementDefinition element,
|
||||
String containerType,
|
||||
String containerId,
|
||||
Element.SpecialElement containingResourceType,
|
||||
String path,
|
||||
String url) {
|
||||
if (base != null) {
|
||||
return base.policyForContained(validator, appContext, structure, element, containerType, containerId, containingResourceType, path, url);
|
||||
} else {
|
||||
return super.policyForContained(validator, appContext, structure, element, containerType, containerId, containingResourceType, path, url);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<ResourceValidationAction> policyForResource(IResourceValidator validator,
|
||||
Object appContext,
|
||||
StructureDefinition type,
|
||||
String path) {
|
||||
if (base != null) {
|
||||
return base.policyForResource(validator, appContext, type, path);
|
||||
} else {
|
||||
return super.policyForResource(validator, appContext, type, path);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<ElementValidationAction> policyForElement(IResourceValidator validator,
|
||||
Object appContext,
|
||||
StructureDefinition structure,
|
||||
ElementDefinition element,
|
||||
String path) {
|
||||
if (base != null) {
|
||||
return base.policyForElement(validator, appContext, structure, element, path);
|
||||
} else {
|
||||
return super.policyForElement(validator, appContext, structure, element, path);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public EnumSet<CodedContentValidationAction> policyForCodedContent(IResourceValidator validator,
|
||||
Object appContext,
|
||||
String stackPath,
|
||||
ElementDefinition definition,
|
||||
StructureDefinition structure,
|
||||
BindingKind kind,
|
||||
AdditionalBindingPurpose purpose,
|
||||
ValueSet valueSet,
|
||||
List<String> systems) {
|
||||
if (base != null) {
|
||||
return base.policyForCodedContent(validator, appContext, stackPath, definition, structure, kind, purpose, valueSet, systems);
|
||||
} else {
|
||||
return super.policyForCodedContent(validator, appContext, stackPath, definition, structure, kind, purpose, valueSet, systems);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public List<StructureDefinition> getImpliedProfilesForResource(IResourceValidator validator,
|
||||
Object appContext,
|
||||
String stackPath,
|
||||
ElementDefinition definition,
|
||||
StructureDefinition structure,
|
||||
Element resource,
|
||||
boolean valid,
|
||||
IMessagingServices msgServices,
|
||||
List<ValidationMessage> messages) {
|
||||
if (base != null) {
|
||||
return base.getImpliedProfilesForResource(validator, appContext, stackPath, definition, structure, resource, valid, msgServices, messages);
|
||||
} else {
|
||||
return super.getImpliedProfilesForResource(validator, appContext, stackPath, definition, structure, resource, valid, msgServices, messages);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
}
|
|
@ -0,0 +1,61 @@
|
|||
package org.hl7.fhir.validation.instance.advisor;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.File;
|
||||
import java.io.FileReader;
|
||||
import java.io.IOException;
|
||||
|
||||
import javax.annotation.Nonnull;
|
||||
|
||||
import org.hl7.fhir.r5.utils.validation.IValidationPolicyAdvisor;
|
||||
import org.hl7.fhir.r5.utils.validation.constants.ReferenceValidationPolicy;
|
||||
import org.hl7.fhir.utilities.Utilities;
|
||||
import org.hl7.fhir.utilities.json.JsonException;
|
||||
|
||||
public class TextDrivenPolicyAdvisor extends RulesDrivenPolicyAdvisor {
|
||||
|
||||
public TextDrivenPolicyAdvisor(IValidationPolicyAdvisor base, File source) throws JsonException, IOException {
|
||||
super(base);
|
||||
load(source);
|
||||
}
|
||||
|
||||
public TextDrivenPolicyAdvisor(ReferenceValidationPolicy refpol, File source) throws JsonException, IOException {
|
||||
super(refpol);
|
||||
load(source);
|
||||
}
|
||||
|
||||
private void load(File source) throws JsonException, IOException {
|
||||
BufferedReader reader = new BufferedReader(new FileReader(source));
|
||||
String line = reader.readLine();
|
||||
while (line != null) {
|
||||
processLine(line);
|
||||
line = reader.readLine();
|
||||
}
|
||||
reader.close();
|
||||
}
|
||||
|
||||
private void processLine(String line) {
|
||||
line = line.trim();
|
||||
if (Utilities.noString(line) || line.startsWith("#")) {
|
||||
return;
|
||||
}
|
||||
if (line.startsWith("-")) {
|
||||
@Nonnull String s = line.substring(1).trim();
|
||||
if (s.contains("@")) {
|
||||
String id = s.substring(0, s.indexOf("@"));
|
||||
String path = s.substring(s.indexOf("@")+1);
|
||||
boolean regex = false;
|
||||
if (path.startsWith("^")) {
|
||||
regex = true;
|
||||
path = path.substring(1);
|
||||
}
|
||||
addSuppressMessageRule(id, path, regex);
|
||||
} else {
|
||||
addSuppressMessageRule(s);
|
||||
}
|
||||
} else {
|
||||
// ignore it for now
|
||||
}
|
||||
|
||||
}
|
||||
}
|
|
@ -16,6 +16,7 @@ import org.hl7.fhir.r5.model.Coding;
|
|||
import org.hl7.fhir.r5.model.DateType;
|
||||
import org.hl7.fhir.r5.model.IntegerType;
|
||||
import org.hl7.fhir.r5.model.Questionnaire;
|
||||
import org.hl7.fhir.r5.model.Questionnaire.QuestionnaireAnswerConstraint;
|
||||
import org.hl7.fhir.r5.model.Questionnaire.QuestionnaireItemAnswerOptionComponent;
|
||||
import org.hl7.fhir.r5.model.Questionnaire.QuestionnaireItemComponent;
|
||||
import org.hl7.fhir.r5.model.Questionnaire.QuestionnaireItemType;
|
||||
|
@ -812,7 +813,7 @@ public class QuestionnaireValidator extends BaseValidator {
|
|||
}
|
||||
|
||||
private boolean checkOption(List<ValidationMessage> errors, Element answer, NodeStack stack, QuestionnaireWithContext qSrc, QuestionnaireItemComponent qItem, String type) {
|
||||
return checkOption(errors, answer, stack, qSrc, qItem, type, false);
|
||||
return checkOption(errors, answer, stack, qSrc, qItem, type, qItem.getAnswerConstraint() == QuestionnaireAnswerConstraint.OPTIONSORSTRING);
|
||||
}
|
||||
|
||||
private boolean checkOption(List<ValidationMessage> errors, Element answer, NodeStack stack, QuestionnaireWithContext qSrc, QuestionnaireItemComponent qItem, String type, boolean openChoice) {
|
||||
|
@ -928,31 +929,29 @@ public class QuestionnaireValidator extends BaseValidator {
|
|||
Element v = answer.getNamedChild("valueString", false);
|
||||
NodeStack ns = stack.push(v, -1, null, null);
|
||||
if (qItem.getAnswerOption().size() > 0) {
|
||||
List<StringType> list = new ArrayList<StringType>();
|
||||
boolean found = false;
|
||||
boolean empty = true;
|
||||
for (QuestionnaireItemAnswerOptionComponent components : qItem.getAnswerOption()) {
|
||||
try {
|
||||
if (components.getValue() != null) {
|
||||
list.add(components.getValueStringType());
|
||||
}
|
||||
} catch (FHIRException e) {
|
||||
// If it's the wrong type, just keep going
|
||||
if (components.getValue() != null && components.hasValueStringType()) {
|
||||
empty = false;
|
||||
found = found || components.getValue().primitiveValue().equals((v.primitiveValue()));
|
||||
}
|
||||
}
|
||||
if (!openChoice) {
|
||||
if (list.isEmpty()) {
|
||||
if (empty) {
|
||||
ok = rule(errors, NO_RULE_DATE, IssueType.STRUCTURE, v.line(), v.col(), stack.getLiteralPath(), false, I18nConstants.QUESTIONNAIRE_QR_ITEM_NOOPTIONSSTRING) && ok;
|
||||
} else {
|
||||
boolean found = false;
|
||||
for (StringType item : list) {
|
||||
if (item.getValue().equals((v.primitiveValue()))) {
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!found) {
|
||||
ok = rule(errors, NO_RULE_DATE, IssueType.STRUCTURE, v.line(), v.col(), stack.getLiteralPath(), found, I18nConstants.QUESTIONNAIRE_QR_ITEM_NOSTRING, v.primitiveValue()) && ok;
|
||||
ok = rule(errors, NO_RULE_DATE, IssueType.STRUCTURE, v.line(), v.col(), stack.getLiteralPath(), found, I18nConstants.QUESTIONNAIRE_QR_ITEM_NOSTRING, v.primitiveValue()) && ok;
|
||||
}
|
||||
} else {
|
||||
found = false;
|
||||
for (QuestionnaireItemAnswerOptionComponent components : qItem.getAnswerOption()) {
|
||||
if (components.getValue() != null && components.hasValueCoding()) {
|
||||
Coding c = components.getValueCoding();
|
||||
found = found || (c.hasDisplay() && c.getDisplay().equalsIgnoreCase(v.primitiveValue())) || (c.hasCode() && c.getCode().equalsIgnoreCase(v.primitiveValue()));
|
||||
}
|
||||
}
|
||||
ok = warning(errors, NO_RULE_DATE, IssueType.STRUCTURE, v.line(), v.col(), stack.getLiteralPath(), !found, I18nConstants.QUESTIONNAIRE_QR_ITEM_STRING_IN_CODING, v.primitiveValue()) && ok;
|
||||
}
|
||||
} else {
|
||||
hint(errors, NO_RULE_DATE, IssueType.STRUCTURE, v.line(), v.col(), stack.getLiteralPath(), false, I18nConstants.QUESTIONNAIRE_QR_ITEM_STRINGNOOPTIONS);
|
||||
|
|
|
@ -34,7 +34,7 @@ Some notes on this approach:
|
|||
* the messageId is directly tied to the format of the message. For some messages, there are multiple variants of the message, each with their own id, and these all need to be suppressed
|
||||
* message ids are stable (including across languages), but sometimes an existing message is improved in some contexts, and new message ids are introduced in later versions
|
||||
* the underlying validation still happens, even when the messages are suppressed
|
||||
|
||||
* for invariants in profiles, the message id is the profile URL and the invariant id e.g. http://hl7.org/fhir/StructureDefinition/DomainResource#dom-6
|
||||
|
||||
## Controlling what validation runs
|
||||
|
||||
|
|
|
@ -920,4 +920,14 @@ public class ValidationTests implements IEvaluationContext, IValidatorResourceFe
|
|||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean isSuppressMessageId(String path, String messageId) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ReferenceValidationPolicy getReferencePolicy() {
|
||||
return ReferenceValidationPolicy.IGNORE;
|
||||
}
|
||||
}
|
4
pom.xml
4
pom.xml
|
@ -14,7 +14,7 @@
|
|||
HAPI FHIR
|
||||
-->
|
||||
<artifactId>org.hl7.fhir.core</artifactId>
|
||||
<version>6.3.31-SNAPSHOT</version>
|
||||
<version>6.3.33-SNAPSHOT</version>
|
||||
<packaging>pom</packaging>
|
||||
|
||||
<properties>
|
||||
|
@ -22,7 +22,7 @@
|
|||
<commons_io_version>2.17.0</commons_io_version>
|
||||
<guava_version>32.0.1-jre</guava_version>
|
||||
<hapi_fhir_version>6.4.1</hapi_fhir_version>
|
||||
<validator_test_case_version>1.5.25</validator_test_case_version>
|
||||
<validator_test_case_version>1.5.27</validator_test_case_version>
|
||||
<jackson_version>2.17.0</jackson_version>
|
||||
<junit_jupiter_version>5.9.2</junit_jupiter_version>
|
||||
<junit_platform_launcher_version>1.8.2</junit_platform_launcher_version>
|
||||
|
|
Loading…
Reference in New Issue