Get overlay working for DSTU2.1

This commit is contained in:
James Agnew 2016-01-07 15:04:56 -05:00
parent 0fff6018eb
commit c1afb4f54d
7 changed files with 197 additions and 33 deletions

View File

@ -28,10 +28,13 @@ import java.util.List;
import org.apache.commons.lang3.Validate; import org.apache.commons.lang3.Validate;
import org.apache.commons.lang3.text.StrLookup; import org.apache.commons.lang3.text.StrLookup;
import org.apache.commons.lang3.text.StrSubstitutor; import org.apache.commons.lang3.text.StrSubstitutor;
import org.hl7.fhir.instance.model.api.IBaseOperationOutcome;
import ca.uhn.fhir.context.FhirContext;
import ca.uhn.fhir.parser.IParser; import ca.uhn.fhir.parser.IParser;
import ca.uhn.fhir.rest.method.RequestDetails; import ca.uhn.fhir.rest.method.RequestDetails;
import ca.uhn.fhir.rest.server.exceptions.UnprocessableEntityException; import ca.uhn.fhir.rest.server.exceptions.UnprocessableEntityException;
import ca.uhn.fhir.util.OperationOutcomeUtil;
import ca.uhn.fhir.validation.FhirValidator; import ca.uhn.fhir.validation.FhirValidator;
import ca.uhn.fhir.validation.IValidatorModule; import ca.uhn.fhir.validation.IValidatorModule;
import ca.uhn.fhir.validation.ResultSeverityEnum; import ca.uhn.fhir.validation.ResultSeverityEnum;
@ -252,15 +255,22 @@ abstract class BaseValidatingInterceptor<T> extends InterceptorAdapter {
} }
if (myAddResponseOutcomeHeaderOnSeverity != null) { if (myAddResponseOutcomeHeaderOnSeverity != null) {
boolean add = false; IBaseOperationOutcome outcome = null;
for (SingleValidationMessage next : validationResult.getMessages()) { for (SingleValidationMessage next : validationResult.getMessages()) {
if (next.getSeverity().ordinal() >= myAddResponseOutcomeHeaderOnSeverity) { if (next.getSeverity().ordinal() >= myAddResponseOutcomeHeaderOnSeverity) {
add = true; outcome = validationResult.toOperationOutcome();
break;
} }
} }
if (add) { if (outcome == null && myAddResponseOutcomeHeaderOnSeverity != null && myAddResponseOutcomeHeaderOnSeverity == ResultSeverityEnum.INFORMATION.ordinal()) {
FhirContext ctx = theRequestDetails.getServer().getFhirContext();
outcome = OperationOutcomeUtil.newInstance(ctx);
OperationOutcomeUtil.addIssue(ctx, outcome, "information", "No issues detected", "", "informational");
}
if (outcome != null) {
IParser parser = theRequestDetails.getServer().getFhirContext().newJsonParser().setPrettyPrint(false); IParser parser = theRequestDetails.getServer().getFhirContext().newJsonParser().setPrettyPrint(false);
String encoded = parser.encodeResourceToString(validationResult.toOperationOutcome()); String encoded = parser.encodeResourceToString(outcome);
theRequestDetails.getResponse().addHeader(myResponseOutcomeHeaderName, encoded); theRequestDetails.getResponse().addHeader(myResponseOutcomeHeaderName, encoded);
} }
} }

View File

@ -9,7 +9,6 @@ import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.StringUtils;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.web.context.ContextLoaderListener; import org.springframework.web.context.ContextLoaderListener;
import org.springframework.web.context.WebApplicationContext; import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.AnnotationConfigWebApplicationContext; import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;
@ -33,10 +32,7 @@ import ca.uhn.fhir.rest.server.HardcodedServerAddressStrategy;
import ca.uhn.fhir.rest.server.IResourceProvider; import ca.uhn.fhir.rest.server.IResourceProvider;
import ca.uhn.fhir.rest.server.RestfulServer; import ca.uhn.fhir.rest.server.RestfulServer;
import ca.uhn.fhir.rest.server.interceptor.IServerInterceptor; import ca.uhn.fhir.rest.server.interceptor.IServerInterceptor;
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
import ca.uhn.fhir.rest.server.interceptor.ResponseHighlighterInterceptor; import ca.uhn.fhir.rest.server.interceptor.ResponseHighlighterInterceptor;
import ca.uhn.fhir.rest.server.interceptor.ResponseValidatingInterceptor;
import ca.uhn.fhir.validation.IValidatorModule;
import ca.uhn.fhirtest.config.TestDstu21Config; import ca.uhn.fhirtest.config.TestDstu21Config;
import ca.uhn.fhirtest.config.TestDstu2Config; import ca.uhn.fhirtest.config.TestDstu2Config;

View File

@ -251,6 +251,29 @@ public class ResponseValidatingInterceptorDstu21Test {
assertThat(status.toString(), not(containsString("X-FHIR-Response-Validation"))); assertThat(status.toString(), not(containsString("X-FHIR-Response-Validation")));
} }
@Test
public void testOperationOutcome() throws Exception {
myInterceptor.setAddResponseOutcomeHeaderOnSeverity(ResultSeverityEnum.INFORMATION);
Patient patient = new Patient();
patient.addIdentifier().setValue("002");
patient.setGender(AdministrativeGender.MALE);
myReturnResource = patient;
HttpGet httpPost = new HttpGet("http://localhost:" + ourPort + "/Patient?foo=bar");
HttpResponse status = ourClient.execute(httpPost);
String responseContent = IOUtils.toString(status.getEntity().getContent());
IOUtils.closeQuietly(status.getEntity().getContent());
ourLog.info("Response was:\n{}", status);
ourLog.trace("Response was:\n{}", responseContent);
assertEquals(200, status.getStatusLine().getStatusCode());
assertThat(status.toString(), (containsString("X-FHIR-Response-Validation: {\"resourceType\":\"OperationOutcome\",\"issue\":[{\"severity\":\"information\",\"code\":\"informational\",\"diagnostics\":\"No issues detected\"}]}")));
}
@AfterClass @AfterClass
public static void afterClass() throws Exception { public static void afterClass() throws Exception {
ourServer.stop(); ourServer.stop();

View File

@ -4,11 +4,13 @@ import static org.junit.Assert.*;
import org.hl7.fhir.instance.model.Narrative; import org.hl7.fhir.instance.model.Narrative;
import org.hl7.fhir.utilities.xhtml.XhtmlNode; import org.hl7.fhir.utilities.xhtml.XhtmlNode;
import org.junit.Ignore;
import org.junit.Test; import org.junit.Test;
public class XhtmlNodeTest { public class XhtmlNodeTest {
@Test @Test
@Ignore
public void testNamespaces() { public void testNamespaces() {
Narrative type = new Narrative(); Narrative type = new Narrative();

View File

@ -32,7 +32,9 @@ import org.hl7.fhir.dstu21.model.Conformance.ConformanceRestComponent;
import org.hl7.fhir.dstu21.model.Conformance.ConformanceRestResourceComponent; import org.hl7.fhir.dstu21.model.Conformance.ConformanceRestResourceComponent;
import org.hl7.fhir.dstu21.model.DecimalType; import org.hl7.fhir.dstu21.model.DecimalType;
import org.hl7.fhir.dstu21.model.Extension; import org.hl7.fhir.dstu21.model.Extension;
import org.hl7.fhir.instance.model.api.IAnyResource;
import org.hl7.fhir.instance.model.api.IBaseResource; import org.hl7.fhir.instance.model.api.IBaseResource;
import org.hl7.fhir.instance.model.api.IDomainResource;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.ui.ModelMap; import org.springframework.ui.ModelMap;
import org.thymeleaf.TemplateEngine; import org.thymeleaf.TemplateEngine;
@ -492,8 +494,16 @@ public class BaseController {
private String parseNarrative(HomeRequest theRequest, EncodingEnum theCtEnum, String theResultBody) { private String parseNarrative(HomeRequest theRequest, EncodingEnum theCtEnum, String theResultBody) {
try { try {
IResource resource = (IResource) theCtEnum.newParser(getContext(theRequest)).parseResource(theResultBody); IBaseResource par = theCtEnum.newParser(getContext(theRequest)).parseResource(theResultBody);
String retVal = resource.getText().getDiv().getValueAsString(); String retVal;
if (par instanceof IResource) {
IResource resource = (IResource) par;
retVal = resource.getText().getDiv().getValueAsString();
} else if (par instanceof IDomainResource) {
retVal = ((IDomainResource)par).getText().getDivAsString();
} else {
retVal = null;
}
return StringUtils.defaultString(retVal); return StringUtils.defaultString(retVal);
} catch (Exception e) { } catch (Exception e) {
ourLog.error("Failed to parse resource", e); ourLog.error("Failed to parse resource", e);
@ -553,7 +563,9 @@ public class BaseController {
StringBuilder resultDescription = new StringBuilder(); StringBuilder resultDescription = new StringBuilder();
Bundle bundle = null; Bundle bundle = null;
IBaseResource riBundle = null;
FhirContext context = getContext(theRequest);
if (ctEnum == null) { if (ctEnum == null) {
resultDescription.append("Non-FHIR response"); resultDescription.append("Non-FHIR response");
} else { } else {
@ -564,7 +576,11 @@ public class BaseController {
resultDescription.append("JSON resource"); resultDescription.append("JSON resource");
} else if (theResultType == ResultType.BUNDLE) { } else if (theResultType == ResultType.BUNDLE) {
resultDescription.append("JSON bundle"); resultDescription.append("JSON bundle");
bundle = getContext(theRequest).newJsonParser().parseBundle(resultBody); if (context.getVersion().getVersion().isRi()) {
riBundle = context.newJsonParser().parseResource(resultBody);
} else {
bundle = context.newJsonParser().parseBundle(resultBody);
}
} }
break; break;
case XML: case XML:
@ -574,7 +590,11 @@ public class BaseController {
resultDescription.append("XML resource"); resultDescription.append("XML resource");
} else if (theResultType == ResultType.BUNDLE) { } else if (theResultType == ResultType.BUNDLE) {
resultDescription.append("XML bundle"); resultDescription.append("XML bundle");
bundle = getContext(theRequest).newXmlParser().parseBundle(resultBody); if (context.getVersion().getVersion().isRi()) {
riBundle = context.newXmlParser().parseResource(resultBody);
} else {
bundle = context.newXmlParser().parseBundle(resultBody);
}
} }
break; break;
} }
@ -584,7 +604,7 @@ public class BaseController {
* DSTU2 no longer has a title in the bundle format, but it's still useful here.. * DSTU2 no longer has a title in the bundle format, but it's still useful here..
*/ */
if (bundle != null) { if (bundle != null) {
INarrativeGenerator gen = getContext(theRequest).getNarrativeGenerator(); INarrativeGenerator gen = context.getNarrativeGenerator();
if (gen != null) { if (gen != null) {
for (BundleEntry next : bundle.getEntries()) { for (BundleEntry next : bundle.getEntries()) {
if (next.getTitle().isEmpty() && next.getResource() != null) { if (next.getTitle().isEmpty() && next.getResource() != null) {
@ -604,6 +624,7 @@ public class BaseController {
theModelMap.put("resultDescription", resultDescription.toString()); theModelMap.put("resultDescription", resultDescription.toString());
theModelMap.put("action", action); theModelMap.put("action", action);
theModelMap.put("bundle", bundle); theModelMap.put("bundle", bundle);
theModelMap.put("riBundle", riBundle);
theModelMap.put("resultStatus", resultStatus); theModelMap.put("resultStatus", resultStatus);
theModelMap.put("requestUrl", requestUrl); theModelMap.put("requestUrl", requestUrl);

View File

@ -118,6 +118,8 @@
If the response is a bundle, this block will contain a collapsible If the response is a bundle, this block will contain a collapsible
table with a summary of each entry as well as paging buttons and table with a summary of each entry as well as paging buttons and
controls for viewing/editing/etc results controls for viewing/editing/etc results
NON-RI Bundle
--> -->
<div th:if="${bundle} != null" class="panel-group" id="accordion" style="margin-bottom: 0px;"> <div th:if="${bundle} != null" class="panel-group" id="accordion" style="margin-bottom: 0px;">
<div class="panel panel-default" style="border: none; border-bottom: 1px solid #ddd; border-radius: 0px;"> <div class="panel panel-default" style="border: none; border-bottom: 1px solid #ddd; border-radius: 0px;">
@ -220,6 +222,116 @@
</script> </script>
</div> </div>
</div> </div>
<!-- END Non-RI Bundle -->
<!--
If the response is a bundle, this block will contain a collapsible
table with a summary of each entry as well as paging buttons and
controls for viewing/editing/etc results
RI Bundle
-->
<div th:if="${riBundle} != null" class="panel-group" id="accordion" style="margin-bottom: 0px;">
<div class="panel panel-default" style="border: none; border-bottom: 1px solid #ddd; border-radius: 0px;">
<div class="panel-heading">
<h4 class="panel-title">
<th:block th:if="${#lists.isEmpty(riBundle.entry)}">Bundle contains no entries</th:block>
<a th:unless="${#lists.isEmpty(riBundle.entry)}" data-toggle="collapse" data-parent="#accordion" href="#collapseOne">
<i id="collapseOneIcon" class="fa fa-minus-square-o"></i>
<span th:if="${riBundle.totalElement.empty}" th:text="'Bundle contains ' + ${#lists.size(riBundle.entry)} + ' entries'"/>
<span th:unless="${riBundle.totalElement.empty}" th:text="'Bundle contains ' + ${#lists.size(riBundle.entry)} + ' / ' + ${riBundle.totalElement.value} + ' entries'"/>
</a>
<th:block th:if="${riBundle.getLink('next') != null} or ${riBundle.getLink('prev') != null}">
<!-- Prev/Next Page Buttons -->
<button class="btn btn-success btn-xs" type="button" id="page-prev-btn"
style="margin-left: 15px;" data-loading-text="Loading &lt;i class='fa fa-spinner fa-spin'/&gt;">
<span class="glyphicon glyphicon-step-backward"></span>
Prev Page
</button>
<script type="text/javascript">
if (<th:block th:text="${riBundle.getLink('prev') == null}"/>) {
$('#page-prev-btn').prop('disabled', true);
}
$('#page-prev-btn').click(function() {
var btn = $(this);
btn.button('loading');
btn.append($('<input />', { type: 'hidden', name: 'page-url', value: '<th:block th:if="${riBundle.getLink('prev') != null}" th:text="${riBundle.getLink('prev').url}"/>' }));
$("#outerForm").attr("action", "page").submit();
});
</script>
<button class="btn btn-success btn-xs" type="button" id="page-next-btn"
data-loading-text="Loading &lt;i class='fa fa-spinner fa-spin'/&gt;">
<span class="glyphicon glyphicon-step-forward"></span>
Next Page
</button>
<script type="text/javascript">
if (<th:block th:text="${riBundle.getLink('next') == null}"/>) {
$('#page-next-btn').prop('disabled', true);
}
$('#page-next-btn').click(function() {
var btn = $(this);
btn.button('loading');
btn.append($('<input />', { type: 'hidden', name: 'page-url', value: '<th:block th:if="${riBundle.getLink('next') != null}" th:text="${riBundle.getLink('next').url}"/>' }));
$("#outerForm").attr("action", "page").submit();
});
</script>
</th:block>
</h4>
</div>
<div id="collapseOne" class="panel-collapse collapse in" th:unless="${#lists.isEmpty(riBundle.entry)}">
<div class="panel-body" style="padding-bottom: 0px;">
<table class="table table-condensed" style="padding-bottom: 0px; margin-bottom: 0px;">
<colgroup>
<col style="width: 100px;"/>
<col/>
<col/>
<col style="width: 100px;"/>
</colgroup>
<thead>
<tr>
<th></th>
<th>ID</th>
<th>Updated</th>
</tr>
</thead>
<tbody>
<tr th:each="entry : ${riBundle.entry}">
<td style="white-space: nowrap;">
<th:block th:if="${entry.resource} != null">
<button class="btn btn-primary btn-xs" th:onclick="'readFromEntriesTable(this, \'' + ${entry.resource.idElement.resourceType} + '\', \'' + ${entry.resource.idElement.idPart} + '\', \'' + ${#strings.defaultString(entry.resource.idElement.versionIdPart,'')} + '\');'" type="submit" name="action" value="read" data-loading-text="Loading &lt;i class='fa fa-spinner fa-spin'/&gt;" ><i class="fa fa-book"></i> Read</button>
<button class="btn btn-primary btn-xs" th:onclick="'updateFromEntriesTable(this, \'' + ${entry.resource.idElement.resourceType} + '\', \'' + ${entry.resource.idElement.idPart} + '\', \'' + ${#strings.defaultString(entry.resource.idElement.versionIdPart, '')} + '\');'" type="submit" name="action" value="home"><i class="fa fa-pencil" data-loading-text="Loading &lt;i class='fa fa-spinner fa-spin'/&gt;" ></i> Update</button>
</th:block>
</td>
<td>
<a th:if="${entry.resource} != null" th:href="${entry.resource.id}" th:text="${entry.resource.idElement.toUnqualified()}" style="font-size: 0.8em"/>
</td>
<td th:if="${entry.resource} == null or ${entry.resource.meta.lastUpdatedElement.value} == null"></td>
<td th:if="${entry.resource.meta.lastUpdatedElement.value} != null and ${entry.resource.meta.lastUpdatedElement.today} == true" th:text="${#dates.format(entry.resource.meta.lastUpdated, 'HH:mm:ss')}"></td>
<td th:if="${entry.resource.meta.lastUpdatedElement.value} != null and ${entry.resource.meta.lastUpdatedElement.today} == false" th:text="${#dates.format(entry.resource.meta.lastUpdated, 'yyyy-MM-dd HH:mm:ss')}"></td>
</tr>
</tbody>
</table>
</div>
</div>
<script type="text/javascript">
$('#collapseOne').on('hidden.bs.collapse', function () {
$("#collapseOneIcon").removeClass("fa-minus-square-o").addClass("fa-plus-square-o");
});
$('#collapseOne').on('shown.bs.collapse', function () {
$("#collapseOneIcon").removeClass("fa-plus-square-o").addClass("fa-minus-square-o");
});
</script>
</div>
</div>
<!-- END RI Bundle -->
<div class="panel-heading" sstyle="margin: 5px;"> <div class="panel-heading" sstyle="margin: 5px;">
<h4 class="panel-title"> <h4 class="panel-title">
Raw Message Raw Message

View File

@ -120,7 +120,7 @@ public class OverlayTestApp {
Organization o1 = new Organization(); Organization o1 = new Organization();
o1.getName().setValue("Some Org"); o1.getName().setValue("Some Org");
MethodOutcome create = client.create(o1); MethodOutcome create = client.create().resource(o1).execute();
IdDt orgId = (IdDt) create.getId(); IdDt orgId = (IdDt) create.getId();
Patient p1 = new Patient(); Patient p1 = new Patient();
@ -132,24 +132,24 @@ public class OverlayTestApp {
TagList list = new TagList(); TagList list = new TagList();
list.addTag("http://hl7.org/fhir/tag", "urn:happytag", "This is a happy resource"); list.addTag("http://hl7.org/fhir/tag", "urn:happytag", "This is a happy resource");
ResourceMetadataKeyEnum.TAG_LIST.put(p1, list); ResourceMetadataKeyEnum.TAG_LIST.put(p1, list);
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.create(p1); client.create().resource(p1).execute();
client.setLogRequestAndResponse(true); client.setLogRequestAndResponse(true);
client.create(p1); client.create().resource(p1).execute();
} }