package ca.uhn.fhir.jpa.provider.dstu3; import static org.apache.commons.lang3.StringUtils.isNotBlank; import static org.hamcrest.Matchers.contains; import static org.hamcrest.Matchers.containsInAnyOrder; import static org.hamcrest.Matchers.containsInRelativeOrder; import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.empty; import static org.hamcrest.Matchers.greaterThan; import static org.hamcrest.Matchers.hasItem; import static org.hamcrest.Matchers.hasItems; import static org.hamcrest.Matchers.hasSize; import static org.hamcrest.Matchers.not; import static org.hamcrest.Matchers.startsWith; import static org.hamcrest.Matchers.stringContainsInOrder; import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertFalse; import static org.junit.Assert.assertNotEquals; import static org.junit.Assert.assertNotNull; import static org.junit.Assert.assertThat; import static org.junit.Assert.assertTrue; import static org.junit.Assert.fail; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.net.InetSocketAddress; import java.net.Socket; import java.net.SocketTimeoutException; import java.net.URLEncoder; import java.nio.charset.StandardCharsets; import java.util.ArrayList; import java.util.Arrays; import java.util.Date; import java.util.HashSet; import java.util.List; import java.util.Map; import java.util.Set; import java.util.TreeSet; import org.apache.commons.io.IOUtils; import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.Validate; import org.apache.http.NameValuePair; import org.apache.http.client.ClientProtocolException; import org.apache.http.client.entity.UrlEncodedFormEntity; import org.apache.http.client.methods.CloseableHttpResponse; import org.apache.http.client.methods.HttpDelete; import org.apache.http.client.methods.HttpGet; import org.apache.http.client.methods.HttpPatch; import org.apache.http.client.methods.HttpPost; import org.apache.http.client.methods.HttpPut; import org.apache.http.entity.ByteArrayEntity; import org.apache.http.entity.ContentType; import org.apache.http.entity.StringEntity; import org.apache.http.message.BasicNameValuePair; import org.hl7.fhir.dstu3.model.AuditEvent; import org.hl7.fhir.dstu3.model.BaseResource; import org.hl7.fhir.dstu3.model.Basic; import org.hl7.fhir.dstu3.model.Binary; import org.hl7.fhir.dstu3.model.Bundle; import org.hl7.fhir.dstu3.model.Bundle.BundleEntryComponent; import org.hl7.fhir.dstu3.model.Bundle.BundleLinkComponent; import org.hl7.fhir.dstu3.model.Bundle.BundleType; import org.hl7.fhir.dstu3.model.Bundle.HTTPVerb; import org.hl7.fhir.dstu3.model.Bundle.SearchEntryMode; import org.hl7.fhir.dstu3.model.CodeSystem; import org.hl7.fhir.dstu3.model.CodeType; import org.hl7.fhir.dstu3.model.Coding; import org.hl7.fhir.dstu3.model.Condition; import org.hl7.fhir.dstu3.model.DateTimeType; import org.hl7.fhir.dstu3.model.DateType; import org.hl7.fhir.dstu3.model.Device; import org.hl7.fhir.dstu3.model.ProcedureRequest; import org.hl7.fhir.dstu3.model.DocumentManifest; import org.hl7.fhir.dstu3.model.DocumentReference; import org.hl7.fhir.dstu3.model.Encounter; import org.hl7.fhir.dstu3.model.Encounter.EncounterLocationComponent; import org.hl7.fhir.dstu3.model.Encounter.EncounterStatus; import org.hl7.fhir.dstu3.model.Enumerations.AdministrativeGender; import org.hl7.fhir.dstu3.model.Extension; import org.hl7.fhir.dstu3.model.IdType; import org.hl7.fhir.dstu3.model.ImagingStudy; import org.hl7.fhir.dstu3.model.InstantType; import org.hl7.fhir.dstu3.model.IntegerType; import org.hl7.fhir.dstu3.model.Location; import org.hl7.fhir.dstu3.model.Medication; import org.hl7.fhir.dstu3.model.MedicationAdministration; import org.hl7.fhir.dstu3.model.MedicationRequest; import org.hl7.fhir.dstu3.model.Meta; import org.hl7.fhir.dstu3.model.Narrative.NarrativeStatus; import org.hl7.fhir.dstu3.model.Observation; import org.hl7.fhir.dstu3.model.Observation.ObservationStatus; import org.hl7.fhir.dstu3.model.OperationOutcome; import org.hl7.fhir.dstu3.model.Organization; import org.hl7.fhir.dstu3.model.Parameters; import org.hl7.fhir.dstu3.model.Patient; import org.hl7.fhir.dstu3.model.Period; import org.hl7.fhir.dstu3.model.Practitioner; import org.hl7.fhir.dstu3.model.Quantity; import org.hl7.fhir.dstu3.model.Questionnaire; import org.hl7.fhir.dstu3.model.Questionnaire.QuestionnaireItemType; import org.hl7.fhir.dstu3.model.QuestionnaireResponse; import org.hl7.fhir.dstu3.model.Reference; import org.hl7.fhir.dstu3.model.StringType; import org.hl7.fhir.dstu3.model.StructureDefinition; import org.hl7.fhir.dstu3.model.Subscription; import org.hl7.fhir.dstu3.model.Subscription.SubscriptionChannelType; import org.hl7.fhir.dstu3.model.Subscription.SubscriptionStatus; import org.hl7.fhir.dstu3.model.TemporalPrecisionEnum; import org.hl7.fhir.dstu3.model.UnsignedIntType; import org.hl7.fhir.dstu3.model.ValueSet; import org.hl7.fhir.instance.model.api.IBaseOperationOutcome; import org.hl7.fhir.instance.model.api.IBaseResource; import org.hl7.fhir.instance.model.api.IIdType; import org.junit.After; import org.junit.AfterClass; import org.junit.Before; import org.junit.Ignore; import org.junit.Test; import org.springframework.transaction.TransactionStatus; import org.springframework.transaction.support.TransactionCallback; import org.springframework.transaction.support.TransactionCallbackWithoutResult; import com.google.common.collect.Lists; import ca.uhn.fhir.jpa.dao.DaoConfig; import ca.uhn.fhir.jpa.entity.Search; import ca.uhn.fhir.model.primitive.InstantDt; import ca.uhn.fhir.model.primitive.UriDt; import ca.uhn.fhir.parser.IParser; import ca.uhn.fhir.parser.StrictErrorHandler; import ca.uhn.fhir.rest.api.MethodOutcome; import ca.uhn.fhir.rest.api.SummaryEnum; import ca.uhn.fhir.rest.client.IGenericClient; import ca.uhn.fhir.rest.gclient.StringClientParam; import ca.uhn.fhir.rest.param.DateRangeParam; import ca.uhn.fhir.rest.param.ParamPrefixEnum; import ca.uhn.fhir.rest.param.StringAndListParam; import ca.uhn.fhir.rest.param.StringOrListParam; import ca.uhn.fhir.rest.param.StringParam; import ca.uhn.fhir.rest.server.Constants; import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; import ca.uhn.fhir.rest.server.exceptions.PreconditionFailedException; import ca.uhn.fhir.rest.server.exceptions.ResourceGoneException; import ca.uhn.fhir.rest.server.exceptions.UnprocessableEntityException; import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor; import ca.uhn.fhir.util.TestUtil; import ca.uhn.fhir.util.UrlUtil; public class ResourceProviderDstu3Test extends BaseResourceProviderDstu3Test { private static final org.slf4j.Logger ourLog = org.slf4j.LoggerFactory.getLogger(ResourceProviderDstu3Test.class); @Before public void beforeDisableResultReuse() { myDaoConfig.setReuseCachedSearchResultsForMillis(null); } @Override @After public void after() throws Exception { super.after(); myDaoConfig.setAllowMultipleDelete(new DaoConfig().isAllowMultipleDelete()); myDaoConfig.setAllowExternalReferences(new DaoConfig().isAllowExternalReferences()); myDaoConfig.setReuseCachedSearchResultsForMillis(new DaoConfig().getReuseCachedSearchResultsForMillis()); } @Override public void before() throws Exception { super.before(); myFhirCtx.setParserErrorHandler(new StrictErrorHandler()); myDaoConfig.setAllowMultipleDelete(true); } private void checkParamMissing(String paramName) throws IOException, ClientProtocolException { HttpGet get = new HttpGet(ourServerBase + "/Observation?" + paramName + ":missing=false"); CloseableHttpResponse resp = ourHttpClient.execute(get); IOUtils.closeQuietly(resp.getEntity().getContent()); assertEquals(200, resp.getStatusLine().getStatusCode()); } private ArrayList<IBaseResource> genResourcesOfType(Bundle theRes, Class<? extends IBaseResource> theClass) { ArrayList<IBaseResource> retVal = new ArrayList<IBaseResource>(); for (BundleEntryComponent next : theRes.getEntry()) { if (next.getResource() != null) { if (theClass.isAssignableFrom(next.getResource().getClass())) { retVal.add(next.getResource()); } } } return retVal; } /** * See #484 */ @Test public void saveAndRetrieveBasicResource() throws IOException { String input = IOUtils.toString(getClass().getResourceAsStream("/basic-stu3.xml"), StandardCharsets.UTF_8); String respString = ourClient.transaction().withBundle(input).prettyPrint().execute(); ourLog.info(respString); Bundle bundle = myFhirCtx.newXmlParser().parseResource(Bundle.class, respString); IdType id = new IdType(bundle.getEntry().get(0).getResponse().getLocation()); Basic basic = ourClient.read().resource(Basic.class).withId(id).execute(); List<Extension> exts = basic.getExtensionsByUrl("http://localhost:1080/hapi-fhir-jpaserver-example/baseDstu2/StructureDefinition/DateID"); assertEquals(1, exts.size()); } private List<String> searchAndReturnUnqualifiedIdValues(String uri) throws IOException, ClientProtocolException { List<String> ids; HttpGet get = new HttpGet(uri); CloseableHttpResponse response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); Bundle bundle = myFhirCtx.newXmlParser().parseResource(Bundle.class, resp); ids = toUnqualifiedIdValues(bundle); } finally { IOUtils.closeQuietly(response); } return ids; } private List<String> searchAndReturnUnqualifiedVersionlessIdValues(String uri) throws IOException, ClientProtocolException { List<String> ids; HttpGet get = new HttpGet(uri); CloseableHttpResponse response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); Bundle bundle = myFhirCtx.newXmlParser().parseResource(Bundle.class, resp); ids = toUnqualifiedVersionlessIdValues(bundle); } finally { IOUtils.closeQuietly(response); } return ids; } // Y @Test public void testBundleCreate() throws Exception { IGenericClient client = ourClient; String resBody = IOUtils.toString(ResourceProviderDstu3Test.class.getResource("/document-father-dstu3.json"), StandardCharsets.UTF_8); IIdType id = client.create().resource(resBody).execute().getId(); ourLog.info("Created: {}", id); Bundle bundle = client.read().resource(Bundle.class).withId(id).execute(); ourLog.info(myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(bundle)); } @Test public void testBundleCreateWithTypeTransaction() throws Exception { IGenericClient client = ourClient; String resBody = IOUtils.toString(ResourceProviderDstu3Test.class.getResource("/document-father-dstu3.json"), StandardCharsets.UTF_8); resBody = resBody.replace("\"type\": \"document\"", "\"type\": \"transaction\""); try { client.create().resource(resBody).execute().getId(); fail(); } catch (UnprocessableEntityException e) { assertThat(e.getMessage(), containsString("Unable to store a Bundle resource on this server with a Bundle.type value of: transaction")); } } @Test public void testCodeSearch() { Subscription subs = new Subscription(); subs.setStatus(SubscriptionStatus.ACTIVE); subs.getChannel().setType(SubscriptionChannelType.WEBSOCKET); subs.setCriteria("Observation?"); IIdType id = ourClient.create().resource(subs).execute().getId().toUnqualifiedVersionless(); //@formatter:off Bundle resp = ourClient .search() .forResource(Subscription.class) .where(Subscription.TYPE.exactly().code(SubscriptionChannelType.WEBSOCKET.toCode())) .and(Subscription.STATUS.exactly().code(SubscriptionStatus.ACTIVE.toCode())) .returnBundle(Bundle.class) .execute(); //@formatter:off assertThat(toUnqualifiedVersionlessIds(resp), contains(id)); //@formatter:off resp = ourClient .search() .forResource(Subscription.class) .where(Subscription.TYPE.exactly().systemAndCode(SubscriptionChannelType.WEBSOCKET.getSystem(), SubscriptionChannelType.WEBSOCKET.toCode())) .and(Subscription.STATUS.exactly().systemAndCode(SubscriptionStatus.ACTIVE.getSystem(), SubscriptionStatus.ACTIVE.toCode())) .returnBundle(Bundle.class) .execute(); //@formatter:off assertThat(toUnqualifiedVersionlessIds(resp), contains(id)); //@formatter:off resp = ourClient .search() .forResource(Subscription.class) .where(Subscription.TYPE.exactly().systemAndCode(SubscriptionChannelType.WEBSOCKET.getSystem(), SubscriptionChannelType.WEBSOCKET.toCode())) .and(Subscription.STATUS.exactly().systemAndCode("foo", SubscriptionStatus.ACTIVE.toCode())) .returnBundle(Bundle.class) .execute(); //@formatter:off assertThat(toUnqualifiedVersionlessIds(resp), empty()); } @Test public void testCountParam() throws Exception { List<IBaseResource> resources = new ArrayList<IBaseResource>(); for (int i = 0; i < 100; i++) { Organization org = new Organization(); org.setName("rpdstu2_testCountParam_01"); resources.add(org); } ourClient.transaction().withResources(resources).prettyPrint().encodedXml().execute(); Bundle found = ourClient.search().forResource(Organization.class).where(Organization.NAME.matches().value("rpdstu2_testCountParam_01")).count(10).returnBundle(Bundle.class).execute(); assertEquals(100, found.getTotal()); assertEquals(10, found.getEntry().size()); found = ourClient.search().forResource(Organization.class).where(Organization.NAME.matches().value("rpdstu2_testCountParam_01")).count(999).returnBundle(Bundle.class).execute(); assertEquals(100, found.getTotal()); assertEquals(50, found.getEntry().size()); } /** * See #438 */ @Test public void testCreateAndUpdateBinary() throws ClientProtocolException, Exception { byte[] arr = { 1, 21, 74, 123, -44 }; Binary binary = new Binary(); binary.setContent(arr); binary.setContentType("dansk"); IIdType resource = ourClient.create().resource(binary).execute().getId(); Binary fromDB = ourClient.read().resource(Binary.class).withId(resource.toVersionless()).execute(); assertEquals("1", fromDB.getIdElement().getVersionIdPart()); arr[0] = 2; HttpPut putRequest = new HttpPut(ourServerBase + "/Binary/" + resource.getIdPart()); putRequest.setEntity(new ByteArrayEntity(arr, ContentType.parse("dansk"))); CloseableHttpResponse resp = ourHttpClient.execute(putRequest); try { assertEquals(200, resp.getStatusLine().getStatusCode()); assertEquals(resource.withVersion("2").getValue(), resp.getFirstHeader("Location").getValue()); } finally { IOUtils.closeQuietly(resp); } fromDB = ourClient.read().resource(Binary.class).withId(resource.toVersionless()).execute(); assertEquals("2", fromDB.getIdElement().getVersionIdPart()); arr[0] = 3; fromDB.setContent(arr); String encoded = myFhirCtx.newJsonParser().encodeResourceToString(fromDB); putRequest = new HttpPut(ourServerBase + "/Binary/" + resource.getIdPart()); putRequest.setEntity(new StringEntity(encoded, ContentType.parse("application/json+fhir"))); resp = ourHttpClient.execute(putRequest); try { assertEquals(200, resp.getStatusLine().getStatusCode()); assertEquals(resource.withVersion("3").getValue(), resp.getFirstHeader("Location").getValue()); } finally { IOUtils.closeQuietly(resp); } fromDB = ourClient.read().resource(Binary.class).withId(resource.toVersionless()).execute(); assertEquals("3", fromDB.getIdElement().getVersionIdPart()); // Now an update with the wrong ID in the body arr[0] = 4; binary.setId(""); encoded = myFhirCtx.newJsonParser().encodeResourceToString(binary); putRequest = new HttpPut(ourServerBase + "/Binary/" + resource.getIdPart()); putRequest.setEntity(new StringEntity(encoded, ContentType.parse("application/json+fhir"))); resp = ourHttpClient.execute(putRequest); try { assertEquals(400, resp.getStatusLine().getStatusCode()); } finally { IOUtils.closeQuietly(resp); } fromDB = ourClient.read().resource(Binary.class).withId(resource.toVersionless()).execute(); assertEquals("3", fromDB.getIdElement().getVersionIdPart()); } /** * Issue submitted by Bryn */ @Test public void testCreateBundle() throws IOException { String input = IOUtils.toString(getClass().getResourceAsStream("/bryn-bundle.json"), StandardCharsets.UTF_8); Validate.notNull(input); ourClient.create().resource(input).execute().getResource(); } @Test public void testCreateConditional() { Patient patient = new Patient(); patient.addIdentifier().setSystem("http://uhn.ca/mrns").setValue("100"); patient.addName().setFamily("Tester").addGiven("Raghad"); MethodOutcome output1 = ourClient.update().resource(patient).conditionalByUrl("Patient?identifier=http://uhn.ca/mrns|100").execute(); patient = new Patient(); patient.addIdentifier().setSystem("http://uhn.ca/mrns").setValue("100"); patient.addName().setFamily("Tester").addGiven("Raghad"); MethodOutcome output2 = ourClient.update().resource(patient).conditionalByUrl("Patient?identifier=http://uhn.ca/mrns|100").execute(); assertEquals(output1.getId().getIdPart(), output2.getId().getIdPart()); } @Test public void testCreateIncludesRequestValidatorInterceptorOutcome() throws IOException { RequestValidatingInterceptor interceptor = new RequestValidatingInterceptor(); assertTrue(interceptor.isAddValidationResultsToResponseOperationOutcome()); interceptor.setFailOnSeverity(null); ourRestServer.registerInterceptor(interceptor); try { // Missing status, which is mandatory Observation obs = new Observation(); obs.addIdentifier().setSystem("urn:foo").setValue("bar"); IBaseResource outcome = ourClient.create().resource(obs).execute().getOperationOutcome(); String encodedOo = myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(outcome); ourLog.info(encodedOo); assertThat(encodedOo, containsString("cvc-complex-type.2.4.b")); assertThat(encodedOo, containsString("Successfully created resource \\\"Observation/")); interceptor.setAddValidationResultsToResponseOperationOutcome(false); outcome = ourClient.create().resource(obs).execute().getOperationOutcome(); encodedOo = myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(outcome); ourLog.info(encodedOo); assertThat(encodedOo, not(containsString("cvc-complex-type.2.4.b"))); assertThat(encodedOo, containsString("Successfully created resource \\\"Observation/")); } finally { ourRestServer.unregisterInterceptor(interceptor); } } @Test @Ignore public void testCreateQuestionnaireResponseWithValidation() throws IOException { CodeSystem cs = new CodeSystem(); cs.setUrl("http://urn/system"); cs.addConcept().setCode("code0"); ourClient.create().resource(cs).execute(); ValueSet options = new ValueSet(); options.getCompose().addInclude().setSystem("http://urn/system"); IIdType optId = ourClient.create().resource(options).execute().getId(); Questionnaire q = new Questionnaire(); q.addItem().setLinkId("link0").setRequired(false).setType(QuestionnaireItemType.CHOICE).setOptions(new Reference(optId)); IIdType qId = ourClient.create().resource(q).execute().getId(); QuestionnaireResponse qa; // Good code qa = new QuestionnaireResponse(); qa.getQuestionnaire().setReference(qId.toUnqualifiedVersionless().getValue()); qa.addItem().setLinkId("link0").addAnswer().setValue(new Coding().setSystem("urn:system").setCode("code0")); ourClient.create().resource(qa).execute(); // Bad code qa = new QuestionnaireResponse(); qa.getQuestionnaire().setReference(qId.toUnqualifiedVersionless().getValue()); qa.addItem().setLinkId("link0").addAnswer().setValue(new Coding().setSystem("urn:system").setCode("code1")); try { ourClient.create().resource(qa).execute(); fail(); } catch (UnprocessableEntityException e) { assertThat(e.getMessage(), containsString("Question with linkId[link0]")); } } @Test public void testCreateResourceConditional() throws IOException { String methodName = "testCreateResourceConditional"; Patient pt = new Patient(); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.addHeader(Constants.HEADER_IF_NONE_EXIST, "Patient?name=" + methodName); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); IdType id; try { assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } post = new HttpPost(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); post.addHeader(Constants.HEADER_IF_NONE_EXIST, "Patient?name=" + methodName); response = ourHttpClient.execute(post); try { assertEquals(200, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertEquals(id.getValue(), newIdString); // version should match for conditional create } finally { response.close(); } } @Test public void testCreateResourceConditionalComplex() throws IOException { Patient pt = new Patient(); pt.addIdentifier().setSystem("http://general-hospital.co.uk/Identifiers").setValue("09832345234543876876"); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.addHeader(Constants.HEADER_IF_NONE_EXIST, "Patient?identifier=http://general-hospital.co.uk/Identifiers|09832345234543876876"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); IdType id; CloseableHttpResponse response = ourHttpClient.execute(post); try { assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } IdType id2; response = ourHttpClient.execute(post); try { assertEquals(200, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id2 = new IdType(newIdString); } finally { response.close(); } // //@formatter:off // IIdType id3 = ourClient // .update() // .resource(pt) // .conditionalByUrl("Patient?identifier=http://general-hospital.co.uk/Identifiers|09832345234543876876") // .execute().getId(); // //@formatter:on assertEquals(id.getValue(), id2.getValue()); } @Test public void testCreateResourceReturnsOperationOutcomeByDefault() throws IOException { String resource = "<Patient xmlns=\"http://hl7.org/fhir\"></Patient>"; HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { assertEquals(201, response.getStatusLine().getStatusCode()); String respString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(response.toString()); ourLog.info(respString); assertThat(respString, containsString("<OperationOutcome xmlns=\"http://hl7.org/fhir\">")); } finally { response.getEntity().getContent().close(); response.close(); } } @Test public void testCreateResourceWithNumericId() throws IOException { String resource = "<Patient xmlns=\"http://hl7.org/fhir\"><id value=\"2\"/></Patient>"; HttpPost post = new HttpPost(ourServerBase + "/Patient/2"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertEquals(400, response.getStatusLine().getStatusCode()); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, responseString); assertEquals("Can not create resource with ID \"2\", ID must not be supplied on a create (POST) operation (use an HTTP PUT / update operation if you wish to supply an ID)", oo.getIssue().get(0).getDiagnostics()); } finally { response.getEntity().getContent().close(); response.close(); } } @Test public void testCreateWithForcedId() throws IOException { String methodName = "testCreateWithForcedId"; Patient p = new Patient(); p.addName().setFamily(methodName); p.setId(methodName); IIdType optId = ourClient.update().resource(p).execute().getId(); assertEquals(methodName, optId.getIdPart()); assertEquals("1", optId.getVersionIdPart()); } @Test public void testDeepChaining() { Location l1 = new Location(); l1.getNameElement().setValue("testDeepChainingL1"); IIdType l1id = ourClient.create().resource(l1).execute().getId(); Location l2 = new Location(); l2.getNameElement().setValue("testDeepChainingL2"); l2.getPartOf().setReferenceElement(l1id.toVersionless().toUnqualified()); IIdType l2id = ourClient.create().resource(l2).execute().getId(); Encounter e1 = new Encounter(); e1.addIdentifier().setSystem("urn:foo").setValue("testDeepChainingE1"); e1.getStatusElement().setValue(EncounterStatus.INPROGRESS); EncounterLocationComponent location = e1.addLocation(); location.getLocation().setReferenceElement(l2id.toUnqualifiedVersionless()); location.setPeriod(new Period().setStart(new Date(), TemporalPrecisionEnum.SECOND).setEnd(new Date(), TemporalPrecisionEnum.SECOND)); IIdType e1id = ourClient.create().resource(e1).execute().getId(); //@formatter:off Bundle res = ourClient.search() .forResource(Encounter.class) .where(Encounter.IDENTIFIER.exactly().systemAndCode("urn:foo", "testDeepChainingE1")) .include(Encounter.INCLUDE_LOCATION.asRecursive()) .include(Location.INCLUDE_PARTOF.asRecursive()) .returnBundle(Bundle.class) .execute(); //@formatter:on assertEquals(3, res.getEntry().size()); assertEquals(1, genResourcesOfType(res, Encounter.class).size()); assertEquals(e1id.toUnqualifiedVersionless(), genResourcesOfType(res, Encounter.class).get(0).getIdElement().toUnqualifiedVersionless()); } @Test public void testDeleteConditionalMultiple() { String methodName = "testDeleteConditionalMultiple"; myDaoConfig.setAllowMultipleDelete(false); Patient p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().setFamily("FAM1"); IIdType id1 = myPatientDao.create(p, mySrd).getId().toUnqualifiedVersionless(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().setFamily("FAM2"); IIdType id2 = myPatientDao.create(p, mySrd).getId().toUnqualifiedVersionless(); try { //@formatter:off ourClient .delete() .resourceConditionalByType(Patient.class) .where(Patient.IDENTIFIER.exactly().code(methodName)) .execute(); //@formatter:on fail(); } catch (PreconditionFailedException e) { assertEquals("HTTP 412 Precondition Failed: Failed to DELETE resource with match URL \"Patient?identifier=testDeleteConditionalMultiple\" because this search matched 2 resources", e.getMessage()); } // Not deleted yet.. ourClient.read().resource("Patient").withId(id1).execute(); ourClient.read().resource("Patient").withId(id2).execute(); myDaoConfig.setAllowMultipleDelete(true); //@formatter:off IBaseOperationOutcome response = ourClient .delete() .resourceConditionalByType(Patient.class) .where(Patient.IDENTIFIER.exactly().code(methodName)) .execute(); //@formatter:on String encoded = myFhirCtx.newXmlParser().encodeResourceToString(response); ourLog.info(encoded); assertThat(encoded, containsString( "<issue><severity value=\"information\"/><code value=\"informational\"/><diagnostics value=\"Successfully deleted 2 resource(s) in ")); try { ourClient.read().resource("Patient").withId(id1).execute(); fail(); } catch (ResourceGoneException e) { // good } try { ourClient.read().resource("Patient").withId(id2).execute(); fail(); } catch (ResourceGoneException e) { // good } } @Test public void testDeleteConditionalNoMatches() throws Exception { String methodName = "testDeleteConditionalNoMatches"; HttpDelete delete = new HttpDelete(ourServerBase + "/Patient?identifier=" + methodName); CloseableHttpResponse resp = ourHttpClient.execute(delete); try { ourLog.info(resp.toString()); String response = IOUtils.toString(resp.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(response); assertEquals(200, resp.getStatusLine().getStatusCode()); assertThat(response, containsString( "<issue><severity value=\"warning\"/><code value=\"not-found\"/><diagnostics value=\"Unable to find resource matching URL "Patient?identifier=testDeleteConditionalNoMatches". Deletion failed.\"/></issue>")); } finally { IOUtils.closeQuietly(resp); } } @Test public void testDeleteInvalidReference() throws IOException { HttpDelete delete = new HttpDelete(ourServerBase + "/Patient"); CloseableHttpResponse response = ourHttpClient.execute(delete); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertEquals(400, response.getStatusLine().getStatusCode()); assertThat(responseString, containsString("Can not perform delete, no ID provided")); } finally { response.close(); } } /** * Test for #345 */ @Test public void testDeleteNormal() { Patient p = new Patient(); p.addName().setFamily("FAM"); IIdType id = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); ourClient.read().resource(Patient.class).withId(id).execute(); ourClient.delete().resourceById(id).execute(); try { ourClient.read().resource(Patient.class).withId(id).execute(); fail(); } catch (ResourceGoneException e) { // good } } @Test public void testDeleteResourceConditional1() throws IOException { String methodName = "testDeleteResourceConditional1"; Patient pt = new Patient(); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); IdType id; try { assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } HttpDelete delete = new HttpDelete(ourServerBase + "/Patient?name=" + methodName); response = ourHttpClient.execute(delete); try { assertEquals(200, response.getStatusLine().getStatusCode()); String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, resp); assertThat(oo.getIssueFirstRep().getDiagnostics(), startsWith("Successfully deleted 1 resource(s) in ")); } finally { response.close(); } HttpGet read = new HttpGet(ourServerBase + "/Patient/" + id.getIdPart()); response = ourHttpClient.execute(read); try { ourLog.info(response.toString()); assertEquals(Constants.STATUS_HTTP_410_GONE, response.getStatusLine().getStatusCode()); String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, resp); assertThat(oo.getIssueFirstRep().getDiagnostics(), startsWith("Resource was deleted at")); } finally { response.close(); } // Delete should now have no matches delete = new HttpDelete(ourServerBase + "/Patient?name=" + methodName); response = ourHttpClient.execute(delete); try { assertEquals(200, response.getStatusLine().getStatusCode()); String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, resp); assertThat(oo.getIssueFirstRep().getDiagnostics(), startsWith("Unable to find resource matching URL \"Patient?name=testDeleteResourceConditional1\". Deletion failed.")); } finally { response.close(); } } /** * Based on email from Rene Spronk */ @Test public void testDeleteResourceConditional2() throws IOException, Exception { String methodName = "testDeleteResourceConditional2"; Patient pt = new Patient(); pt.addName().setFamily(methodName); pt.addIdentifier().setSystem("http://ghh.org/patient").setValue(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); IdType id; try { assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } /* * Try it with a raw socket call. The Apache client won't let us use the unescaped "|" in the URL but we want to make sure that works too.. */ Socket sock = new Socket(); sock.setSoTimeout(3000); try { sock.connect(new InetSocketAddress("localhost", ourPort)); sock.getOutputStream().write(("DELETE /fhir/context/Patient?identifier=http://ghh.org/patient|" + methodName + " HTTP/1.1\n").getBytes("UTF-8")); sock.getOutputStream().write("Host: localhost\n".getBytes("UTF-8")); sock.getOutputStream().write("\n".getBytes("UTF-8")); BufferedReader socketInput = new BufferedReader(new InputStreamReader(sock.getInputStream())); // String response = ""; StringBuilder b = new StringBuilder(); char[] buf = new char[1000]; while (socketInput.read(buf) != -1) { b.append(buf); } String resp = b.toString(); ourLog.info("Resp: {}", resp); } catch (SocketTimeoutException e) { e.printStackTrace(); } finally { sock.close(); } Thread.sleep(1000); HttpGet read = new HttpGet(ourServerBase + "/Patient/" + id.getIdPart()); response = ourHttpClient.execute(read); try { ourLog.info(response.toString()); assertEquals(Constants.STATUS_HTTP_410_GONE, response.getStatusLine().getStatusCode()); } finally { response.close(); } } @Test public void testDeleteReturnsOperationOutcome() { Patient p = new Patient(); p.addName().setFamily("FAM"); IIdType id = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); IBaseOperationOutcome resp = ourClient.delete().resourceById(id).execute(); OperationOutcome oo = (OperationOutcome) resp; assertThat(oo.getIssueFirstRep().getDiagnostics(), startsWith("Successfully deleted 1 resource(s) in ")); } /** * See issue #52 */ @Test public void testDocumentManifestResources() throws Exception { myFhirCtx.getResourceDefinition(Practitioner.class); myFhirCtx.getResourceDefinition(DocumentManifest.class); IGenericClient client = ourClient; int initialSize = client.search().forResource(DocumentManifest.class).returnBundle(Bundle.class).execute().getEntry().size(); String resBody = IOUtils.toString(ResourceProviderDstu3Test.class.getResource("/documentmanifest.json"), StandardCharsets.UTF_8); client.create().resource(resBody).execute(); int newSize = client.search().forResource(DocumentManifest.class).returnBundle(Bundle.class).execute().getEntry().size(); assertEquals(1, newSize - initialSize); } /** * See issue #52 */ @Test public void testDocumentReferenceResources() throws Exception { IGenericClient client = ourClient; int initialSize = client.search().forResource(DocumentReference.class).returnBundle(Bundle.class).execute().getEntry().size(); String resBody = IOUtils.toString(ResourceProviderDstu3Test.class.getResource("/documentreference.json"), StandardCharsets.UTF_8); client.create().resource(resBody).execute(); int newSize = client.search().forResource(DocumentReference.class).returnBundle(Bundle.class).execute().getEntry().size(); assertEquals(1, newSize - initialSize); } @Test public void testEmptySearch() throws Exception { Bundle responseBundle; responseBundle = ourClient.search().forResource(Patient.class).returnBundle(Bundle.class).execute(); assertEquals(0, responseBundle.getTotal()); responseBundle = ourClient.search().forResource(Patient.class).where(Patient.NAME.matches().value("AAA")).returnBundle(Bundle.class).execute(); assertEquals(0, responseBundle.getTotal()); responseBundle = ourClient.search().forResource(Patient.class).where(new StringClientParam("_content").matches().value("AAA")).returnBundle(Bundle.class).execute(); assertEquals(0, responseBundle.getTotal()); } @Test public void testEverythingEncounterInstance() throws Exception { String methodName = "testEverythingEncounterInstance"; Organization org1parent = new Organization(); org1parent.setId("org1parent"); org1parent.setName(methodName + "1parent"); IIdType orgId1parent = ourClient.update().resource(org1parent).execute().getId().toUnqualifiedVersionless(); Organization org1 = new Organization(); org1.setName(methodName + "1"); org1.getPartOf().setReferenceElement(orgId1parent); IIdType orgId1 = ourClient.create().resource(org1).execute().getId().toUnqualifiedVersionless(); Patient p = new Patient(); p.addName().setFamily(methodName); p.getManagingOrganization().setReferenceElement(orgId1); IIdType patientId = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); Organization org2 = new Organization(); org2.setName(methodName + "1"); IIdType orgId2 = ourClient.create().resource(org2).execute().getId().toUnqualifiedVersionless(); Device dev = new Device(); dev.setModel(methodName); dev.getOwner().setReferenceElement(orgId2); IIdType devId = ourClient.create().resource(dev).execute().getId().toUnqualifiedVersionless(); Location locParent = new Location(); locParent.setName(methodName + "Parent"); IIdType locPId = ourClient.create().resource(locParent).execute().getId().toUnqualifiedVersionless(); Location locChild = new Location(); locChild.setName(methodName); locChild.getPartOf().setReferenceElement(locPId); IIdType locCId = ourClient.create().resource(locChild).execute().getId().toUnqualifiedVersionless(); Encounter encU = new Encounter(); encU.getSubject().setReferenceElement(patientId); encU.addLocation().getLocation().setReferenceElement(locCId); IIdType encUId = ourClient.create().resource(encU).execute().getId().toUnqualifiedVersionless(); Encounter enc = new Encounter(); enc.getSubject().setReferenceElement(patientId); enc.addLocation().getLocation().setReferenceElement(locCId); IIdType encId = ourClient.create().resource(enc).execute().getId().toUnqualifiedVersionless(); Observation obs = new Observation(); obs.getSubject().setReferenceElement(patientId); obs.getDevice().setReferenceElement(devId); obs.getContext().setReferenceElement(encId); IIdType obsId = ourClient.create().resource(obs).execute().getId().toUnqualifiedVersionless(); ourLog.info("IDs: EncU:" + encUId.getIdPart() + " Enc:" + encId.getIdPart() + " " + patientId.toUnqualifiedVersionless()); Parameters output = ourClient.operation().onInstance(encId).named("everything").withNoParameters(Parameters.class).execute(); Bundle b = (Bundle) output.getParameter().get(0).getResource(); List<IIdType> ids = toUnqualifiedVersionlessIds(b); assertThat(ids, containsInAnyOrder(patientId, encId, orgId1, orgId2, orgId1parent, locPId, locCId, obsId, devId)); assertThat(ids, not(containsInRelativeOrder(encUId))); ourLog.info(ids.toString()); } @Test public void testEverythingEncounterType() throws Exception { String methodName = "testEverythingEncounterInstance"; Organization org1parent = new Organization(); org1parent.setId("org1parent"); org1parent.setName(methodName + "1parent"); IIdType orgId1parent = ourClient.update().resource(org1parent).execute().getId().toUnqualifiedVersionless(); Organization org1 = new Organization(); org1.setName(methodName + "1"); org1.getPartOf().setReferenceElement(orgId1parent); IIdType orgId1 = ourClient.create().resource(org1).execute().getId().toUnqualifiedVersionless(); Patient p = new Patient(); p.addName().setFamily(methodName); p.getManagingOrganization().setReferenceElement(orgId1); IIdType patientId = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); Organization org2 = new Organization(); org2.setName(methodName + "1"); IIdType orgId2 = ourClient.create().resource(org2).execute().getId().toUnqualifiedVersionless(); Device dev = new Device(); dev.setModel(methodName); dev.getOwner().setReferenceElement(orgId2); IIdType devId = ourClient.create().resource(dev).execute().getId().toUnqualifiedVersionless(); Location locParent = new Location(); locParent.setName(methodName + "Parent"); IIdType locPId = ourClient.create().resource(locParent).execute().getId().toUnqualifiedVersionless(); Location locChild = new Location(); locChild.setName(methodName); locChild.getPartOf().setReferenceElement(locPId); IIdType locCId = ourClient.create().resource(locChild).execute().getId().toUnqualifiedVersionless(); Encounter encU = new Encounter(); encU.addIdentifier().setValue(methodName); IIdType encUId = ourClient.create().resource(encU).execute().getId().toUnqualifiedVersionless(); Encounter enc = new Encounter(); enc.getSubject().setReferenceElement(patientId); enc.addLocation().getLocation().setReferenceElement(locCId); IIdType encId = ourClient.create().resource(enc).execute().getId().toUnqualifiedVersionless(); Observation obs = new Observation(); obs.getSubject().setReferenceElement(patientId); obs.getDevice().setReferenceElement(devId); obs.getContext().setReferenceElement(encId); IIdType obsId = ourClient.create().resource(obs).execute().getId().toUnqualifiedVersionless(); Parameters output = ourClient.operation().onType(Encounter.class).named("everything").withNoParameters(Parameters.class).execute(); Bundle b = (Bundle) output.getParameter().get(0).getResource(); List<IIdType> ids = toUnqualifiedVersionlessIds(b); assertThat(ids, containsInAnyOrder(patientId, encUId, encId, orgId1, orgId2, orgId1parent, locPId, locCId, obsId, devId)); ourLog.info(ids.toString()); } @Test public void testEverythingInstanceWithContentFilter() { Patient pt1 = new Patient(); pt1.addName().setFamily("Everything").addGiven("Arthur"); IIdType ptId1 = myPatientDao.create(pt1, mySrd).getId().toUnqualifiedVersionless(); Patient pt2 = new Patient(); pt2.addName().setFamily("Everything").addGiven("Arthur"); IIdType ptId2 = myPatientDao.create(pt2, mySrd).getId().toUnqualifiedVersionless(); Device dev1 = new Device(); dev1.setManufacturer("Some Manufacturer"); IIdType devId1 = myDeviceDao.create(dev1, mySrd).getId().toUnqualifiedVersionless(); Device dev2 = new Device(); dev2.setManufacturer("Some Manufacturer 2"); myDeviceDao.create(dev2, mySrd).getId().toUnqualifiedVersionless(); Observation obs1 = new Observation(); obs1.getText().setDivAsString("<div>OBSTEXT1</div>"); obs1.getSubject().setReferenceElement(ptId1); obs1.getCode().addCoding().setCode("CODE1"); obs1.setValue(new StringType("obsvalue1")); obs1.getDevice().setReferenceElement(devId1); IIdType obsId1 = myObservationDao.create(obs1, mySrd).getId().toUnqualifiedVersionless(); Observation obs2 = new Observation(); obs2.getSubject().setReferenceElement(ptId1); obs2.getCode().addCoding().setCode("CODE2"); obs2.setValue(new StringType("obsvalue2")); IIdType obsId2 = myObservationDao.create(obs2, mySrd).getId().toUnqualifiedVersionless(); Observation obs3 = new Observation(); obs3.getSubject().setReferenceElement(ptId2); obs3.getCode().addCoding().setCode("CODE3"); obs3.setValue(new StringType("obsvalue3")); IIdType obsId3 = myObservationDao.create(obs3, mySrd).getId().toUnqualifiedVersionless(); List<IIdType> actual; StringAndListParam param; ourLog.info("Pt1:{} Pt2:{} Obs1:{} Obs2:{} Obs3:{}", new Object[] { ptId1.getIdPart(), ptId2.getIdPart(), obsId1.getIdPart(), obsId2.getIdPart(), obsId3.getIdPart() }); param = new StringAndListParam(); param.addAnd(new StringOrListParam().addOr(new StringParam("obsvalue1"))); //@formatter:off Parameters response = ourClient .operation() .onInstance(ptId1) .named("everything") .withParameter(Parameters.class, Constants.PARAM_CONTENT, new StringType("obsvalue1")) .execute(); //@formatter:on actual = toUnqualifiedVersionlessIds((Bundle) response.getParameter().get(0).getResource()); assertThat(actual, containsInAnyOrder(ptId1, obsId1, devId1)); } /** * See #147"Patient" */ @Test public void testEverythingPatientDoesntRepeatPatient() throws Exception { Bundle b; IParser parser = myFhirCtx.newJsonParser(); b = parser.parseResource(Bundle.class, new InputStreamReader(ResourceProviderDstu3Test.class.getResourceAsStream("/bug147-bundle-dstu3.json"))); Bundle resp = ourClient.transaction().withBundle(b).execute(); List<IdType> ids = new ArrayList<IdType>(); for (BundleEntryComponent next : resp.getEntry()) { IdType toAdd = new IdType(next.getResponse().getLocation()).toUnqualifiedVersionless(); ids.add(toAdd); } ourLog.info("Created: " + ids.toString()); IdType patientId = new IdType(resp.getEntry().get(0).getResponse().getLocation()); assertEquals("Patient", patientId.getResourceType()); { Parameters output = ourClient.operation().onInstance(patientId).named("everything").withNoParameters(Parameters.class).execute(); b = (Bundle) output.getParameter().get(0).getResource(); ids = new ArrayList<IdType>(); boolean dupes = false; for (BundleEntryComponent next : b.getEntry()) { IdType toAdd = next.getResource().getIdElement().toUnqualifiedVersionless(); dupes = dupes | ids.contains(toAdd); ids.add(toAdd); } ourLog.info("$everything: " + ids.toString()); assertFalse(ids.toString(), dupes); } /* * Now try with a size specified */ { Parameters input = new Parameters(); input.addParameter().setName(Constants.PARAM_COUNT).setValue(new UnsignedIntType(100)); Parameters output = ourClient.operation().onInstance(patientId).named("everything").withParameters(input).execute(); b = (Bundle) output.getParameter().get(0).getResource(); ids = new ArrayList<IdType>(); boolean dupes = false; for (BundleEntryComponent next : b.getEntry()) { IdType toAdd = next.getResource().getIdElement().toUnqualifiedVersionless(); dupes = dupes | ids.contains(toAdd); ids.add(toAdd); } ourLog.info("$everything: " + ids.toString()); assertFalse(ids.toString(), dupes); assertThat(ids.toString(), containsString("Condition")); assertThat(ids.size(), greaterThan(10)); } } /** * Test for #226 */ @Test public void testEverythingPatientIncludesBackReferences() throws Exception { String methodName = "testEverythingIncludesBackReferences"; Medication med = new Medication(); med.getCode().setText(methodName); IIdType medId = myMedicationDao.create(med, mySrd).getId().toUnqualifiedVersionless(); Patient pat = new Patient(); pat.addAddress().addLine(methodName); IIdType patId = myPatientDao.create(pat, mySrd).getId().toUnqualifiedVersionless(); MedicationRequest mo = new MedicationRequest(); mo.getSubject().setReferenceElement(patId); mo.setMedication(new Reference(medId)); IIdType moId = myMedicationRequestDao.create(mo, mySrd).getId().toUnqualifiedVersionless(); Parameters output = ourClient.operation().onInstance(patId).named("everything").withNoParameters(Parameters.class).execute(); Bundle b = (Bundle) output.getParameter().get(0).getResource(); List<IIdType> ids = toUnqualifiedVersionlessIds(b); ourLog.info(ids.toString()); assertThat(ids, containsInAnyOrder(patId, medId, moId)); } /** * See #148 */ @Test public void testEverythingPatientIncludesCondition() throws Exception { Bundle b = new Bundle(); Patient p = new Patient(); p.setId("1"); b.addEntry().setResource(p).getRequest().setMethod(HTTPVerb.POST); Condition c = new Condition(); c.getSubject().setReference("Patient/1"); b.addEntry().setResource(c).getRequest().setMethod(HTTPVerb.POST); Bundle resp = ourClient.transaction().withBundle(b).execute(); ourLog.info(myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(resp)); IdType patientId = new IdType(resp.getEntry().get(0).getResponse().getLocation()); assertEquals("Patient", patientId.getResourceType()); Parameters output = ourClient.operation().onInstance(patientId).named("everything").withNoParameters(Parameters.class).execute(); b = (Bundle) output.getParameter().get(0).getResource(); List<IdType> ids = new ArrayList<IdType>(); for (BundleEntryComponent next : b.getEntry()) { IdType toAdd = next.getResource().getIdElement().toUnqualifiedVersionless(); ids.add(toAdd); } assertThat(ids.toString(), containsString("Patient/")); assertThat(ids.toString(), containsString("Condition/")); } // private void delete(String theResourceType, String theParamName, String theParamValue) { // Bundle resources; // do { // IQuery<Bundle> forResource = ourClient.search().forResource(theResourceType); // if (theParamName != null) { // forResource = forResource.where(new StringClientParam(theParamName).matches().value(theParamValue)); // } // resources = forResource.execute(); // for (IResource next : resources.toListOfResources()) { // ourLog.info("Deleting resource: {}", next.getId()); // ourClient.delete().resource(next).execute(); // } // } while (resources.size() > 0); // } // // private void deleteToken(String theResourceType, String theParamName, String theParamSystem, String theParamValue) // { // Bundle resources = ourClient.search().forResource(theResourceType).where(new // TokenClientParam(theParamName).exactly().systemAndCode(theParamSystem, theParamValue)).execute(); // for (IResource next : resources.toListOfResources()) { // ourLog.info("Deleting resource: {}", next.getId()); // ourClient.delete().resource(next).execute(); // } // } @Test public void testEverythingPatientOperation() throws Exception { String methodName = "testEverythingOperation"; Organization org1parent = new Organization(); org1parent.setId("org1parent"); org1parent.setName(methodName + "1parent"); IIdType orgId1parent = ourClient.update().resource(org1parent).execute().getId().toUnqualifiedVersionless(); Organization org1 = new Organization(); org1.setName(methodName + "1"); org1.getPartOf().setReferenceElement(orgId1parent); IIdType orgId1 = ourClient.create().resource(org1).execute().getId().toUnqualifiedVersionless(); Patient p = new Patient(); p.addName().setFamily(methodName); p.getManagingOrganization().setReferenceElement(orgId1); IIdType patientId = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); Organization org2 = new Organization(); org2.setName(methodName + "1"); IIdType orgId2 = ourClient.create().resource(org2).execute().getId().toUnqualifiedVersionless(); Device dev = new Device(); dev.setModel(methodName); dev.getOwner().setReferenceElement(orgId2); IIdType devId = ourClient.create().resource(dev).execute().getId().toUnqualifiedVersionless(); Observation obs = new Observation(); obs.getSubject().setReferenceElement(patientId); obs.getDevice().setReferenceElement(devId); IIdType obsId = ourClient.create().resource(obs).execute().getId().toUnqualifiedVersionless(); Encounter enc = new Encounter(); enc.getSubject().setReferenceElement(patientId); IIdType encId = ourClient.create().resource(enc).execute().getId().toUnqualifiedVersionless(); Parameters output = ourClient.operation().onInstance(patientId).named("everything").withNoParameters(Parameters.class).execute(); Bundle b = (Bundle) output.getParameter().get(0).getResource(); List<IIdType> ids = toUnqualifiedVersionlessIds(b); assertThat(ids, containsInAnyOrder(patientId, devId, obsId, encId, orgId1, orgId2, orgId1parent)); ourLog.info(ids.toString()); } @Test public void testEverythingPatientType() throws Exception { String methodName = "testEverythingPatientType"; Organization o1 = new Organization(); o1.setName(methodName + "1"); IIdType o1Id = ourClient.create().resource(o1).execute().getId().toUnqualifiedVersionless(); Organization o2 = new Organization(); o2.setName(methodName + "2"); IIdType o2Id = ourClient.create().resource(o2).execute().getId().toUnqualifiedVersionless(); Patient p1 = new Patient(); p1.addName().setFamily(methodName + "1"); p1.getManagingOrganization().setReferenceElement(o1Id); IIdType p1Id = ourClient.create().resource(p1).execute().getId().toUnqualifiedVersionless(); Patient p2 = new Patient(); p2.addName().setFamily(methodName + "2"); p2.getManagingOrganization().setReferenceElement(o2Id); IIdType p2Id = ourClient.create().resource(p2).execute().getId().toUnqualifiedVersionless(); Condition c1 = new Condition(); c1.getSubject().setReferenceElement(p1Id); IIdType c1Id = ourClient.create().resource(c1).execute().getId().toUnqualifiedVersionless(); Condition c2 = new Condition(); c2.getSubject().setReferenceElement(p2Id); IIdType c2Id = ourClient.create().resource(c2).execute().getId().toUnqualifiedVersionless(); Condition c3 = new Condition(); c3.addIdentifier().setValue(methodName + "3"); IIdType c3Id = ourClient.create().resource(c3).execute().getId().toUnqualifiedVersionless(); Parameters output = ourClient.operation().onType(Patient.class).named("everything").withNoParameters(Parameters.class).execute(); Bundle b = (Bundle) output.getParameter().get(0).getResource(); assertEquals(BundleType.SEARCHSET, b.getType()); List<IIdType> ids = toUnqualifiedVersionlessIds(b); assertThat(ids, containsInAnyOrder(o1Id, o2Id, p1Id, p2Id, c1Id, c2Id)); assertThat(ids, not(containsInRelativeOrder(c3Id))); } // retest @Test public void testEverythingPatientWithLastUpdatedAndSort() throws Exception { String methodName = "testEverythingWithLastUpdatedAndSort"; Organization org = new Organization(); org.setName(methodName); IIdType oId = ourClient.create().resource(org).execute().getId().toUnqualifiedVersionless(); long time1 = System.currentTimeMillis(); Thread.sleep(10); Patient p = new Patient(); p.addName().setFamily(methodName); p.getManagingOrganization().setReferenceElement(oId); IIdType pId = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); long time2 = System.currentTimeMillis(); Thread.sleep(10); Condition c = new Condition(); c.getCode().setText(methodName); c.getSubject().setReferenceElement(pId); IIdType cId = ourClient.create().resource(c).execute().getId().toUnqualifiedVersionless(); Thread.sleep(10); long time3 = System.currentTimeMillis(); // %3E=> %3C=< HttpGet get = new HttpGet(ourServerBase + "/Patient/" + pId.getIdPart() + "/$everything?_lastUpdated=%3E" + new InstantType(new Date(time1)).getValueAsString()); CloseableHttpResponse response = ourHttpClient.execute(get); try { assertEquals(200, response.getStatusLine().getStatusCode()); String output = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); IOUtils.closeQuietly(response.getEntity().getContent()); ourLog.info(output); List<IIdType> ids = toUnqualifiedVersionlessIds(myFhirCtx.newXmlParser().parseResource(Bundle.class, output)); ourLog.info(ids.toString()); assertThat(ids, containsInAnyOrder(pId, cId)); } finally { response.close(); } get = new HttpGet(ourServerBase + "/Patient/" + pId.getIdPart() + "/$everything?_lastUpdated=%3E" + new InstantType(new Date(time2)).getValueAsString() + "&_lastUpdated=%3C" + new InstantType(new Date(time3)).getValueAsString()); response = ourHttpClient.execute(get); try { assertEquals(200, response.getStatusLine().getStatusCode()); String output = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); IOUtils.closeQuietly(response.getEntity().getContent()); ourLog.info(output); List<IIdType> ids = toUnqualifiedVersionlessIds(myFhirCtx.newXmlParser().parseResource(Bundle.class, output)); ourLog.info(ids.toString()); assertThat(ids, containsInAnyOrder(cId)); } finally { response.close(); } /* * Sorting is not working since the performance enhancements in 2.4 but * sorting for lastupdated is non-standard anyhow.. Hopefully at some point * we can bring this back */ // get = new HttpGet(ourServerBase + "/Patient/" + pId.getIdPart() + "/$everything?_lastUpdated=%3E" + new InstantType(new Date(time1)).getValueAsString() + "&_sort=_lastUpdated"); // response = ourHttpClient.execute(get); // try { // assertEquals(200, response.getStatusLine().getStatusCode()); // String output = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); // IOUtils.closeQuietly(response.getEntity().getContent()); // ourLog.info(output); // List<IIdType> ids = toUnqualifiedVersionlessIds(myFhirCtx.newXmlParser().parseResource(Bundle.class, output)); // ourLog.info(ids.toString()); // assertThat(ids, contains(pId, cId)); // } finally { // response.close(); // } // // get = new HttpGet(ourServerBase + "/Patient/" + pId.getIdPart() + "/$everything?_sort:desc=_lastUpdated"); // response = ourHttpClient.execute(get); // try { // assertEquals(200, response.getStatusLine().getStatusCode()); // String output = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); // IOUtils.closeQuietly(response.getEntity().getContent()); // ourLog.info(output); // List<IIdType> ids = toUnqualifiedVersionlessIds(myFhirCtx.newXmlParser().parseResource(Bundle.class, output)); // ourLog.info(ids.toString()); // assertThat(ids, contains(cId, pId, oId)); // } finally { // response.close(); // } } /** * Per message from David Hay on Skype */ @Test public void testEverythingWithLargeSet() throws Exception { String inputString = IOUtils.toString(getClass().getResourceAsStream("/david_big_bundle.json"), StandardCharsets.UTF_8); Bundle inputBundle = myFhirCtx.newJsonParser().parseResource(Bundle.class, inputString); inputBundle.setType(BundleType.TRANSACTION); assertEquals(53, inputBundle.getEntry().size()); Set<String> allIds = new TreeSet<String>(); for (BundleEntryComponent nextEntry : inputBundle.getEntry()) { nextEntry.getRequest().setMethod(HTTPVerb.PUT); nextEntry.getRequest().setUrl(nextEntry.getResource().getId()); allIds.add(nextEntry.getResource().getIdElement().toUnqualifiedVersionless().getValue()); } assertEquals(53, allIds.size()); mySystemDao.transaction(mySrd, inputBundle); Bundle responseBundle = ourClient .operation() .onInstance(new IdType("Patient/A161443")) .named("everything") .withParameter(Parameters.class, "_count", new IntegerType(20)) .useHttpGet() .returnResourceType(Bundle.class) .execute(); ourLog.info(myFhirCtx.newXmlParser().setPrettyPrint(true).encodeResourceToString(responseBundle)); assertEquals(23, responseBundle.getEntry().size()); TreeSet<String> ids = new TreeSet<String>(); for (int i = 0; i < responseBundle.getEntry().size(); i++) { for (BundleEntryComponent nextEntry : responseBundle.getEntry()) { ids.add(nextEntry.getResource().getIdElement().toUnqualifiedVersionless().getValue()); } } String nextUrl = responseBundle.getLink("next").getUrl(); responseBundle = ourClient.fetchResourceFromUrl(Bundle.class, nextUrl); for (int i = 0; i < responseBundle.getEntry().size(); i++) { for (BundleEntryComponent nextEntry : responseBundle.getEntry()) { ids.add(nextEntry.getResource().getIdElement().toUnqualifiedVersionless().getValue()); } } nextUrl = responseBundle.getLink("next").getUrl(); responseBundle = ourClient.fetchResourceFromUrl(Bundle.class, nextUrl); for (int i = 0; i < responseBundle.getEntry().size(); i++) { for (BundleEntryComponent nextEntry : responseBundle.getEntry()) { ids.add(nextEntry.getResource().getIdElement().toUnqualifiedVersionless().getValue()); } } assertEquals(null, responseBundle.getLink("next")); assertThat(ids, hasItem("List/A161444")); assertThat(ids, hasItem("List/A161468")); assertThat(ids, hasItem("List/A161500")); ourLog.info("Expected {} - {}", allIds.size(), allIds); ourLog.info("Actual {} - {}", ids.size(), ids); assertEquals(allIds, ids); } /** * Per message from David Hay on Skype */ @Test public void testEverythingWithLargeSet2() throws Exception { Patient p = new Patient(); p.setActive(true); IIdType id = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); for (int i = 1; i < 77; i++) { Observation obs = new Observation(); obs.setId("A" + StringUtils.leftPad(Integer.toString(i), 2, '0')); obs.setSubject(new Reference(id)); ourClient.update().resource(obs).execute(); } Bundle responseBundle = ourClient.operation().onInstance(id).named("everything").withParameter(Parameters.class, "_count", new IntegerType(50)).useHttpGet().returnResourceType(Bundle.class) .execute(); TreeSet<String> ids = new TreeSet<String>(); for (int i = 0; i < responseBundle.getEntry().size(); i++) { for (BundleEntryComponent nextEntry : responseBundle.getEntry()) { ids.add(nextEntry.getResource().getIdElement().getIdPart()); } } ourLog.info("Have {} IDs: {}", ids.size(), ids); BundleLinkComponent nextLink = responseBundle.getLink("next"); while (nextLink != null) { String nextUrl = nextLink.getUrl(); responseBundle = ourClient.fetchResourceFromUrl(Bundle.class, nextUrl); for (int i = 0; i < responseBundle.getEntry().size(); i++) { for (BundleEntryComponent nextEntry : responseBundle.getEntry()) { ids.add(nextEntry.getResource().getIdElement().getIdPart()); } } ourLog.info("Have {} IDs: {}", ids.size(), ids); nextLink = responseBundle.getLink("next"); } assertThat(ids, hasItem(id.getIdPart())); for (int i = 1; i < 77; i++) { assertThat(ids, hasItem("A" + StringUtils.leftPad(Integer.toString(i), 2, '0'))); } assertEquals(77, ids.size()); } @SuppressWarnings("unused") @Test public void testFullTextSearch() throws RuntimeException, Exception { Observation obs1 = new Observation(); obs1.getCode().setText("Systolic Blood Pressure"); obs1.setStatus(ObservationStatus.FINAL); obs1.setValue(new Quantity(123)); obs1.setComment("obs1"); IIdType id1 = myObservationDao.create(obs1, mySrd).getId().toUnqualifiedVersionless(); Observation obs2 = new Observation(); obs2.getCode().setText("Diastolic Blood Pressure"); obs2.setStatus(ObservationStatus.FINAL); obs2.setValue(new Quantity(81)); IIdType id2 = myObservationDao.create(obs2, mySrd).getId().toUnqualifiedVersionless(); HttpGet get = new HttpGet(ourServerBase + "/Observation?_content=systolic&_pretty=true"); CloseableHttpResponse response = ourHttpClient.execute(get); try { assertEquals(200, response.getStatusLine().getStatusCode()); String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertThat(responseString, containsString(id1.getIdPart())); } finally { response.close(); } } @Test public void testGetResourceCountsOperation() throws Exception { String methodName = "testMetaOperations"; Patient pt = new Patient(); pt.addName().setFamily(methodName); ourClient.create().resource(pt).execute().getId().toUnqualifiedVersionless(); HttpGet get = new HttpGet(ourServerBase + "/$get-resource-counts"); CloseableHttpResponse response = ourHttpClient.execute(get); try { assertEquals(200, response.getStatusLine().getStatusCode()); String output = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); IOUtils.closeQuietly(response.getEntity().getContent()); ourLog.info(output); assertThat(output, containsString("<parameter><name value=\"Patient\"/><valueInteger value=\"")); } finally { response.close(); } } @Test public void testHasParameter() throws Exception { IIdType pid0; { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("001"); patient.addName().setFamily("Tester").addGiven("Joe"); pid0 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("001"); patient.addName().setFamily("Tester").addGiven("Joe"); myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } { Observation obs = new Observation(); obs.addIdentifier().setSystem("urn:system").setValue("FOO"); obs.getSubject().setReferenceElement(pid0); myObservationDao.create(obs, mySrd); } { Device device = new Device(); device.addIdentifier().setValue("DEVICEID"); IIdType devId = myDeviceDao.create(device, mySrd).getId().toUnqualifiedVersionless(); Observation obs = new Observation(); obs.addIdentifier().setSystem("urn:system").setValue("NOLINK"); obs.setDevice(new Reference(devId)); myObservationDao.create(obs, mySrd); } String uri = ourServerBase + "/Patient?_has:Observation:subject:identifier=" + UrlUtil.escape("urn:system|FOO"); List<String> ids = searchAndReturnUnqualifiedVersionlessIdValues(uri); assertThat(ids, contains(pid0.getValue())); } @Test public void testHasParameterNoResults() throws Exception { HttpGet get = new HttpGet(ourServerBase + "/AllergyIntolerance?_has=Provenance:target:userID=12345"); CloseableHttpResponse response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertThat(resp, containsString("Invalid _has parameter syntax: _has")); } finally { IOUtils.closeQuietly(response); } } @Test public void testHistoryWithAtParameter() throws Exception { String methodName = "testHistoryWithFromAndTo"; Patient patient = new Patient(); patient.addName().setFamily(methodName); IIdType id = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); List<Date> preDates = Lists.newArrayList(); List<String> ids = Lists.newArrayList(); for (int i = 0; i < 10; i++) { Thread.sleep(100); preDates.add(new Date()); Thread.sleep(100); patient.setId(id); patient.getName().get(0).getFamilyElement().setValue(methodName + "_i"+i); ids.add(myPatientDao.update(patient, mySrd).getId().toUnqualified().getValue()); } List<String> idValues; idValues = searchAndReturnUnqualifiedIdValues(ourServerBase + "/Patient/" + id.getIdPart() + "/_history?_at=gt" + toStr(preDates.get(0)) + "&_at=lt" + toStr(preDates.get(3))); assertThat(idValues.toString(), idValues, contains(ids.get(2), ids.get(1), ids.get(0))); idValues = searchAndReturnUnqualifiedIdValues(ourServerBase + "/Patient/_history?_at=gt" + toStr(preDates.get(0)) + "&_at=lt" + toStr(preDates.get(3))); assertThat(idValues.toString(), idValues, contains(ids.get(2), ids.get(1), ids.get(0))); idValues = searchAndReturnUnqualifiedIdValues(ourServerBase + "/_history?_at=gt" + toStr(preDates.get(0)) + "&_at=lt" + toStr(preDates.get(3))); assertThat(idValues.toString(), idValues, contains(ids.get(2), ids.get(1), ids.get(0))); idValues = searchAndReturnUnqualifiedIdValues(ourServerBase + "/_history?_at=gt2060"); assertThat(idValues.toString(), idValues, empty()); idValues = searchAndReturnUnqualifiedIdValues(ourServerBase + "/_history?_at=" + InstantDt.withCurrentTime().getYear()); assertThat(idValues.toString(), idValues, hasSize(10)); // 10 is the page size idValues = searchAndReturnUnqualifiedIdValues(ourServerBase + "/_history?_at=ge" + InstantDt.withCurrentTime().getYear()); assertThat(idValues.toString(), idValues, hasSize(10)); idValues = searchAndReturnUnqualifiedIdValues(ourServerBase + "/_history?_at=gt" + InstantDt.withCurrentTime().getYear()); assertThat(idValues, hasSize(0)); } @Test public void testHistoryWithDeletedResource() { String methodName = "testHistoryWithDeletedResource"; Patient patient = new Patient(); patient.addName().setFamily(methodName); IIdType id = ourClient.create().resource(patient).execute().getId().toVersionless(); ourClient.delete().resourceById(id).execute(); patient.setId(id); ourClient.update().resource(patient).execute(); Bundle history = ourClient.history().onInstance(id).andReturnBundle(Bundle.class).prettyPrint().summaryMode(SummaryEnum.DATA).execute(); assertEquals(3, history.getEntry().size()); assertEquals(id.withVersion("3").getValue(), history.getEntry().get(0).getResource().getId()); assertEquals(1, ((Patient) history.getEntry().get(0).getResource()).getName().size()); assertEquals(id.withVersion("2").getValue(), history.getEntry().get(1).getResource().getId()); assertEquals(HTTPVerb.DELETE, history.getEntry().get(1).getRequest().getMethodElement().getValue()); assertEquals("http://localhost:" + ourPort + "/fhir/context/Patient/" + id.getIdPart() + "/_history/2", history.getEntry().get(1).getRequest().getUrl()); assertEquals(0, ((Patient) history.getEntry().get(1).getResource()).getName().size()); assertEquals(id.withVersion("1").getValue(), history.getEntry().get(2).getResource().getId()); assertEquals(1, ((Patient) history.getEntry().get(2).getResource()).getName().size()); try { myBundleDao.validate(history, null, null, null, null, null, mySrd); } catch (PreconditionFailedException e) { ourLog.info(myFhirCtx.newJsonParser().setPrettyPrint(true).encodeResourceToString(e.getOperationOutcome())); throw e; } } @Test public void testIdAndVersionInBodyForCreate() throws IOException { String methodName = "testIdAndVersionInBodyForCreate"; Patient pt = new Patient(); pt.setId("Patient/AAA/_history/4"); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().setPrettyPrint(true).encodeResourceToString(pt); ourLog.info("Input: {}", resource); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); IdType id; try { String respString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info("Response: {}", respString); assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } assertEquals("1", id.getVersionIdPart()); assertNotEquals("AAA", id.getIdPart()); HttpGet get = new HttpGet(ourServerBase + "/Patient/" + id.getIdPart()); response = ourHttpClient.execute(get); try { assertEquals(200, response.getStatusLine().getStatusCode()); String respString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info("Response: {}", respString); assertThat(respString, containsString("<id value=\"" + id.getIdPart() + "\"/>")); assertThat(respString, containsString("<versionId value=\"1\"/>")); } finally { response.close(); } } @Test public void testIdAndVersionInBodyForUpdate() throws IOException { String methodName = "testIdAndVersionInBodyForUpdate"; Patient pt = new Patient(); pt.setId("Patient/AAA/_history/4"); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().setPrettyPrint(true).encodeResourceToString(pt); ourLog.info("Input: {}", resource); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); IdType id; try { String respString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info("Response: {}", respString); assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } assertEquals("1", id.getVersionIdPart()); assertNotEquals("AAA", id.getIdPart()); HttpPut put = new HttpPut(ourServerBase + "/Patient/" + id.getIdPart() + "/_history/1"); put.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); response = ourHttpClient.execute(put); try { String respString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info("Response: {}", respString); assertEquals(400, response.getStatusLine().getStatusCode()); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, respString); assertEquals( "Can not update resource, resource body must contain an ID element which matches the request URL for update (PUT) operation - Resource body ID of \"AAA\" does not match URL ID of \"" + id.getIdPart() + "\"", oo.getIssue().get(0).getDiagnostics()); } finally { response.close(); } } /** * See issue #52 */ @Test public void testImagingStudyResources() throws Exception { IGenericClient client = ourClient; int initialSize = client.search().forResource(ImagingStudy.class).returnBundle(Bundle.class).execute().getEntry().size(); String resBody = IOUtils.toString(ResourceProviderDstu3Test.class.getResource("/imagingstudy.json"), StandardCharsets.UTF_8); client.create().resource(resBody).execute(); int newSize = client.search().forResource(ImagingStudy.class).returnBundle(Bundle.class).execute().getEntry().size(); assertEquals(1, newSize - initialSize); } @Test public void testIncludeWithExternalReferences() { myDaoConfig.setAllowExternalReferences(true); Patient p = new Patient(); p.getManagingOrganization().setReference("http://example.com/Organization/123"); ourClient.create().resource(p).execute(); Bundle b = ourClient.search().forResource("Patient").include(Patient.INCLUDE_ORGANIZATION).returnBundle(Bundle.class).execute(); assertEquals(1, b.getEntry().size()); } @Test public void testMetadata() throws Exception { HttpGet get = new HttpGet(ourServerBase + "/metadata"); CloseableHttpResponse response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(200, response.getStatusLine().getStatusCode()); assertThat(resp, stringContainsInOrder("THIS IS THE DESC")); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } @SuppressWarnings("unused") @Test public void testMetadataSuperParamsAreIncluded() throws IOException { StructureDefinition p = new StructureDefinition(); p.setAbstract(true); p.setUrl("http://example.com/foo"); IIdType id = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); Bundle resp = ourClient .search() .forResource(StructureDefinition.class) .where(StructureDefinition.URL.matches().value("http://example.com/foo")) .returnBundle(Bundle.class) .execute(); assertEquals(1, resp.getTotal()); } @Test public void testMetaOperations() throws Exception { String methodName = "testMetaOperations"; Patient pt = new Patient(); pt.addName().setFamily(methodName); IIdType id = ourClient.create().resource(pt).execute().getId().toUnqualifiedVersionless(); Meta meta = ourClient.meta().get(Meta.class).fromResource(id).execute(); assertEquals(0, meta.getTag().size()); Meta inMeta = new Meta(); inMeta.addTag().setSystem("urn:system1").setCode("urn:code1"); meta = ourClient.meta().add().onResource(id).meta(inMeta).execute(); assertEquals(1, meta.getTag().size()); inMeta = new Meta(); inMeta.addTag().setSystem("urn:system1").setCode("urn:code1"); meta = ourClient.meta().delete().onResource(id).meta(inMeta).execute(); assertEquals(0, meta.getTag().size()); } @Test public void testMetaOperationWithNoMetaParameter() throws Exception { Patient p = new Patient(); p.addName().setFamily("testMetaAddInvalid"); IIdType id = ourClient.create().resource(p).execute().getId().toUnqualifiedVersionless(); //@formatter:off String input = "<Parameters>\n" + " <meta>\n" + " <tag>\n" + " <system value=\"http://example.org/codes/tags\"/>\n" + " <code value=\"record-lost\"/>\n" + " <display value=\"Patient File Lost\"/>\n" + " </tag>\n" + " </meta>\n" + "</Parameters>"; //@formatter:on HttpPost post = new HttpPost(ourServerBase + "/Patient/" + id.getIdPart() + "/$meta-add"); post.setEntity(new StringEntity(input, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String output = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(output); assertEquals(400, response.getStatusLine().getStatusCode()); assertThat(output, containsString("Input contains no parameter with name 'meta'")); } finally { response.close(); } post = new HttpPost(ourServerBase + "/Patient/" + id.getIdPart() + "/$meta-delete"); post.setEntity(new StringEntity(input, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); response = ourHttpClient.execute(post); try { String output = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(output); assertEquals(400, response.getStatusLine().getStatusCode()); assertThat(output, containsString("Input contains no parameter with name 'meta'")); } finally { response.close(); } } @Test public void testPatchUsingJsonPatch() throws Exception { String methodName = "testPatchUsingJsonPatch"; IIdType pid1; { Patient patient = new Patient(); patient.setActive(true); patient.addIdentifier().setSystem("urn:system").setValue("0"); patient.addName().setFamily(methodName).addGiven("Joe"); pid1 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } HttpPatch patch = new HttpPatch(ourServerBase + "/Patient/" + pid1.getIdPart()); patch.setEntity(new StringEntity("[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]", ContentType.parse(Constants.CT_JSON_PATCH + Constants.CHARSET_UTF8_CTSUFFIX))); CloseableHttpResponse response = ourHttpClient.execute(patch); try { assertEquals(200, response.getStatusLine().getStatusCode()); String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); assertThat(responseString, containsString("<OperationOutcome")); assertThat(responseString, containsString("INFORMATION")); } finally { response.close(); } Patient newPt = ourClient.read().resource(Patient.class).withId(pid1.getIdPart()).execute(); assertEquals("2", newPt.getIdElement().getVersionIdPart()); assertEquals(false, newPt.getActive()); } @Test public void testPatchUsingJsonPatchWithContentionCheckBad() throws Exception { String methodName = "testPatchUsingJsonPatchWithContentionCheckBad"; IIdType pid1; { Patient patient = new Patient(); patient.setActive(true); patient.addIdentifier().setSystem("urn:system").setValue("0"); patient.addName().setFamily(methodName).addGiven("Joe"); pid1 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } HttpPatch patch = new HttpPatch(ourServerBase + "/Patient/" + pid1.getIdPart()); patch.setEntity(new StringEntity("[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]", ContentType.parse(Constants.CT_JSON_PATCH + Constants.CHARSET_UTF8_CTSUFFIX))); patch.addHeader("If-Match", "W/\"9\""); CloseableHttpResponse response = ourHttpClient.execute(patch); try { assertEquals(409, response.getStatusLine().getStatusCode()); String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); assertThat(responseString, containsString("<OperationOutcome")); assertThat(responseString, containsString("<diagnostics value=\"Version 9 is not the most recent version of this resource, unable to apply patch\"/>")); } finally { response.close(); } Patient newPt = ourClient.read().resource(Patient.class).withId(pid1.getIdPart()).execute(); assertEquals("1", newPt.getIdElement().getVersionIdPart()); assertEquals(true, newPt.getActive()); } @Test public void testPatchUsingJsonPatchWithContentionCheckGood() throws Exception { String methodName = "testPatchUsingJsonPatchWithContentionCheckGood"; IIdType pid1; { Patient patient = new Patient(); patient.setActive(true); patient.addIdentifier().setSystem("urn:system").setValue("0"); patient.addName().setFamily(methodName).addGiven("Joe"); pid1 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } HttpPatch patch = new HttpPatch(ourServerBase + "/Patient/" + pid1.getIdPart()); patch.setEntity(new StringEntity("[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]", ContentType.parse(Constants.CT_JSON_PATCH + Constants.CHARSET_UTF8_CTSUFFIX))); patch.addHeader("If-Match", "W/\"1\""); CloseableHttpResponse response = ourHttpClient.execute(patch); try { assertEquals(200, response.getStatusLine().getStatusCode()); String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); assertThat(responseString, containsString("<OperationOutcome")); assertThat(responseString, containsString("INFORMATION")); } finally { response.close(); } Patient newPt = ourClient.read().resource(Patient.class).withId(pid1.getIdPart()).execute(); assertEquals("2", newPt.getIdElement().getVersionIdPart()); assertEquals(false, newPt.getActive()); } @Test public void testPatchUsingXmlPatch() throws Exception { String methodName = "testPatchUsingXmlPatch"; IIdType pid1; { Patient patient = new Patient(); patient.setActive(true); patient.addIdentifier().setSystem("urn:system").setValue("0"); patient.addName().setFamily(methodName).addGiven("Joe"); pid1 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } HttpPatch patch = new HttpPatch(ourServerBase + "/Patient/" + pid1.getIdPart()); String patchString = "<?xml version=\"1.0\" encoding=\"UTF-8\"?><diff xmlns:fhir=\"http://hl7.org/fhir\"><replace sel=\"fhir:Patient/fhir:active/@value\">false</replace></diff>"; patch.setEntity(new StringEntity(patchString, ContentType.parse(Constants.CT_XML_PATCH + Constants.CHARSET_UTF8_CTSUFFIX))); CloseableHttpResponse response = ourHttpClient.execute(patch); try { assertEquals(200, response.getStatusLine().getStatusCode()); String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); assertThat(responseString, containsString("<OperationOutcome")); assertThat(responseString, containsString("INFORMATION")); } finally { response.close(); } Patient newPt = ourClient.read().resource(Patient.class).withId(pid1.getIdPart()).execute(); assertEquals("2", newPt.getIdElement().getVersionIdPart()); assertEquals(false, newPt.getActive()); } @Test public void testPreserveVersionsOnAuditEvent() { Organization org = new Organization(); org.setName("ORG"); IIdType orgId = ourClient.create().resource(org).execute().getId(); assertEquals("1", orgId.getVersionIdPart()); Patient patient = new Patient(); patient.addIdentifier().setSystem("http://uhn.ca/mrns").setValue("100"); patient.getManagingOrganization().setReference(orgId.toUnqualified().getValue()); IIdType patientId = ourClient.create().resource(patient).execute().getId(); assertEquals("1", patientId.getVersionIdPart()); AuditEvent ae = new org.hl7.fhir.dstu3.model.AuditEvent(); ae.addEntity().getReference().setReference(patientId.toUnqualified().getValue()); IIdType aeId = ourClient.create().resource(ae).execute().getId(); assertEquals("1", aeId.getVersionIdPart()); patient = ourClient.read().resource(Patient.class).withId(patientId).execute(); assertTrue(patient.getManagingOrganization().getReferenceElement().hasIdPart()); assertFalse(patient.getManagingOrganization().getReferenceElement().hasVersionIdPart()); ae = ourClient.read().resource(AuditEvent.class).withId(aeId).execute(); assertTrue(ae.getEntityFirstRep().getReference().getReferenceElement().hasIdPart()); assertTrue(ae.getEntityFirstRep().getReference().getReferenceElement().hasVersionIdPart()); } /** * See issue #52 */ @Test public void testProcedureRequestResources() throws Exception { IGenericClient client = ourClient; int initialSize = client.search().forResource(ProcedureRequest.class).returnBundle(Bundle.class).execute().getEntry().size(); ProcedureRequest res = new ProcedureRequest(); res.addIdentifier().setSystem("urn:foo").setValue("123"); client.create().resource(res).execute(); int newSize = client.search().forResource(ProcedureRequest.class).returnBundle(Bundle.class).execute().getEntry().size(); assertEquals(1, newSize - initialSize); } /** * Test for issue #60 */ @Test public void testReadAllInstancesOfType() throws Exception { Patient pat; pat = new Patient(); pat.addIdentifier().setSystem("urn:system").setValue("testReadAllInstancesOfType_01"); ourClient.create().resource(pat).prettyPrint().encodedXml().execute().getId(); pat = new Patient(); pat.addIdentifier().setSystem("urn:system").setValue("testReadAllInstancesOfType_02"); ourClient.create().resource(pat).prettyPrint().encodedXml().execute().getId(); { Bundle returned = ourClient.search().forResource(Patient.class).encodedXml().returnBundle(Bundle.class).execute(); assertThat(returned.getEntry().size(), greaterThan(1)); assertEquals(BundleType.SEARCHSET, returned.getType()); } { Bundle returned = ourClient.search().forResource(Patient.class).encodedJson().returnBundle(Bundle.class).execute(); assertThat(returned.getEntry().size(), greaterThan(1)); } } @Test public void testSaveAndRetrieveExistingNarrativeJson() { Patient p1 = new Patient(); p1.getText().setStatus(NarrativeStatus.GENERATED); p1.getText().getDiv().setValueAsString("<div>HELLO WORLD</div>"); p1.addIdentifier().setSystem("urn:system").setValue("testSaveAndRetrieveExistingNarrative01"); IIdType newId = ourClient.create().resource(p1).encodedJson().execute().getId(); Patient actual = ourClient.read().resource(Patient.class).withId(newId).encodedJson().execute(); assertEquals("<div xmlns=\"http://www.w3.org/1999/xhtml\">HELLO WORLD</div>", actual.getText().getDiv().getValueAsString()); } @Test public void testSaveAndRetrieveExistingNarrativeXml() { Patient p1 = new Patient(); p1.getText().setStatus(NarrativeStatus.GENERATED); p1.getText().getDiv().setValueAsString("<div>HELLO WORLD</div>"); p1.addIdentifier().setSystem("urn:system").setValue("testSaveAndRetrieveExistingNarrative01"); IIdType newId = ourClient.create().resource(p1).encodedXml().execute().getId(); Patient actual = ourClient.read().resource(Patient.class).withId(newId).encodedXml().execute(); assertEquals("<div xmlns=\"http://www.w3.org/1999/xhtml\">HELLO WORLD</div>", actual.getText().getDiv().getValueAsString()); } @Test public void testSaveAndRetrieveWithContained() { Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system:rpdstu2").setValue("testSaveAndRetrieveWithContained01"); Organization o1 = new Organization(); o1.addIdentifier().setSystem("urn:system:rpdstu2").setValue("testSaveAndRetrieveWithContained02"); p1.getManagingOrganization().setResource(o1); IIdType newId = ourClient.create().resource(p1).execute().getId(); Patient actual = ourClient.read(Patient.class, new UriDt(newId.getValue())); assertEquals(1, actual.getContained().size()); //@formatter:off Bundle b = ourClient .search() .forResource("Patient") .where(Patient.IDENTIFIER.exactly().systemAndCode("urn:system:rpdstu2", "testSaveAndRetrieveWithContained01")) .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on assertEquals(1, b.getEntry().size()); } @Test public void testSaveAndRetrieveWithoutNarrative() { Patient p1 = new Patient(); p1.getText().setDivAsString("<div><td>Identifier</td><td>testSearchByResourceChain01</td></div>"); p1.addIdentifier().setSystem("urn:system").setValue("testSearchByResourceChain01"); IdType newId = (IdType) ourClient.create().resource(p1).execute().getId(); Patient actual = ourClient.read(Patient.class, newId.getIdPart()); assertThat(actual.getText().getDiv().getValueAsString(), containsString("<td>Identifier</td><td>testSearchByResourceChain01</td>")); } @Test public void testSearchBundleDoesntIncludeTextElement() throws Exception { HttpGet read = new HttpGet(ourServerBase + "/Patient?_format=json"); CloseableHttpResponse response = ourHttpClient.execute(read); try { String text = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(text); assertEquals(Constants.STATUS_HTTP_200_OK, response.getStatusLine().getStatusCode()); assertThat(text, not(containsString("\"text\",\"type\""))); } finally { response.close(); } } @Test public void testSearchByExtendedChars() throws Exception { for (int i = 0; i < 10; i++) { Patient p = new Patient(); p.addName().setFamily("Jernelöv"); p.addIdentifier().setValue("ID" + i); myPatientDao.create(p, mySrd); } String uri = ourServerBase + "/Patient?name=" + URLEncoder.encode("Jernelöv", "UTF-8") + "&_count=5&_pretty=true"; ourLog.info("URI: {}", uri); HttpGet get = new HttpGet(uri); CloseableHttpResponse resp = ourHttpClient.execute(get); try { assertEquals(200, resp.getStatusLine().getStatusCode()); String output = IOUtils.toString(resp.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(output); Bundle b = myFhirCtx.newXmlParser().parseResource(Bundle.class, output); assertEquals("http://localhost:" + ourPort + "/fhir/context/Patient?_count=5&_pretty=true&name=Jernel%C3%B6v", b.getLink("self").getUrl()); Patient p = (Patient) b.getEntry().get(0).getResource(); assertEquals("Jernelöv", p.getName().get(0).getFamily()); } finally { IOUtils.closeQuietly(resp.getEntity().getContent()); } } @Test public void testSearchByIdentifier() { Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system").setValue("testSearchByIdentifier01"); p1.addName().setFamily("testSearchByIdentifierFamily01").addGiven("testSearchByIdentifierGiven01"); IdType p1Id = (IdType) ourClient.create().resource(p1).execute().getId(); Patient p2 = new Patient(); p2.addIdentifier().setSystem("urn:system").setValue("testSearchByIdentifier02"); p2.addName().setFamily("testSearchByIdentifierFamily01").addGiven("testSearchByIdentifierGiven02"); ourClient.create().resource(p2).execute().getId(); //@formatter:off Bundle actual = ourClient .search() .forResource(Patient.class) .where(Patient.IDENTIFIER.exactly().systemAndCode("urn:system", "testSearchByIdentifier01")) .encodedJson() .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on assertEquals(1, actual.getEntry().size()); assertEquals(ourServerBase + "/Patient/" + p1Id.getIdPart(), actual.getEntry().get(0).getFullUrl()); assertEquals(p1Id.getIdPart(), actual.getEntry().get(0).getResource().getIdElement().getIdPart()); assertEquals(SearchEntryMode.MATCH, actual.getEntry().get(0).getSearch().getModeElement().getValue()); } @Test public void testSearchByIdentifierWithoutSystem() { Patient p1 = new Patient(); p1.addIdentifier().setValue("testSearchByIdentifierWithoutSystem01"); IdType p1Id = (IdType) ourClient.create().resource(p1).execute().getId(); //@formatter:off Bundle actual = ourClient .search() .forResource(Patient.class) .where(Patient.IDENTIFIER.exactly().systemAndCode(null, "testSearchByIdentifierWithoutSystem01")) .encodedJson() .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on assertEquals(1, actual.getEntry().size()); assertEquals(p1Id.getIdPart(), actual.getEntry().get(0).getResource().getIdElement().getIdPart()); } @Test public void testSearchByIdOr() { IIdType id1; { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("001"); id1 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } IIdType id2; { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("001"); id2 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } Bundle found; found = ourClient .search() .forResource(Patient.class) .where(BaseResource.RES_ID.exactly().systemAndValues(null, id1.getIdPart(), id2.getIdPart())) .returnBundle(Bundle.class) .execute(); assertThat(toUnqualifiedVersionlessIds(found), containsInAnyOrder(id1, id2)); found = ourClient .search() .forResource(Patient.class) .where(BaseResource.RES_ID.exactly().systemAndValues(null, Arrays.asList(id1.getIdPart(), id2.getIdPart(), "FOOOOO"))) .returnBundle(Bundle.class) .execute(); assertThat(toUnqualifiedVersionlessIds(found), containsInAnyOrder(id1, id2)); found = ourClient .search() .forResource(Patient.class) .where(BaseResource.RES_ID.exactly().systemAndCode(null, id1.getIdPart())) .returnBundle(Bundle.class) .execute(); assertThat(toUnqualifiedVersionlessIds(found), containsInAnyOrder(id1)); found = ourClient .search() .forResource(Patient.class) .where(BaseResource.RES_ID.exactly().codes(id1.getIdPart(), id2.getIdPart())) .and(BaseResource.RES_ID.exactly().code(id1.getIdPart())) .returnBundle(Bundle.class) .execute(); assertThat(toUnqualifiedVersionlessIds(found), containsInAnyOrder(id1)); found = ourClient .search() .forResource(Patient.class) .where(BaseResource.RES_ID.exactly().codes(Arrays.asList(id1.getIdPart(), id2.getIdPart(), "FOOOOO"))) .and(BaseResource.RES_ID.exactly().code(id1.getIdPart())) .returnBundle(Bundle.class) .execute(); assertThat(toUnqualifiedVersionlessIds(found), containsInAnyOrder(id1)); found = ourClient .search() .forResource(Patient.class) .where(BaseResource.RES_ID.exactly().codes(id1.getIdPart(), id2.getIdPart(), "FOOO")) .returnBundle(Bundle.class) .execute(); assertThat(toUnqualifiedVersionlessIds(found), containsInAnyOrder(id1, id2)); found = ourClient .search() .forResource(Patient.class) .where(BaseResource.RES_ID.exactly().codes("FOOO")) .returnBundle(Bundle.class) .execute(); assertThat(toUnqualifiedVersionlessIds(found), empty()); } @Test public void testSearchByReferenceIds() { Organization o1 = new Organization(); o1.setName("testSearchByResourceChainName01"); IIdType o1id = ourClient.create().resource(o1).execute().getId().toUnqualifiedVersionless(); Organization o2 = new Organization(); o2.setName("testSearchByResourceChainName02"); IIdType o2id = ourClient.create().resource(o2).execute().getId().toUnqualifiedVersionless(); Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system").setValue("testSearchByReferenceIds01"); p1.addName().setFamily("testSearchByReferenceIdsFamily01").addGiven("testSearchByReferenceIdsGiven01"); p1.setManagingOrganization(new Reference(o1id.toUnqualifiedVersionless())); IIdType p1Id = ourClient.create().resource(p1).execute().getId(); Patient p2 = new Patient(); p2.addIdentifier().setSystem("urn:system").setValue("testSearchByReferenceIds02"); p2.addName().setFamily("testSearchByReferenceIdsFamily02").addGiven("testSearchByReferenceIdsGiven02"); p2.setManagingOrganization(new Reference(o2id.toUnqualifiedVersionless())); IIdType p2Id = ourClient.create().resource(p2).execute().getId(); //@formatter:off Bundle actual = ourClient.search() .forResource(Patient.class) .where(Patient.ORGANIZATION.hasAnyOfIds(Arrays.asList(o1id.getIdPart(), o2id.getIdPart()))) .encodedJson().prettyPrint().returnBundle(Bundle.class).execute(); //@formatter:on Set<String> expectedIds = new HashSet<String>(); expectedIds.add(p1Id.getIdPart()); expectedIds.add(p2Id.getIdPart()); Set<String> actualIds = new HashSet<String>(); for (BundleEntryComponent ele : actual.getEntry()) { actualIds.add(ele.getResource().getIdElement().getIdPart()); } assertEquals("Expects to retrieve the 2 patients which reference the two different organizations", expectedIds, actualIds); } @Test public void testSearchByResourceChain() { Organization o1 = new Organization(); o1.setName("testSearchByResourceChainName01"); IdType o1id = (IdType) ourClient.create().resource(o1).execute().getId().toUnqualifiedVersionless(); Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system").setValue("testSearchByResourceChain01"); p1.addName().setFamily("testSearchByResourceChainFamily01").addGiven("testSearchByResourceChainGiven01"); p1.setManagingOrganization(new Reference(o1id.toUnqualifiedVersionless())); IdType p1Id = (IdType) ourClient.create().resource(p1).execute().getId(); //@formatter:off Bundle actual = ourClient.search() .forResource(Patient.class) .where(Patient.ORGANIZATION.hasId(o1id.getIdPart())) .encodedJson().prettyPrint().returnBundle(Bundle.class).execute(); //@formatter:on assertEquals(1, actual.getEntry().size()); assertEquals(p1Id.getIdPart(), actual.getEntry().get(0).getResource().getIdElement().getIdPart()); //@formatter:off actual = ourClient.search() .forResource(Patient.class) .where(Patient.ORGANIZATION.hasId(o1id.getValue())) .encodedJson().prettyPrint().returnBundle(Bundle.class).execute(); //@formatter:on assertEquals(1, actual.getEntry().size()); assertEquals(p1Id.getIdPart(), actual.getEntry().get(0).getResource().getIdElement().getIdPart()); } @Test public void testSearchLastUpdatedParamRp() throws InterruptedException { String methodName = "testSearchLastUpdatedParamRp"; int sleep = 100; Thread.sleep(sleep); DateTimeType beforeAny = new DateTimeType(new Date(), TemporalPrecisionEnum.MILLI); IIdType id1a; { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("001"); patient.addName().setFamily(methodName).addGiven("Joe"); id1a = (IdType) ourClient.create().resource(patient).execute().getId().toUnqualifiedVersionless(); } IIdType id1b; { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("002"); patient.addName().setFamily(methodName + "XXXX").addGiven("Joe"); id1b = (IdType) ourClient.create().resource(patient).execute().getId().toUnqualifiedVersionless(); } Thread.sleep(1100); DateTimeType beforeR2 = new DateTimeType(new Date(), TemporalPrecisionEnum.MILLI); Thread.sleep(1100); IIdType id2; { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("002"); patient.addName().setFamily(methodName).addGiven("John"); id2 = (IdType) ourClient.create().resource(patient).execute().getId().toUnqualifiedVersionless(); } { //@formatter:off Bundle found = ourClient.search() .forResource(Patient.class) .where(Patient.NAME.matches().value("testSearchLastUpdatedParamRp")) .returnBundle(Bundle.class) .execute(); //@formatter:on List<IIdType> patients = toUnqualifiedVersionlessIds(found); assertThat(patients, hasItems(id1a, id1b, id2)); } { //@formatter:off Bundle found = ourClient.search() .forResource(Patient.class) .where(Patient.NAME.matches().value("testSearchLastUpdatedParamRp")) .lastUpdated(new DateRangeParam(beforeAny, null)) .returnBundle(Bundle.class) .execute(); //@formatter:on List<IIdType> patients = toUnqualifiedVersionlessIds(found); assertThat(patients, hasItems(id1a, id1b, id2)); } { //@formatter:off Bundle found = ourClient.search() .forResource(Patient.class) .where(Patient.NAME.matches().value("testSearchLastUpdatedParamRp")) .lastUpdated(new DateRangeParam(beforeR2, null)) .returnBundle(Bundle.class) .execute(); //@formatter:on List<IIdType> patients = toUnqualifiedVersionlessIds(found); assertThat(patients, hasItems(id2)); assertThat(patients, not(hasItems(id1a, id1b))); } { //@formatter:off Bundle found = ourClient.search() .forResource(Patient.class) .where(Patient.NAME.matches().value("testSearchLastUpdatedParamRp")) .lastUpdated(new DateRangeParam(beforeAny, beforeR2)) .returnBundle(Bundle.class) .execute(); //@formatter:on List<IIdType> patients = toUnqualifiedVersionlessIds(found); assertThat(patients.toString(), patients, not(hasItems(id2))); assertThat(patients.toString(), patients, (hasItems(id1a, id1b))); } { //@formatter:off Bundle found = ourClient.search() .forResource(Patient.class) .where(Patient.NAME.matches().value("testSearchLastUpdatedParamRp")) .lastUpdated(new DateRangeParam(null, beforeR2)) .returnBundle(Bundle.class) .execute(); //@formatter:on List<IIdType> patients = toUnqualifiedVersionlessIds(found); assertThat(patients, (hasItems(id1a, id1b))); assertThat(patients, not(hasItems(id2))); } } /** * See #441 */ @Test public void testSearchMedicationChain() throws Exception { Medication medication = new Medication(); medication.getCode().addCoding().setSystem("SYSTEM").setCode("04823543"); IIdType medId = myMedicationDao.create(medication).getId().toUnqualifiedVersionless(); MedicationAdministration ma = new MedicationAdministration(); ma.setMedication(new Reference(medId)); IIdType moId = myMedicationAdministrationDao.create(ma).getId().toUnqualifiedVersionless(); HttpGet get = new HttpGet(ourServerBase + "/MedicationAdministration?medication.code=04823543"); CloseableHttpResponse response = ourHttpClient.execute(get); try { assertEquals(200, response.getStatusLine().getStatusCode()); String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertThat(responseString, containsString(moId.getIdPart())); } finally { response.close(); } } @SuppressWarnings("unused") @Test public void testSearchPagingKeepsOldSearches() throws Exception { String methodName = "testSearchPagingKeepsOldSearches"; IIdType pid1; { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("0"); patient.addName().setFamily(methodName).addGiven("Joe"); pid1 = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } for (int i = 1; i <= 20; i++) { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue(Integer.toString(i)); patient.addName().setFamily(methodName).addGiven("Joe"); myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); } List<String> linkNext = Lists.newArrayList(); for (int i = 0; i < 100; i++) { Bundle bundle = ourClient.search().forResource(Patient.class).where(Patient.NAME.matches().value("testSearchPagingKeepsOldSearches")).count(5).returnBundle(Bundle.class).execute(); assertTrue(isNotBlank(bundle.getLink("next").getUrl())); assertEquals(5, bundle.getEntry().size()); linkNext.add(bundle.getLink("next").getUrl()); } int index = 0; for (String nextLink : linkNext) { ourLog.info("Fetching index {}", index++); Bundle b = ourClient.fetchResourceFromUrl(Bundle.class, nextLink); assertEquals(5, b.getEntry().size()); } } private void testSearchReturnsResults(String search) throws IOException, ClientProtocolException { int matches; HttpGet get = new HttpGet(ourServerBase + search); CloseableHttpResponse response = ourHttpClient.execute(get); String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); IOUtils.closeQuietly(response.getEntity().getContent()); ourLog.info(resp); Bundle bundle = myFhirCtx.newXmlParser().parseResource(Bundle.class, resp); matches = bundle.getTotal(); assertThat(matches, greaterThan(0)); } @Test public void testSearchReturnsSearchDate() throws Exception { Date before = new Date(); Thread.sleep(1); //@formatter:off Bundle found = ourClient .search() .forResource(Patient.class) .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on Thread.sleep(1); Date after = new Date(); InstantType updated = found.getMeta().getLastUpdatedElement(); assertNotNull(updated); Date value = updated.getValue(); assertNotNull(value); ourLog.info(value.getTime() + ""); ourLog.info(before.getTime() + ""); assertTrue(value.after(before)); assertTrue(new InstantDt(value) + " should be before " + new InstantDt(after), value.before(after)); } @Test public void testSearchReusesNoParams() throws Exception { List<IBaseResource> resources = new ArrayList<IBaseResource>(); for (int i = 0; i < 50; i++) { Organization org = new Organization(); org.setName("HELLO"); resources.add(org); } ourClient.transaction().withResources(resources).prettyPrint().encodedXml().execute(); myDaoConfig.setReuseCachedSearchResultsForMillis(1000L); Bundle result1 = ourClient .search() .forResource("Organization") .returnBundle(Bundle.class) .execute(); final String uuid1 = toSearchUuidFromLinkNext(result1); Bundle result2 = ourClient .search() .forResource("Organization") .returnBundle(Bundle.class) .execute(); final String uuid2 = toSearchUuidFromLinkNext(result2); assertEquals(uuid1, uuid2); } @Test public void testSearchReusesResultsDisabled() throws Exception { List<IBaseResource> resources = new ArrayList<IBaseResource>(); for (int i = 0; i < 50; i++) { Organization org = new Organization(); org.setName("HELLO"); resources.add(org); } ourClient.transaction().withResources(resources).prettyPrint().encodedXml().execute(); myDaoConfig.setReuseCachedSearchResultsForMillis(null); Bundle result1 = ourClient .search() .forResource("Organization") .where(Organization.NAME.matches().value("HELLO")) .count(5) .returnBundle(Bundle.class) .execute(); final String uuid1 = toSearchUuidFromLinkNext(result1); Bundle result2 = ourClient .search() .forResource("Organization") .where(Organization.NAME.matches().value("HELLO")) .count(5) .returnBundle(Bundle.class) .execute(); final String uuid2 = toSearchUuidFromLinkNext(result2); Bundle result3 = ourClient .search() .forResource("Organization") .where(Organization.NAME.matches().value("HELLO")) .count(5) .returnBundle(Bundle.class) .execute(); String uuid3 = toSearchUuidFromLinkNext(result3); assertNotEquals(uuid1, uuid2); assertNotEquals(uuid1, uuid3); } @Test public void testSearchReusesResultsEnabled() throws Exception { List<IBaseResource> resources = new ArrayList<IBaseResource>(); for (int i = 0; i < 50; i++) { Organization org = new Organization(); org.setName("HELLO"); resources.add(org); } ourClient.transaction().withResources(resources).prettyPrint().encodedXml().execute(); myDaoConfig.setReuseCachedSearchResultsForMillis(1000L); Bundle result1 = ourClient .search() .forResource("Organization") .where(Organization.NAME.matches().value("HELLO")) .count(5) .returnBundle(Bundle.class) .execute(); final String uuid1 = toSearchUuidFromLinkNext(result1); Search search1 = newTxTemplate().execute(new TransactionCallback<Search>() { @Override public Search doInTransaction(TransactionStatus theStatus) { return mySearchEntityDao.findByUuid(uuid1); } }); Date lastReturned1 = search1.getSearchLastReturned(); Bundle result2 = ourClient .search() .forResource("Organization") .where(Organization.NAME.matches().value("HELLO")) .count(5) .returnBundle(Bundle.class) .execute(); final String uuid2 = toSearchUuidFromLinkNext(result2); Search search2 = newTxTemplate().execute(new TransactionCallback<Search>() { @Override public Search doInTransaction(TransactionStatus theStatus) { return mySearchEntityDao.findByUuid(uuid2); } }); Date lastReturned2 = search2.getSearchLastReturned(); assertTrue(lastReturned2.getTime() > lastReturned1.getTime()); Thread.sleep(1500); Bundle result3 = ourClient .search() .forResource("Organization") .where(Organization.NAME.matches().value("HELLO")) .count(5) .returnBundle(Bundle.class) .execute(); String uuid3 = toSearchUuidFromLinkNext(result3); assertEquals(uuid1, uuid2); assertNotEquals(uuid1, uuid3); } @Test public void testSearchReusesResultsEnabledNoParams() throws Exception { List<IBaseResource> resources = new ArrayList<IBaseResource>(); for (int i = 0; i < 50; i++) { Organization org = new Organization(); org.setName("HELLO"); resources.add(org); } ourClient.transaction().withResources(resources).prettyPrint().encodedXml().execute(); myDaoConfig.setReuseCachedSearchResultsForMillis(100000L); Bundle result1 = ourClient .search() .forResource("Organization") .returnBundle(Bundle.class) .execute(); final String uuid1 = toSearchUuidFromLinkNext(result1); Search search1 = newTxTemplate().execute(new TransactionCallback<Search>() { @Override public Search doInTransaction(TransactionStatus theStatus) { return mySearchEntityDao.findByUuid(uuid1); } }); Date lastReturned1 = search1.getSearchLastReturned(); Bundle result2 = ourClient .search() .forResource("Organization") .returnBundle(Bundle.class) .execute(); final String uuid2 = toSearchUuidFromLinkNext(result2); Search search2 = newTxTemplate().execute(new TransactionCallback<Search>() { @Override public Search doInTransaction(TransactionStatus theStatus) { return mySearchEntityDao.findByUuid(uuid2); } }); Date lastReturned2 = search2.getSearchLastReturned(); assertTrue(lastReturned2.getTime() > lastReturned1.getTime()); assertEquals(uuid1, uuid2); } /** * See #316 */ @Test public void testSearchThenTagThenSearch() { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system2").setValue("testSearchTokenParam002"); patient.addName().setFamily("Tester").addGiven("testSearchTokenParam2"); ourClient.create().resource(patient).execute(); //@formatter:off Bundle response = ourClient .search() .forResource(Patient.class) .returnBundle(Bundle.class) .execute(); //@formatter:on patient = (Patient) response.getEntry().get(0).getResource(); //@formatter:off ourClient .meta() .add() .onResource(patient.getIdElement()) .meta(new Meta().addTag("http://system", "tag1", "display")) .execute(); //@formatter:on //@formatter:off response = ourClient .search() .forResource(Patient.class) .returnBundle(Bundle.class) .execute(); //@formatter:on patient = (Patient) response.getEntry().get(0).getResource(); assertEquals(1, patient.getMeta().getTag().size()); } @Test public void testSearchTokenParamNoValue() { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("testSearchTokenParam001"); patient.addName().setFamily("Tester").addGiven("testSearchTokenParam1"); patient.addCommunication().getLanguage().setText("testSearchTokenParamComText").addCoding().setCode("testSearchTokenParamCode").setSystem("testSearchTokenParamSystem") .setDisplay("testSearchTokenParamDisplay"); myPatientDao.create(patient, mySrd); patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("testSearchTokenParam002"); patient.addName().setFamily("Tester").addGiven("testSearchTokenParam2"); myPatientDao.create(patient, mySrd); patient = new Patient(); patient.addIdentifier().setSystem("urn:system2").setValue("testSearchTokenParam002"); patient.addName().setFamily("Tester").addGiven("testSearchTokenParam2"); myPatientDao.create(patient, mySrd); //@formatter:off Bundle response = ourClient .search() .forResource(Patient.class) .where(Patient.IDENTIFIER.hasSystemWithAnyCode("urn:system")) .returnBundle(Bundle.class) .execute(); //@formatter:on assertEquals(2, response.getEntry().size()); } @Test public void testSearchWithInclude() throws Exception { Organization org = new Organization(); org.addIdentifier().setSystem("urn:system:rpdstu2").setValue("testSearchWithInclude01"); IdType orgId = (IdType) ourClient.create().resource(org).prettyPrint().encodedXml().execute().getId(); Patient pat = new Patient(); pat.addIdentifier().setSystem("urn:system:rpdstu2").setValue("testSearchWithInclude02"); pat.getManagingOrganization().setReferenceElement(orgId.toUnqualifiedVersionless()); ourClient.create().resource(pat).prettyPrint().encodedXml().execute().getId(); //@formatter:off Bundle found = ourClient .search() .forResource(Patient.class) .where(Patient.IDENTIFIER.exactly().systemAndIdentifier("urn:system:rpdstu2","testSearchWithInclude02")) .include(Patient.INCLUDE_ORGANIZATION) .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on assertEquals(2, found.getEntry().size()); assertEquals(Patient.class, found.getEntry().get(0).getResource().getClass()); assertEquals(SearchEntryMode.MATCH, found.getEntry().get(0).getSearch().getMode()); assertEquals(Organization.class, found.getEntry().get(1).getResource().getClass()); assertEquals(SearchEntryMode.INCLUDE, found.getEntry().get(1).getSearch().getMode()); } @Test() public void testSearchWithInvalidNumberPrefix() throws Exception { try { //@formatter:off ourClient .search() .forResource(Encounter.class) .where(Encounter.LENGTH.withPrefix(ParamPrefixEnum.ENDS_BEFORE).number(100)) .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on fail(); } catch (InvalidRequestException e) { assertThat(e.getMessage(), containsString("Unable to handle number prefix \"eb\" for value: eb100")); } } @Test() public void testSearchWithInvalidQuantityPrefix() throws Exception { Observation o = new Observation(); o.getCode().setText("testSearchWithInvalidSort"); myObservationDao.create(o, mySrd); try { //@formatter:off ourClient .search() .forResource(Observation.class) .where(Observation.VALUE_QUANTITY.withPrefix(ParamPrefixEnum.ENDS_BEFORE).number(100).andNoUnits()) .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on fail(); } catch (InvalidRequestException e) { assertThat(e.getMessage(), containsString("Unable to handle quantity prefix \"eb\" for value: eb100||")); } } @Test(expected = InvalidRequestException.class) public void testSearchWithInvalidSort() throws Exception { Observation o = new Observation(); o.getCode().setText("testSearchWithInvalidSort"); myObservationDao.create(o, mySrd); //@formatter:off ourClient .search() .forResource(Observation.class) .sort().ascending(Observation.CODE_VALUE_QUANTITY) // composite sort not supported yet .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on } @Test public void testSearchWithMissing() throws Exception { ourLog.info("Starting testSearchWithMissing"); String methodName = "testSearchWithMissing"; Organization org = new Organization(); IIdType deletedIdMissingTrue = ourClient.create().resource(org).execute().getId().toUnqualifiedVersionless(); ourClient.delete().resourceById(deletedIdMissingTrue).execute(); org = new Organization(); org.setName("Help I'm a Bug"); IIdType deletedIdMissingFalse = ourClient.create().resource(org).execute().getId().toUnqualifiedVersionless(); ourClient.delete().resourceById(deletedIdMissingFalse).execute(); List<IBaseResource> resources = new ArrayList<IBaseResource>(); for (int i = 0; i < 20; i++) { org = new Organization(); org.setName(methodName + "_0" + i); resources.add(org); } ourClient.transaction().withResources(resources).prettyPrint().encodedXml().execute(); org = new Organization(); org.addIdentifier().setSystem("urn:system:rpdstu2").setValue(methodName + "01"); org.setName(methodName + "name"); IIdType orgNotMissing = ourClient.create().resource(org).prettyPrint().encodedXml().execute().getId().toUnqualifiedVersionless(); org = new Organization(); org.addIdentifier().setSystem("urn:system:rpdstu2").setValue(methodName + "01"); IIdType orgMissing = ourClient.create().resource(org).prettyPrint().encodedXml().execute().getId().toUnqualifiedVersionless(); { //@formatter:off Bundle found = ourClient .search() .forResource(Organization.class) .where(Organization.NAME.isMissing(false)) .count(100) .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on List<IIdType> list = toUnqualifiedVersionlessIds(found); ourLog.info(methodName + ": " + list.toString()); ourLog.info("Wanted " + orgNotMissing + " and not " + deletedIdMissingFalse + " but got " + list.size() + ": " + list); assertThat("Wanted " + orgNotMissing + " but got " + list.size() + ": " + list, list, containsInRelativeOrder(orgNotMissing)); assertThat(list, not(containsInRelativeOrder(deletedIdMissingFalse))); assertThat(list, not(containsInRelativeOrder(orgMissing))); } //@formatter:off Bundle found = ourClient .search() .forResource(Organization.class) .where(Organization.NAME.isMissing(true)) .count(100) .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on List<IIdType> list = toUnqualifiedVersionlessIds(found); ourLog.info(methodName + " found: " + list.toString() + " - Wanted " + orgMissing + " but not " + orgNotMissing); assertThat(list, not(containsInRelativeOrder(orgNotMissing))); assertThat(list, not(containsInRelativeOrder(deletedIdMissingTrue))); assertThat("Wanted " + orgMissing + " but found: " + list, list, containsInRelativeOrder(orgMissing)); } @Test public void testSearchWithMissing2() throws Exception { checkParamMissing(Observation.SP_CODE); checkParamMissing(Observation.SP_CATEGORY); checkParamMissing(Observation.SP_VALUE_STRING); checkParamMissing(Observation.SP_ENCOUNTER); checkParamMissing(Observation.SP_DATE); } @Test public void testSearchInvalidParam() throws Exception { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("0"); patient.addName().setFamily("testSearchWithMixedParams").addGiven("Joe"); myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); // should be subject._id HttpGet httpPost = new HttpGet(ourServerBase + "/Observation?subject.id=FOO"); CloseableHttpResponse resp = ourHttpClient.execute(httpPost); try { String respString = IOUtils.toString(resp.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(respString); assertThat(respString, containsString("Invalid parameter chain: subject.id")); assertEquals(400, resp.getStatusLine().getStatusCode()); } finally { IOUtils.closeQuietly(resp.getEntity().getContent()); } ourLog.info("Outgoing post: {}", httpPost); } /** * See #411 * * Let's see if we can reproduce this issue in JPA */ @Test public void testSearchWithMixedParams() throws Exception { Patient patient = new Patient(); patient.addIdentifier().setSystem("urn:system").setValue("0"); patient.addName().setFamily("testSearchWithMixedParams").addGiven("Joe"); myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); HttpPost httpPost = new HttpPost(ourServerBase + "/Patient/_search?_format=application/xml"); httpPost.addHeader("Cache-Control", "no-cache"); List<NameValuePair> parameters = Lists.newArrayList(); parameters.add(new BasicNameValuePair("name", "Smith")); httpPost.setEntity(new UrlEncodedFormEntity(parameters)); ourLog.info("Outgoing post: {}", httpPost); CloseableHttpResponse status = ourHttpClient.execute(httpPost); try { String responseContent = IOUtils.toString(status.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseContent); assertEquals(200, status.getStatusLine().getStatusCode()); } finally { IOUtils.closeQuietly(status.getEntity().getContent()); } } @Test public void testSearchWithTextInexactMatch() throws Exception { Observation obs = new Observation(); obs.getCode().setText("THIS_IS_THE_TEXT"); obs.getCode().addCoding().setSystem("SYSTEM").setCode("CODE").setDisplay("THIS_IS_THE_DISPLAY"); ourClient.create().resource(obs).execute(); testSearchReturnsResults("/Observation?code%3Atext=THIS_IS_THE_TEXT"); testSearchReturnsResults("/Observation?code%3Atext=THIS_IS_THE_"); testSearchReturnsResults("/Observation?code%3Atext=this_is_the_"); testSearchReturnsResults("/Observation?code%3Atext=THIS_IS_THE_DISPLAY"); testSearchReturnsResults("/Observation?code%3Atext=THIS_IS_THE_disp"); } /** * See #198 */ @Test public void testSortFromResourceProvider() { Patient p; String methodName = "testSortFromResourceProvider"; p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Daniel").setFamily("Adams"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Aaron").setFamily("Alexis"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Carol").setFamily("Allen"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Ruth").setFamily("Black"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Brian").setFamily("Brooks"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Susan").setFamily("Clark"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Amy").setFamily("Clark"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Anthony").setFamily("Coleman"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Steven").setFamily("Coleman"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Lisa").setFamily("Coleman"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Ruth").setFamily("Cook"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Betty").setFamily("Davis"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Joshua").setFamily("Diaz"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Brian").setFamily("Gracia"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Stephan").setFamily("Graham"); ourClient.create().resource(p).execute(); p = new Patient(); p.addIdentifier().setSystem("urn:system").setValue(methodName); p.addName().addGiven("Sarah").setFamily("Graham"); ourClient.create().resource(p).execute(); Bundle resp = ourClient .search() .forResource(Patient.class) .where(Patient.IDENTIFIER.exactly().systemAndCode("urn:system", methodName)) .sort().ascending(Patient.FAMILY) .sort().ascending(Patient.GIVEN) .count(100) .returnBundle(Bundle.class) .execute(); List<String> names = toNameList(resp); ourLog.info(StringUtils.join(names, '\n')); assertThat(names, contains( // this matches in order only "Daniel Adams", "Aaron Alexis", "Carol Allen", "Ruth Black", "Brian Brooks", "Amy Clark", "Susan Clark", "Anthony Coleman", "Lisa Coleman", "Steven Coleman", "Ruth Cook", "Betty Davis", "Joshua Diaz", "Brian Gracia", "Sarah Graham", "Stephan Graham")); } /** * Test for issue #60 */ @Test public void testStoreUtf8Characters() throws Exception { Organization org = new Organization(); org.setName("測試醫院"); org.addIdentifier().setSystem("urn:system").setValue("testStoreUtf8Characters_01"); IdType orgId = (IdType) ourClient.create().resource(org).prettyPrint().encodedXml().execute().getId(); // Read back directly from the DAO { Organization returned = myOrganizationDao.read(orgId, mySrd); String val = myFhirCtx.newXmlParser().setPrettyPrint(true).encodeResourceToString(returned); ourLog.info(val); assertThat(val, containsString("<name value=\"測試醫院\"/>")); } // Read back through the HTTP API { Organization returned = ourClient.read(Organization.class, orgId.getIdPart()); String val = myFhirCtx.newXmlParser().setPrettyPrint(true).encodeResourceToString(returned); ourLog.info(val); assertThat(val, containsString("<name value=\"測試醫院\"/>")); } } @Test public void testTransaction() throws Exception { String contents = loadClasspath("/update.xml"); HttpPost post = new HttpPost(ourServerBase); post.setEntity(new StringEntity(contents, ContentType.create("application/xml+fhir", "UTF-8"))); CloseableHttpResponse resp = ourHttpClient.execute(post); try { assertEquals(200, resp.getStatusLine().getStatusCode()); String output = IOUtils.toString(resp.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(output); } finally { resp.close(); } } @Test public void testTryToCreateResourceWithReferenceThatDoesntExist() { Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system").setValue("testTryToCreateResourceWithReferenceThatDoesntExist01"); p1.addName().setFamily("testTryToCreateResourceWithReferenceThatDoesntExistFamily01").addGiven("testTryToCreateResourceWithReferenceThatDoesntExistGiven01"); p1.setManagingOrganization(new Reference("Organization/99999999999")); try { ourClient.create().resource(p1).execute().getId(); fail(); } catch (InvalidRequestException e) { assertThat(e.getMessage(), containsString("Organization/99999999999")); } } @Test public void testUpdateInvalidReference() throws IOException, Exception { String methodName = "testUpdateInvalidReference"; Patient pt = new Patient(); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut post = new HttpPut(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertEquals(400, response.getStatusLine().getStatusCode()); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, responseString); assertThat(oo.getIssue().get(0).getDiagnostics(), containsString("Can not update resource, request URL must contain an ID element for update (PUT) operation (it must be of the form [base]/[resource type]/[id])")); } finally { response.close(); } } @Test public void testUpdateInvalidReference2() throws IOException, Exception { String methodName = "testUpdateInvalidReference2"; Patient pt = new Patient(); pt.setId("2"); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut post = new HttpPut(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertThat(responseString, containsString("Can not update resource, request URL must contain an ID element for update (PUT) operation (it must be of the form [base]/[resource type]/[id])")); assertThat(responseString, containsString("<OperationOutcome")); assertEquals(400, response.getStatusLine().getStatusCode()); } finally { response.close(); } } /** * This does not currently cause an error, so this test is disabled */ @Test @Ignore public void testUpdateNoIdInBody() throws IOException, Exception { String methodName = "testUpdateNoIdInBody"; Patient pt = new Patient(); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut post = new HttpPut(ourServerBase + "/Patient/FOO"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertThat(responseString, containsString("Can not update resource, request URL must contain an ID element for update (PUT) operation (it must be of the form [base]/[resource type]/[id])")); assertThat(responseString, containsString("<OperationOutcome")); assertEquals(400, response.getStatusLine().getStatusCode()); } finally { response.close(); } } @Test public void testUpdateRejectsIncorrectIds() throws Exception { Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system").setValue("testUpdateRejectsInvalidTypes"); p1.addName().setFamily("Tester").addGiven("testUpdateRejectsInvalidTypes"); IdType p1id = (IdType) ourClient.create().resource(p1).execute().getId(); // Try to update with the wrong ID in the resource body p1.setId("FOO"); p1.addAddress().addLine("NEWLINE"); String encoded = myFhirCtx.newJsonParser().encodeResourceToString(p1); ourLog.info(encoded); HttpPut put = new HttpPut(ourServerBase + "/Patient/" + p1id.getIdPart()); put.setEntity(new StringEntity(encoded, ContentType.create(Constants.CT_FHIR_JSON, "UTF-8"))); put.addHeader("Accept", Constants.CT_FHIR_JSON); CloseableHttpResponse response = ourHttpClient.execute(put); try { assertEquals(400, response.getStatusLine().getStatusCode()); OperationOutcome oo = myFhirCtx.newJsonParser().parseResource(OperationOutcome.class, new InputStreamReader(response.getEntity().getContent())); assertEquals( "Can not update resource, resource body must contain an ID element which matches the request URL for update (PUT) operation - Resource body ID of \"FOO\" does not match URL ID of \"" + p1id.getIdPart() + "\"", oo.getIssue().get(0).getDiagnostics()); } finally { response.close(); } // Try to update with the no ID in the resource body p1.setId((String) null); encoded = myFhirCtx.newJsonParser().encodeResourceToString(p1); put = new HttpPut(ourServerBase + "/Patient/" + p1id.getIdPart()); put.setEntity(new StringEntity(encoded, ContentType.create(Constants.CT_FHIR_JSON, "UTF-8"))); put.addHeader("Accept", Constants.CT_FHIR_JSON); response = ourHttpClient.execute(put); try { assertEquals(400, response.getStatusLine().getStatusCode()); OperationOutcome oo = myFhirCtx.newJsonParser().parseResource(OperationOutcome.class, new InputStreamReader(response.getEntity().getContent())); assertEquals("Can not update resource, resource body must contain an ID element for update (PUT) operation", oo.getIssue().get(0).getDiagnostics()); } finally { response.close(); } // Try to update with the to correct ID in the resource body p1.setId(p1id.getIdPart()); encoded = myFhirCtx.newJsonParser().encodeResourceToString(p1); put = new HttpPut(ourServerBase + "/Patient/" + p1id.getIdPart()); put.setEntity(new StringEntity(encoded, ContentType.create(Constants.CT_FHIR_JSON, "UTF-8"))); response = ourHttpClient.execute(put); try { assertEquals(200, response.getStatusLine().getStatusCode()); } finally { response.close(); } } @Test public void testUpdateRejectsInvalidTypes() throws InterruptedException { Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system").setValue("testUpdateRejectsInvalidTypes"); p1.addName().setFamily("Tester").addGiven("testUpdateRejectsInvalidTypes"); IdType p1id = (IdType) ourClient.create().resource(p1).execute().getId(); Organization p2 = new Organization(); p2.setId(p1id.getIdPart()); p2.getNameElement().setValue("testUpdateRejectsInvalidTypes"); try { ourClient.update().resource(p2).withId("Organization/" + p1id.getIdPart()).execute(); fail(); } catch (UnprocessableEntityException e) { // good } try { ourClient.update().resource(p2).withId("Patient/" + p1id.getIdPart()).execute(); fail(); } catch (UnprocessableEntityException e) { // good } } @Test public void testUpdateResourceConditional() throws IOException { String methodName = "testUpdateResourceConditional"; Patient pt = new Patient(); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPost post = new HttpPost(ourServerBase + "/Patient?name=" + methodName); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); IdType id; try { assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } pt.addAddress().addLine("AAAAAAAAAAAAAAAAAAA"); resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut put = new HttpPut(ourServerBase + "/Patient?name=" + methodName); put.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); response = ourHttpClient.execute(put); try { assertEquals(200, response.getStatusLine().getStatusCode()); IdType newId = new IdType(response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue()); assertEquals(id.toVersionless(), newId.toVersionless()); // version shouldn't match for conditional update assertNotEquals(id, newId); } finally { response.close(); } } @Test public void testUpdateResourceConditionalComplex() throws IOException { Patient pt = new Patient(); pt.addIdentifier().setSystem("http://general-hospital.co.uk/Identifiers").setValue("09832345234543876876"); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.addHeader(Constants.HEADER_IF_NONE_EXIST, "Patient?identifier=http://general-hospital.co.uk/Identifiers|09832345234543876876"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); IdType id; CloseableHttpResponse response = ourHttpClient.execute(post); try { assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } pt.addName().setFamily("FOO"); resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut put = new HttpPut(ourServerBase + "/Patient?identifier=" + ("http://general-hospital.co.uk/Identifiers|09832345234543876876".replace("|", UrlUtil.escape("|")))); put.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); IdType id2; response = ourHttpClient.execute(put); try { assertEquals(200, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id2 = new IdType(newIdString); } finally { response.close(); } assertEquals(id.getIdPart(), id2.getIdPart()); assertEquals("1", id.getVersionIdPart()); assertEquals("2", id2.getVersionIdPart()); } @Test public void testUpdateResourceWithPrefer() throws IOException, Exception { String methodName = "testUpdateResourceWithPrefer"; Patient pt = new Patient(); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPost post = new HttpPost(ourServerBase + "/Patient"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); IdType id; try { assertEquals(201, response.getStatusLine().getStatusCode()); String newIdString = response.getFirstHeader(Constants.HEADER_LOCATION_LC).getValue(); assertThat(newIdString, startsWith(ourServerBase + "/Patient/")); id = new IdType(newIdString); } finally { response.close(); } Date before = new Date(); Thread.sleep(100); pt = new Patient(); pt.setId(id.getIdPart()); resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut put = new HttpPut(ourServerBase + "/Patient/" + id.getIdPart()); put.addHeader(Constants.HEADER_PREFER, Constants.HEADER_PREFER_RETURN + '=' + Constants.HEADER_PREFER_RETURN_REPRESENTATION); put.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); response = ourHttpClient.execute(put); try { assertEquals(200, response.getStatusLine().getStatusCode()); String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); IOUtils.closeQuietly(response.getEntity().getContent()); Patient respPt = myFhirCtx.newXmlParser().parseResource(Patient.class, responseString); assertEquals("2", respPt.getIdElement().getVersionIdPart()); InstantType updateTime = respPt.getMeta().getLastUpdatedElement(); assertTrue(updateTime.getValue().after(before)); } finally { response.close(); } } @Test public void testUpdateWithClientSuppliedIdWhichDoesntExist() { Patient p1 = new Patient(); p1.addIdentifier().setSystem("urn:system").setValue("testUpdateWithClientSuppliedIdWhichDoesntExistRpDstu2"); MethodOutcome outcome = ourClient.update().resource(p1).withId("testUpdateWithClientSuppliedIdWhichDoesntExistRpDstu2").execute(); assertEquals(true, outcome.getCreated().booleanValue()); IdType p1Id = (IdType) outcome.getId(); assertThat(p1Id.getValue(), containsString("Patient/testUpdateWithClientSuppliedIdWhichDoesntExistRpDstu2/_history")); //@formatter:off Bundle actual = ourClient .search() .forResource(Patient.class) .where(Patient.IDENTIFIER.exactly().systemAndCode("urn:system", "testUpdateWithClientSuppliedIdWhichDoesntExistRpDstu2")) .encodedJson() .prettyPrint() .returnBundle(Bundle.class) .execute(); //@formatter:on assertEquals(1, actual.getEntry().size()); assertEquals(p1Id.getIdPart(), actual.getEntry().get(0).getResource().getIdElement().getIdPart()); } @Test public void testUpdateWithETag() throws IOException, Exception { String methodName = "testUpdateWithETag"; Patient pt = new Patient(); pt.addName().setFamily(methodName); IIdType id = ourClient.create().resource(pt).execute().getId().toUnqualifiedVersionless(); pt.addName().setFamily("FAM2"); pt.setId(id.toUnqualifiedVersionless()); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut put = new HttpPut(ourServerBase + "/Patient/" + id.getIdPart()); put.addHeader(Constants.HEADER_IF_MATCH, "W/\"44\""); put.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(put); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertEquals(409, response.getStatusLine().getStatusCode()); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, responseString); assertThat(oo.getIssue().get(0).getDiagnostics(), containsString("Trying to update Patient/" + id.getIdPart() + "/_history/44 but this is not the current version")); } finally { response.close(); } // Now a good one put = new HttpPut(ourServerBase + "/Patient/" + id.getIdPart()); put.addHeader(Constants.HEADER_IF_MATCH, "W/\"1\""); put.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); response = ourHttpClient.execute(put); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertEquals(200, response.getStatusLine().getStatusCode()); } finally { response.close(); } } @Test public void testUpdateWrongIdInBody() throws IOException, Exception { String methodName = "testUpdateWrongIdInBody"; Patient pt = new Patient(); pt.setId("333"); pt.addName().setFamily(methodName); String resource = myFhirCtx.newXmlParser().encodeResourceToString(pt); HttpPut post = new HttpPut(ourServerBase + "/Patient/A2"); post.setEntity(new StringEntity(resource, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String responseString = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(responseString); assertEquals(400, response.getStatusLine().getStatusCode()); OperationOutcome oo = myFhirCtx.newXmlParser().parseResource(OperationOutcome.class, responseString); assertEquals( "Can not update resource, resource body must contain an ID element which matches the request URL for update (PUT) operation - Resource body ID of \"333\" does not match URL ID of \"A2\"", oo.getIssue().get(0).getDiagnostics()); } finally { response.close(); } } @Test public void testValidateBadInputViaGet() throws IOException { HttpGet get = new HttpGet(ourServerBase + "/Patient/$validate?mode=create"); CloseableHttpResponse response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertThat(resp, containsString("No resource supplied for $validate operation (resource is required unless mode is "delete")")); assertEquals(400, response.getStatusLine().getStatusCode()); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } @Test public void testValidateBadInputViaPost() throws IOException { Parameters input = new Parameters(); input.addParameter().setName("mode").setValue(new CodeType("create")); String inputStr = myFhirCtx.newXmlParser().encodeResourceToString(input); ourLog.info(inputStr); HttpPost post = new HttpPost(ourServerBase + "/Patient/$validate"); post.setEntity(new StringEntity(inputStr, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertThat(resp, containsString("No resource supplied for $validate operation (resource is required unless mode is "delete")")); assertEquals(400, response.getStatusLine().getStatusCode()); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } @Test public void testValidateResourceBaseWithNoIdRaw() throws IOException { Patient patient = new Patient(); patient.addName().addGiven("James"); patient.setBirthDateElement(new DateType("2011-02-02")); patient.addContact().setGender(AdministrativeGender.MALE); String inputStr = myFhirCtx.newXmlParser().encodeResourceToString(patient); HttpPost post = new HttpPost(ourServerBase + "/Patient/$validate"); post.setEntity(new StringEntity(inputStr, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(412, response.getStatusLine().getStatusCode()); assertThat(resp, not(containsString("Resource has no id"))); assertThat(resp, stringContainsInOrder(">ERROR<", "[Patient.contact]", "<pre>SHALL at least contain a contact's details or a reference to an organization", "<issue><severity value=\"error\"/>")); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } // Y @Test public void testValidateResourceHuge() throws IOException { Patient patient = new Patient(); patient.addName().addGiven("James" + StringUtils.leftPad("James", 1000000, 'A')); ; patient.setBirthDateElement(new DateType("2011-02-02")); Parameters input = new Parameters(); input.addParameter().setName("resource").setResource(patient); String inputStr = myFhirCtx.newXmlParser().encodeResourceToString(input); ourLog.debug(inputStr); HttpPost post = new HttpPost(ourServerBase + "/Patient/$validate"); post.setEntity(new StringEntity(inputStr, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(200, response.getStatusLine().getStatusCode()); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } @Test public void testValidateResourceInstanceOnServer() throws IOException { Patient patient = new Patient(); patient.addName().addGiven("James"); patient.setBirthDateElement(new DateType("2011-02-02")); patient.addContact().setGender(AdministrativeGender.MALE); patient.addCommunication().setPreferred(true); // missing language IIdType id = myPatientDao.create(patient, mySrd).getId().toUnqualifiedVersionless(); HttpGet get = new HttpGet(ourServerBase + "/Patient/" + id.getIdPart() + "/$validate"); CloseableHttpResponse response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(412, response.getStatusLine().getStatusCode()); assertThat(resp, containsString("SHALL at least contain a contact's details or a reference to an organization")); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } @Test public void testValidateResourceWithId() throws IOException { Patient patient = new Patient(); patient.setId("A123"); patient.addName().addGiven("James"); patient.setBirthDateElement(new DateType("2011-02-02")); myPatientDao.update(patient, mySrd); Parameters input = new Parameters(); input.addParameter().setName("resource").setResource(patient); String inputStr = myFhirCtx.newXmlParser().encodeResourceToString(input); ourLog.info(inputStr); HttpPost post = new HttpPost(ourServerBase + "/Patient/A123/$validate"); post.setEntity(new StringEntity(inputStr, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(200, response.getStatusLine().getStatusCode()); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } // Y @Test public void testValidateResourceWithNoIdParameters() throws IOException { Patient patient = new Patient(); patient.addName().addGiven("James"); patient.setBirthDateElement(new DateType("2011-02-02")); Parameters input = new Parameters(); input.addParameter().setName("resource").setResource(patient); String inputStr = myFhirCtx.newXmlParser().encodeResourceToString(input); HttpPost post = new HttpPost(ourServerBase + "/Patient/$validate?_pretty=true"); post.setEntity(new StringEntity(inputStr, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(200, response.getStatusLine().getStatusCode()); assertThat(resp, not(containsString("Resource has no id"))); assertThat(resp, containsString("<pre>No issues detected during validation</pre>")); assertThat(resp, stringContainsInOrder("<issue>", "<severity value=\"information\"/>", "<code value=\"informational\"/>", "<diagnostics value=\"No issues detected during validation\"/>", "</issue>")); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } @Test public void testValidateResourceWithNoIdRaw() throws IOException { Patient patient = new Patient(); patient.addName().addGiven("James"); patient.setBirthDateElement(new DateType("2011-02-02")); String inputStr = myFhirCtx.newXmlParser().encodeResourceToString(patient); HttpPost post = new HttpPost(ourServerBase + "/Patient/$validate"); post.setEntity(new StringEntity(inputStr, ContentType.create(Constants.CT_FHIR_XML, "UTF-8"))); CloseableHttpResponse response = ourHttpClient.execute(post); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(200, response.getStatusLine().getStatusCode()); assertThat(resp, not(containsString("Resource has no id"))); assertThat(resp, containsString("<pre>No issues detected during validation</pre>")); assertThat(resp, stringContainsInOrder("<issue>", "<severity value=\"information\"/>", "<code value=\"informational\"/>", "<diagnostics value=\"No issues detected during validation\"/>", "</issue>")); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } @Test public void testValueSetExpandOperation() throws IOException { CodeSystem cs = myFhirCtx.newXmlParser().parseResource(CodeSystem.class, new InputStreamReader(ResourceProviderDstu3Test.class.getResourceAsStream("/extensional-case-3-cs.xml"))); ourClient.create().resource(cs).execute(); ValueSet upload = myFhirCtx.newXmlParser().parseResource(ValueSet.class, new InputStreamReader(ResourceProviderDstu3Test.class.getResourceAsStream("/extensional-case-3-vs.xml"))); IIdType vsid = ourClient.create().resource(upload).execute().getId().toUnqualifiedVersionless(); HttpGet get = new HttpGet(ourServerBase + "/ValueSet/" + vsid.getIdPart() + "/$expand"); CloseableHttpResponse response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(200, response.getStatusLine().getStatusCode()); assertThat(resp, containsString("<ValueSet xmlns=\"http://hl7.org/fhir\">")); assertThat(resp, containsString("<expansion>")); assertThat(resp, containsString("<contains>")); assertThat(resp, containsString("<system value=\"http://acme.org\"/>")); assertThat(resp, containsString("<code value=\"8450-9\"/>")); assertThat(resp, containsString("<display value=\"Systolic blood pressure--expiration\"/>")); assertThat(resp, containsString("</contains>")); assertThat(resp, containsString("<contains>")); assertThat(resp, containsString("<system value=\"http://acme.org\"/>")); assertThat(resp, containsString("<code value=\"11378-7\"/>")); assertThat(resp, containsString("<display value=\"Systolic blood pressure at First encounter\"/>")); assertThat(resp, containsString("</contains>")); assertThat(resp, containsString("</expansion>")); } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } /* * Filter with display name */ get = new HttpGet(ourServerBase + "/ValueSet/" + vsid.getIdPart() + "/$expand?filter=systolic"); response = ourHttpClient.execute(get); try { String resp = IOUtils.toString(response.getEntity().getContent(), StandardCharsets.UTF_8); ourLog.info(resp); assertEquals(200, response.getStatusLine().getStatusCode()); //@formatter:off assertThat(resp, stringContainsInOrder( "<code value=\"11378-7\"/>", "<display value=\"Systolic blood pressure at First encounter\"/>")); //@formatter:on } finally { IOUtils.closeQuietly(response.getEntity().getContent()); response.close(); } } private String toStr(Date theDate) { return new InstantDt(theDate).getValueAsString(); } @AfterClass public static void afterClassClearContext() { TestUtil.clearAllStaticFieldsForUnitTest(); } }