Clean up patch client
This commit is contained in:
parent
d1d3b18729
commit
a03805a6d4
|
@ -24,41 +24,15 @@ import static org.apache.commons.lang3.StringUtils.isNotBlank;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.Reader;
|
import java.io.Reader;
|
||||||
import java.util.ArrayList;
|
import java.util.*;
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Date;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.LinkedHashMap;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Map.Entry;
|
import java.util.Map.Entry;
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import org.apache.commons.io.IOUtils;
|
import org.apache.commons.io.IOUtils;
|
||||||
import org.apache.commons.lang3.StringUtils;
|
import org.apache.commons.lang3.StringUtils;
|
||||||
import org.apache.commons.lang3.Validate;
|
import org.apache.commons.lang3.Validate;
|
||||||
import org.hl7.fhir.instance.model.api.IBase;
|
import org.hl7.fhir.instance.model.api.*;
|
||||||
import org.hl7.fhir.instance.model.api.IBaseBundle;
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseConformance;
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseDatatype;
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseMetaType;
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseOperationOutcome;
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseParameters;
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseResource;
|
|
||||||
import org.hl7.fhir.instance.model.api.IIdType;
|
|
||||||
import org.hl7.fhir.instance.model.api.IPrimitiveType;
|
|
||||||
|
|
||||||
import ca.uhn.fhir.context.BaseRuntimeChildDefinition;
|
import ca.uhn.fhir.context.*;
|
||||||
import ca.uhn.fhir.context.BaseRuntimeElementCompositeDefinition;
|
|
||||||
import ca.uhn.fhir.context.BaseRuntimeElementDefinition;
|
|
||||||
import ca.uhn.fhir.context.FhirContext;
|
|
||||||
import ca.uhn.fhir.context.FhirVersionEnum;
|
|
||||||
import ca.uhn.fhir.context.IRuntimeDatatypeDefinition;
|
|
||||||
import ca.uhn.fhir.context.RuntimeResourceDefinition;
|
|
||||||
import ca.uhn.fhir.model.api.Bundle;
|
import ca.uhn.fhir.model.api.Bundle;
|
||||||
import ca.uhn.fhir.model.api.IQueryParameterType;
|
import ca.uhn.fhir.model.api.IQueryParameterType;
|
||||||
import ca.uhn.fhir.model.api.Include;
|
import ca.uhn.fhir.model.api.Include;
|
||||||
|
@ -71,82 +45,13 @@ import ca.uhn.fhir.model.primitive.InstantDt;
|
||||||
import ca.uhn.fhir.model.primitive.UriDt;
|
import ca.uhn.fhir.model.primitive.UriDt;
|
||||||
import ca.uhn.fhir.parser.DataFormatException;
|
import ca.uhn.fhir.parser.DataFormatException;
|
||||||
import ca.uhn.fhir.parser.IParser;
|
import ca.uhn.fhir.parser.IParser;
|
||||||
import ca.uhn.fhir.rest.api.MethodOutcome;
|
import ca.uhn.fhir.rest.api.*;
|
||||||
import ca.uhn.fhir.rest.api.PatchTypeEnum;
|
|
||||||
import ca.uhn.fhir.rest.api.PreferReturnEnum;
|
|
||||||
import ca.uhn.fhir.rest.api.SortOrderEnum;
|
|
||||||
import ca.uhn.fhir.rest.api.SortSpec;
|
|
||||||
import ca.uhn.fhir.rest.api.SummaryEnum;
|
|
||||||
import ca.uhn.fhir.rest.client.api.IHttpClient;
|
import ca.uhn.fhir.rest.client.api.IHttpClient;
|
||||||
import ca.uhn.fhir.rest.client.api.IHttpRequest;
|
import ca.uhn.fhir.rest.client.api.IHttpRequest;
|
||||||
import ca.uhn.fhir.rest.client.exceptions.NonFhirResponseException;
|
import ca.uhn.fhir.rest.client.exceptions.NonFhirResponseException;
|
||||||
import ca.uhn.fhir.rest.client.interceptor.LoggingInterceptor;
|
import ca.uhn.fhir.rest.client.interceptor.LoggingInterceptor;
|
||||||
import ca.uhn.fhir.rest.gclient.IClientExecutable;
|
import ca.uhn.fhir.rest.gclient.*;
|
||||||
import ca.uhn.fhir.rest.gclient.ICreate;
|
import ca.uhn.fhir.rest.method.*;
|
||||||
import ca.uhn.fhir.rest.gclient.ICreateTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ICreateWithQuery;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ICreateWithQueryTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ICriterion;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ICriterionInternal;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IDelete;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IDeleteTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IDeleteWithQuery;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IDeleteWithQueryTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IFetchConformanceTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IFetchConformanceUntyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IGetPage;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IGetPageTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IGetPageUntyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IGetTags;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IHistory;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IHistoryTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IHistoryUntyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IMeta;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IMetaAddOrDeleteSourced;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IMetaAddOrDeleteUnsourced;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IMetaGetUnsourced;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IOperation;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IOperationUnnamed;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IOperationUntyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IOperationUntypedWithInput;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IOperationUntypedWithInputAndPartialOutput;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IParam;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IPatch;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IPatchExecutable;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IPatchTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IPatchWithQuery;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IPatchWithQueryTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IQuery;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IRead;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IReadExecutable;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IReadIfNoneMatch;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IReadTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ISort;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ITransaction;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ITransactionTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUntypedQuery;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUpdate;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUpdateExecutable;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUpdateTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUpdateWithQuery;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUpdateWithQueryTyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IValidate;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IValidateUntyped;
|
|
||||||
import ca.uhn.fhir.rest.method.DeleteMethodBinding;
|
|
||||||
import ca.uhn.fhir.rest.method.HistoryMethodBinding;
|
|
||||||
import ca.uhn.fhir.rest.method.HttpDeleteClientInvocation;
|
|
||||||
import ca.uhn.fhir.rest.method.HttpGetClientInvocation;
|
|
||||||
import ca.uhn.fhir.rest.method.HttpSimpleGetClientInvocation;
|
|
||||||
import ca.uhn.fhir.rest.method.IClientResponseHandler;
|
|
||||||
import ca.uhn.fhir.rest.method.MethodUtil;
|
|
||||||
import ca.uhn.fhir.rest.method.OperationMethodBinding;
|
|
||||||
import ca.uhn.fhir.rest.method.ReadMethodBinding;
|
|
||||||
import ca.uhn.fhir.rest.method.SearchMethodBinding;
|
|
||||||
import ca.uhn.fhir.rest.method.SearchStyleEnum;
|
|
||||||
import ca.uhn.fhir.rest.method.SortParameter;
|
|
||||||
import ca.uhn.fhir.rest.method.TransactionMethodBinding;
|
|
||||||
import ca.uhn.fhir.rest.method.ValidateMethodBindingDstu1;
|
|
||||||
import ca.uhn.fhir.rest.method.ValidateMethodBindingDstu2Plus;
|
|
||||||
import ca.uhn.fhir.rest.param.DateParam;
|
import ca.uhn.fhir.rest.param.DateParam;
|
||||||
import ca.uhn.fhir.rest.param.DateRangeParam;
|
import ca.uhn.fhir.rest.param.DateRangeParam;
|
||||||
import ca.uhn.fhir.rest.param.TokenParam;
|
import ca.uhn.fhir.rest.param.TokenParam;
|
||||||
|
@ -535,26 +440,6 @@ public class GenericClient extends BaseClient implements IGenericClient {
|
||||||
return new PatchInternal();
|
return new PatchInternal();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public MethodOutcome patch(IdDt theIdDt, IBaseResource theResource) {
|
|
||||||
BaseHttpClientInvocation invocation = MethodUtil.createUpdateInvocation(theResource, null, theIdDt, myContext);
|
|
||||||
if (isKeepResponses()) {
|
|
||||||
myLastRequest = invocation.asHttpRequest(getServerBase(), createExtraParams(), getEncoding(), isPrettyPrint());
|
|
||||||
}
|
|
||||||
|
|
||||||
RuntimeResourceDefinition def = myContext.getResourceDefinition(theResource);
|
|
||||||
final String resourceName = def.getName();
|
|
||||||
|
|
||||||
OutcomeResponseHandler binding = new OutcomeResponseHandler(resourceName);
|
|
||||||
MethodOutcome resp = invokeClient(myContext, binding, invocation, myLogRequestAndResponse);
|
|
||||||
return resp;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public MethodOutcome patch(String theId, IBaseResource theResource) {
|
|
||||||
return update(new IdDt(theId), theResource);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IUpdate update() {
|
public IUpdate update() {
|
||||||
return new UpdateInternal();
|
return new UpdateInternal();
|
||||||
|
@ -1986,7 +1871,7 @@ public class GenericClient extends BaseClient implements IGenericClient {
|
||||||
|
|
||||||
for (Collection<String> profileUris : myProfiles) {
|
for (Collection<String> profileUris : myProfiles) {
|
||||||
StringBuilder builder = new StringBuilder();
|
StringBuilder builder = new StringBuilder();
|
||||||
for (Iterator<String> profileItr = profileUris.iterator(); profileItr.hasNext(); ) {
|
for (Iterator<String> profileItr = profileUris.iterator(); profileItr.hasNext();) {
|
||||||
builder.append(profileItr.next());
|
builder.append(profileItr.next());
|
||||||
if (profileItr.hasNext()) {
|
if (profileItr.hasNext()) {
|
||||||
builder.append(',');
|
builder.append(',');
|
||||||
|
@ -2359,16 +2244,15 @@ public class GenericClient extends BaseClient implements IGenericClient {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private class PatchInternal extends BaseClientExecutable<IPatchExecutable, MethodOutcome> implements IPatch, IPatchTyped, IPatchExecutable, IPatchWithQuery, IPatchWithQueryTyped {
|
private class PatchInternal extends BaseClientExecutable<IPatchExecutable, MethodOutcome> implements IPatch, IPatchWithBody, IPatchExecutable, IPatchWithQuery, IPatchWithQueryTyped {
|
||||||
|
|
||||||
private CriterionList myCriterionList;
|
private CriterionList myCriterionList;
|
||||||
private IIdType myId;
|
private IIdType myId;
|
||||||
private PreferReturnEnum myPrefer;
|
private PreferReturnEnum myPrefer;
|
||||||
private IBaseResource myResource;
|
|
||||||
private String myResourceBody;
|
|
||||||
private String mySearchUrl;
|
|
||||||
private PatchTypeEnum myPatchType;
|
private PatchTypeEnum myPatchType;
|
||||||
private String myPatchBody;
|
private String myPatchBody;
|
||||||
|
private String mySearchUrl;
|
||||||
|
private String myResourceType;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IPatchWithQueryTyped and(ICriterion<?> theCriterion) {
|
public IPatchWithQueryTyped and(ICriterion<?> theCriterion) {
|
||||||
|
@ -2377,28 +2261,22 @@ public class GenericClient extends BaseClient implements IGenericClient {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IPatchWithQuery conditional() {
|
public IPatchWithQuery conditional(String theResourceType) {
|
||||||
|
Validate.notBlank(theResourceType, "theResourceType must not be null");
|
||||||
|
myResourceType = theResourceType;
|
||||||
myCriterionList = new CriterionList();
|
myCriterionList = new CriterionList();
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: This is not longer used.. Deprecate it or just remove it?
|
// TODO: This is not longer used.. Deprecate it or just remove it?
|
||||||
@Override
|
@Override
|
||||||
public IPatchTyped conditionalByUrl(String theSearchUrl) {
|
public IPatchWithBody conditionalByUrl(String theSearchUrl) {
|
||||||
mySearchUrl = validateAndEscapeConditionalUrl(theSearchUrl);
|
mySearchUrl = validateAndEscapeConditionalUrl(theSearchUrl);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public MethodOutcome execute() {
|
public MethodOutcome execute() {
|
||||||
if (myResource == null) {
|
|
||||||
myResource = parseResourceBody(myResourceBody);
|
|
||||||
}
|
|
||||||
|
|
||||||
// If an explicit encoding is chosen, we will re-serialize to ensure the right encoding
|
|
||||||
if (getParamEncoding() != null) {
|
|
||||||
myResourceBody = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (myPatchType == null) {
|
if (myPatchType == null) {
|
||||||
throw new InvalidRequestException("No patch type supplied, cannot invoke server");
|
throw new InvalidRequestException("No patch type supplied, cannot invoke server");
|
||||||
|
@ -2407,21 +2285,21 @@ public class GenericClient extends BaseClient implements IGenericClient {
|
||||||
throw new InvalidRequestException("No patch body supplied, cannot invoke server");
|
throw new InvalidRequestException("No patch body supplied, cannot invoke server");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (myId == null) {
|
BaseHttpClientInvocation invocation;
|
||||||
myId = myResource.getIdElement();
|
if (isNotBlank(mySearchUrl)) {
|
||||||
}
|
invocation = MethodUtil.createPatchInvocation(myContext, mySearchUrl, myPatchType, myPatchBody);
|
||||||
|
} else if (myCriterionList != null) {
|
||||||
|
invocation = MethodUtil.createPatchInvocation(myContext, myPatchType, myPatchBody, myResourceType, myCriterionList.toParamList());
|
||||||
|
} else {
|
||||||
if (myId == null || myId.hasIdPart() == false) {
|
if (myId == null || myId.hasIdPart() == false) {
|
||||||
throw new InvalidRequestException("No ID supplied for resource to update, can not invoke server");
|
throw new InvalidRequestException("No ID supplied for resource to patch, can not invoke server");
|
||||||
|
}
|
||||||
|
invocation = MethodUtil.createPatchInvocation(myContext, myId, myPatchType, myPatchBody);
|
||||||
}
|
}
|
||||||
BaseHttpClientInvocation invocation = MethodUtil.createPatchInvocation(myContext, myId, myPatchType, myPatchBody);
|
|
||||||
|
|
||||||
addPreferHeader(myPrefer, invocation);
|
addPreferHeader(myPrefer, invocation);
|
||||||
|
|
||||||
RuntimeResourceDefinition def = myContext.getResourceDefinition(myResource);
|
OutcomeResponseHandler binding = new OutcomeResponseHandler(null, myPrefer);
|
||||||
final String resourceName = def.getName();
|
|
||||||
|
|
||||||
OutcomeResponseHandler binding = new OutcomeResponseHandler(resourceName, myPrefer);
|
|
||||||
|
|
||||||
Map<String, List<String>> params = new HashMap<String, List<String>>();
|
Map<String, List<String>> params = new HashMap<String, List<String>>();
|
||||||
return invoke(params, binding, invocation);
|
return invoke(params, binding, invocation);
|
||||||
|
@ -2434,19 +2312,6 @@ public class GenericClient extends BaseClient implements IGenericClient {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public IPatchTyped resource(IBaseResource theResource) {
|
|
||||||
Validate.notNull(theResource, "Resource can not be null");
|
|
||||||
myResource = theResource;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public IPatchTyped resource(String theResourceBody) {
|
|
||||||
Validate.notBlank(theResourceBody, "Body can not be null or blank");
|
|
||||||
myResourceBody = theResourceBody;
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IPatchWithQueryTyped where(ICriterion<?> theCriterion) {
|
public IPatchWithQueryTyped where(ICriterion<?> theCriterion) {
|
||||||
|
@ -2479,15 +2344,28 @@ public class GenericClient extends BaseClient implements IGenericClient {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IPatchTyped patchType(PatchTypeEnum patchType) {
|
public IPatchWithBody withBody(String thePatchBody) {
|
||||||
myPatchType = patchType;
|
Validate.notBlank(thePatchBody, "thePatchBody must not be blank");
|
||||||
|
|
||||||
|
myPatchBody = thePatchBody;
|
||||||
|
|
||||||
|
EncodingEnum encoding = MethodUtil.detectEncodingNoDefault(thePatchBody);
|
||||||
|
if (encoding == EncodingEnum.XML) {
|
||||||
|
myPatchType = PatchTypeEnum.XML_PATCH;
|
||||||
|
} else if (encoding == EncodingEnum.JSON) {
|
||||||
|
myPatchType = PatchTypeEnum.JSON_PATCH;
|
||||||
|
} else {
|
||||||
|
throw new IllegalArgumentException("Unable to determine encoding of patch");
|
||||||
|
}
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IPatchTyped patchBody(String patchBody) {
|
public IPatchWithQuery conditional(Class<? extends IBaseResource> theClass) {
|
||||||
myPatchBody = patchBody;
|
Validate.notNull(theClass, "theClass must not be null");
|
||||||
return this;
|
String resourceType = myContext.getResourceDefinition(theClass).getName();
|
||||||
|
return conditional(resourceType);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,7 +29,6 @@ import org.hl7.fhir.instance.model.api.IBaseResource;
|
||||||
import ca.uhn.fhir.model.api.Bundle;
|
import ca.uhn.fhir.model.api.Bundle;
|
||||||
import ca.uhn.fhir.model.api.IQueryParameterType;
|
import ca.uhn.fhir.model.api.IQueryParameterType;
|
||||||
import ca.uhn.fhir.model.api.IResource;
|
import ca.uhn.fhir.model.api.IResource;
|
||||||
import ca.uhn.fhir.model.base.resource.BaseConformance;
|
|
||||||
import ca.uhn.fhir.model.primitive.DateTimeDt;
|
import ca.uhn.fhir.model.primitive.DateTimeDt;
|
||||||
import ca.uhn.fhir.model.primitive.IdDt;
|
import ca.uhn.fhir.model.primitive.IdDt;
|
||||||
import ca.uhn.fhir.model.primitive.UriDt;
|
import ca.uhn.fhir.model.primitive.UriDt;
|
||||||
|
@ -38,20 +37,7 @@ import ca.uhn.fhir.rest.client.api.IRestfulClient;
|
||||||
import ca.uhn.fhir.rest.client.exceptions.FhirClientConnectionException;
|
import ca.uhn.fhir.rest.client.exceptions.FhirClientConnectionException;
|
||||||
import ca.uhn.fhir.rest.client.exceptions.FhirClientInappropriateForServerException;
|
import ca.uhn.fhir.rest.client.exceptions.FhirClientInappropriateForServerException;
|
||||||
import ca.uhn.fhir.rest.client.interceptor.LoggingInterceptor;
|
import ca.uhn.fhir.rest.client.interceptor.LoggingInterceptor;
|
||||||
import ca.uhn.fhir.rest.gclient.ICreate;
|
import ca.uhn.fhir.rest.gclient.*;
|
||||||
import ca.uhn.fhir.rest.gclient.IDelete;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IFetchConformanceUntyped;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IGetPage;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IGetTags;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IHistory;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IMeta;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IOperation;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IPatch;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IRead;
|
|
||||||
import ca.uhn.fhir.rest.gclient.ITransaction;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUntypedQuery;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IUpdate;
|
|
||||||
import ca.uhn.fhir.rest.gclient.IValidate;
|
|
||||||
|
|
||||||
public interface IGenericClient extends IRestfulClient {
|
public interface IGenericClient extends IRestfulClient {
|
||||||
|
|
||||||
|
@ -254,35 +240,11 @@ public interface IGenericClient extends IRestfulClient {
|
||||||
@Override
|
@Override
|
||||||
void registerInterceptor(IClientInterceptor theInterceptor);
|
void registerInterceptor(IClientInterceptor theInterceptor);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fluent method for the "patch" operation, which performs a logical patch on a server resource
|
* Fluent method for the "patch" operation, which performs a logical patch on a server resource
|
||||||
*/
|
*/
|
||||||
IPatch patch();
|
IPatch patch();
|
||||||
|
|
||||||
/**
|
|
||||||
* Implementation of the "instance patch" method.
|
|
||||||
*
|
|
||||||
* @param theId
|
|
||||||
* The ID to update
|
|
||||||
* @param theResource
|
|
||||||
* The new resource body
|
|
||||||
* @return An outcome containing the results and possibly the new version ID
|
|
||||||
*/
|
|
||||||
MethodOutcome patch(IdDt theId, IBaseResource theResource);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Implementation of the "instance update" method.
|
|
||||||
*
|
|
||||||
* @param theId
|
|
||||||
* The ID to update
|
|
||||||
* @param theResource
|
|
||||||
* The new resource body
|
|
||||||
* @return An outcome containing the results and possibly the new version ID
|
|
||||||
*/
|
|
||||||
MethodOutcome patch(String theId, IBaseResource theResource);
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Search for resources matching a given set of criteria. Searching is a very powerful
|
* Search for resources matching a given set of criteria. Searching is a very powerful
|
||||||
* feature in FHIR with many features for specifying exactly what should be seaerched for
|
* feature in FHIR with many features for specifying exactly what should be seaerched for
|
||||||
|
|
|
@ -1,31 +1,14 @@
|
||||||
package ca.uhn.fhir.rest.gclient;
|
package ca.uhn.fhir.rest.gclient;
|
||||||
|
|
||||||
/*
|
|
||||||
* #%L
|
|
||||||
* HAPI FHIR - Core Library
|
|
||||||
* %%
|
|
||||||
* Copyright (C) 2014 - 2016 University Health Network
|
|
||||||
* %%
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
* #L%
|
|
||||||
*/
|
|
||||||
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseResource;
|
|
||||||
|
|
||||||
public interface IPatch {
|
public interface IPatch {
|
||||||
|
|
||||||
IPatchTyped resource(IBaseResource theResource);
|
/**
|
||||||
|
* The body of the patch document serialized in either XML or JSON which conforms to
|
||||||
IPatchTyped resource(String theResourceBody);
|
* http://jsonpatch.com/ or http://tools.ietf.org/html/rfc5261
|
||||||
|
*
|
||||||
|
* @param thePatchBody
|
||||||
|
* The body of the patch
|
||||||
|
*/
|
||||||
|
IPatchWithBody withBody(String thePatchBody);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,61 +0,0 @@
|
||||||
package ca.uhn.fhir.rest.gclient;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* #%L
|
|
||||||
* HAPI FHIR - Core Library
|
|
||||||
* %%
|
|
||||||
* Copyright (C) 2014 - 2016 University Health Network
|
|
||||||
* %%
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
* #L%
|
|
||||||
*/
|
|
||||||
|
|
||||||
import org.hl7.fhir.instance.model.api.IIdType;
|
|
||||||
|
|
||||||
import ca.uhn.fhir.rest.api.PatchTypeEnum;
|
|
||||||
|
|
||||||
public interface IPatchTyped extends IPatchExecutable {
|
|
||||||
|
|
||||||
IPatchExecutable withId(IIdType theId);
|
|
||||||
|
|
||||||
IPatchExecutable withId(String theId);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Specifies that the update should be performed as a conditional create
|
|
||||||
* against a given search URL.
|
|
||||||
*
|
|
||||||
* @param theSearchUrl The search URL to use. The format of this URL should be of the form <code>[ResourceType]?[Parameters]</code>,
|
|
||||||
* for example: <code>Patient?name=Smith&identifier=13.2.4.11.4%7C847366</code>
|
|
||||||
* @since HAPI 0.9 / FHIR DSTU 2
|
|
||||||
*/
|
|
||||||
IPatchTyped conditionalByUrl(String theSearchUrl);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @since HAPI 0.9 / FHIR DSTU 2
|
|
||||||
*/
|
|
||||||
IPatchWithQuery conditional();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Specifies the format of the patch (either XML or JSON)
|
|
||||||
* @param patchType
|
|
||||||
*/
|
|
||||||
IPatchTyped patchType(PatchTypeEnum patchType);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The body of the patch document serialized in either XML or JSON which conforms to
|
|
||||||
* http://jsonpatch.com/ or http://tools.ietf.org/html/rfc5261
|
|
||||||
* @param patchBody
|
|
||||||
*/
|
|
||||||
IPatchTyped patchBody(String patchBody);
|
|
||||||
|
|
||||||
}
|
|
|
@ -0,0 +1,65 @@
|
||||||
|
package ca.uhn.fhir.rest.gclient;
|
||||||
|
|
||||||
|
import org.hl7.fhir.instance.model.api.IBaseResource;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR - Core Library
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2016 University Health Network
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import org.hl7.fhir.instance.model.api.IIdType;
|
||||||
|
|
||||||
|
public interface IPatchWithBody extends IPatchExecutable {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a conditional URL using fluent constants on resource types
|
||||||
|
*
|
||||||
|
* @param theResourceType
|
||||||
|
* The resource type to patch (e.g. "Patient.class")
|
||||||
|
*/
|
||||||
|
IPatchWithQuery conditional(Class<? extends IBaseResource> theClass);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build a conditional URL using fluent constants on resource types
|
||||||
|
*
|
||||||
|
* @param theResourceType
|
||||||
|
* The resource type to patch (e.g. "Patient")
|
||||||
|
*/
|
||||||
|
IPatchWithQuery conditional(String theResourceType);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies that the update should be performed as a conditional create
|
||||||
|
* against a given search URL.
|
||||||
|
*
|
||||||
|
* @param theSearchUrl
|
||||||
|
* The search URL to use. The format of this URL should be of the form <code>[ResourceType]?[Parameters]</code>,
|
||||||
|
* for example: <code>Patient?name=Smith&identifier=13.2.4.11.4%7C847366</code>
|
||||||
|
*/
|
||||||
|
IPatchExecutable conditionalByUrl(String theSearchUrl);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The resource ID to patch
|
||||||
|
*/
|
||||||
|
IPatchExecutable withId(IIdType theId);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The resource ID to patch
|
||||||
|
*/
|
||||||
|
IPatchExecutable withId(String theId);
|
||||||
|
|
||||||
|
}
|
|
@ -20,6 +20,6 @@ package ca.uhn.fhir.rest.gclient;
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
|
|
||||||
public interface IPatchWithQueryTyped extends IPatchTyped, IPatchWithQuery {
|
public interface IPatchWithQueryTyped extends IPatchWithBody, IPatchWithQuery {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,17 +45,13 @@ public class HttpPatchClientInvocation extends BaseHttpClientInvocation {
|
||||||
myContentType = theContentType;
|
myContentType = theContentType;
|
||||||
myContents = theContents;
|
myContents = theContents;
|
||||||
}
|
}
|
||||||
//
|
|
||||||
// public HttpDeleteClientInvocation(FhirContext theContext, String theSearchUrl) {
|
public HttpPatchClientInvocation(FhirContext theContext, String theUrlPath, String theContentType, String theContents) {
|
||||||
// super(theContext);
|
super(theContext);
|
||||||
// myUrlPath = theSearchUrl;
|
myUrlPath = theUrlPath;
|
||||||
// }
|
myContentType = theContentType;
|
||||||
//
|
myContents = theContents;
|
||||||
// public HttpDeleteClientInvocation(FhirContext theContext, String theResourceType, Map<String, List<String>> theParams) {
|
}
|
||||||
// super(theContext);
|
|
||||||
// myUrlPath = theResourceType;
|
|
||||||
// myParams = theParams;
|
|
||||||
// }
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public IHttpRequest asHttpRequest(String theUrlBase, Map<String, List<String>> theExtraParams, EncodingEnum theEncoding, Boolean thePrettyPrint) {
|
public IHttpRequest asHttpRequest(String theUrlBase, Map<String, List<String>> theExtraParams, EncodingEnum theEncoding, Boolean thePrettyPrint) {
|
||||||
|
|
|
@ -160,24 +160,18 @@ public class MethodUtil {
|
||||||
return PatchMethodBinding.createPatchInvocation(theContext, theId, thePatchType, theBody);
|
return PatchMethodBinding.createPatchInvocation(theContext, theId, thePatchType, theBody);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** End Patch **/
|
public static HttpPatchClientInvocation createPatchInvocation(FhirContext theContext, String theUrl, PatchTypeEnum thePatchType, String theBody) {
|
||||||
|
return PatchMethodBinding.createPatchInvocation(theContext, theUrl, thePatchType, theBody);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static HttpPatchClientInvocation createPatchInvocation(FhirContext theContext, PatchTypeEnum thePatchType, String theBody, String theResourceType, Map<String, List<String>> theMatchParams) {
|
||||||
|
return PatchMethodBinding.createPatchInvocation(theContext, thePatchType, theBody, theResourceType, theMatchParams);
|
||||||
|
}
|
||||||
|
|
||||||
public static HttpPutClientInvocation createUpdateInvocation(FhirContext theContext, IBaseResource theResource, String theResourceBody, Map<String, List<String>> theMatchParams) {
|
public static HttpPutClientInvocation createUpdateInvocation(FhirContext theContext, IBaseResource theResource, String theResourceBody, Map<String, List<String>> theMatchParams) {
|
||||||
StringBuilder b = new StringBuilder();
|
|
||||||
|
|
||||||
String resourceType = theContext.getResourceDefinition(theResource).getName();
|
String resourceType = theContext.getResourceDefinition(theResource).getName();
|
||||||
b.append(resourceType);
|
|
||||||
|
|
||||||
boolean haveQuestionMark = false;
|
StringBuilder b = createUrl(resourceType, theMatchParams);
|
||||||
for (Entry<String, List<String>> nextEntry : theMatchParams.entrySet()) {
|
|
||||||
for (String nextValue : nextEntry.getValue()) {
|
|
||||||
b.append(haveQuestionMark ? '&' : '?');
|
|
||||||
haveQuestionMark = true;
|
|
||||||
b.append(UrlUtil.escape(nextEntry.getKey()));
|
|
||||||
b.append('=');
|
|
||||||
b.append(UrlUtil.escape(nextValue));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpPutClientInvocation retVal;
|
HttpPutClientInvocation retVal;
|
||||||
if (StringUtils.isBlank(theResourceBody)) {
|
if (StringUtils.isBlank(theResourceBody)) {
|
||||||
|
@ -192,6 +186,25 @@ public class MethodUtil {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static StringBuilder createUrl(String theResourceType, Map<String, List<String>> theMatchParams) {
|
||||||
|
StringBuilder b = new StringBuilder();
|
||||||
|
|
||||||
|
b.append(theResourceType);
|
||||||
|
|
||||||
|
boolean haveQuestionMark = false;
|
||||||
|
for (Entry<String, List<String>> nextEntry : theMatchParams.entrySet()) {
|
||||||
|
for (String nextValue : nextEntry.getValue()) {
|
||||||
|
b.append(haveQuestionMark ? '&' : '?');
|
||||||
|
haveQuestionMark = true;
|
||||||
|
b.append(UrlUtil.escape(nextEntry.getKey()));
|
||||||
|
b.append('=');
|
||||||
|
b.append(UrlUtil.escape(nextValue));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return b;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public static HttpPutClientInvocation createUpdateInvocation(FhirContext theContext, IBaseResource theResource, String theResourceBody, String theMatchUrl) {
|
public static HttpPutClientInvocation createUpdateInvocation(FhirContext theContext, IBaseResource theResource, String theResourceBody, String theMatchUrl) {
|
||||||
HttpPutClientInvocation retVal;
|
HttpPutClientInvocation retVal;
|
||||||
if (StringUtils.isBlank(theResourceBody)) {
|
if (StringUtils.isBlank(theResourceBody)) {
|
||||||
|
|
|
@ -22,10 +22,7 @@ package ca.uhn.fhir.rest.method;
|
||||||
|
|
||||||
import java.lang.annotation.Annotation;
|
import java.lang.annotation.Annotation;
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
import java.util.Arrays;
|
import java.util.*;
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.ListIterator;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import org.hl7.fhir.instance.model.api.IIdType;
|
import org.hl7.fhir.instance.model.api.IIdType;
|
||||||
|
|
||||||
|
@ -138,6 +135,11 @@ public class PatchMethodBinding extends BaseOutcomeReturningMethodBindingWithRes
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static HttpPatchClientInvocation createPatchInvocation(FhirContext theContext, String theUrlPath, PatchTypeEnum thePatchType, String theBody) {
|
||||||
|
HttpPatchClientInvocation retVal = new HttpPatchClientInvocation(theContext, theUrlPath, thePatchType.getContentType(), theBody);
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void addParametersForServerRequest(RequestDetails theRequest, Object[] theParams) {
|
protected void addParametersForServerRequest(RequestDetails theRequest, Object[] theParams) {
|
||||||
IIdType id = theRequest.getId();
|
IIdType id = theRequest.getId();
|
||||||
|
@ -150,4 +152,11 @@ public class PatchMethodBinding extends BaseOutcomeReturningMethodBindingWithRes
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static HttpPatchClientInvocation createPatchInvocation(FhirContext theContext, PatchTypeEnum thePatchType, String theBody, String theResourceType, Map<String, List<String>> theMatchParams) {
|
||||||
|
StringBuilder urlBuilder = MethodUtil.createUrl(theResourceType, theMatchParams);
|
||||||
|
String url = urlBuilder.toString();
|
||||||
|
HttpPatchClientInvocation retVal = new HttpPatchClientInvocation(theContext, url, thePatchType.getContentType(), theBody);
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -29,6 +29,8 @@ import org.apache.http.client.HttpClient;
|
||||||
import org.apache.http.client.methods.HttpEntityEnclosingRequestBase;
|
import org.apache.http.client.methods.HttpEntityEnclosingRequestBase;
|
||||||
import org.apache.http.client.methods.HttpPut;
|
import org.apache.http.client.methods.HttpPut;
|
||||||
import org.apache.http.client.methods.HttpUriRequest;
|
import org.apache.http.client.methods.HttpUriRequest;
|
||||||
|
import org.apache.http.entity.ContentType;
|
||||||
|
import org.apache.http.entity.StringEntity;
|
||||||
import org.apache.http.message.BasicHeader;
|
import org.apache.http.message.BasicHeader;
|
||||||
import org.apache.http.message.BasicStatusLine;
|
import org.apache.http.message.BasicStatusLine;
|
||||||
import org.hl7.fhir.dstu3.model.*;
|
import org.hl7.fhir.dstu3.model.*;
|
||||||
|
@ -118,6 +120,246 @@ public class GenericClientDstu3Test {
|
||||||
return capt;
|
return capt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPatchJsonByIdString() throws Exception {
|
||||||
|
OperationOutcome conf = new OperationOutcome();
|
||||||
|
conf.getText().setDivAsString("OK!");
|
||||||
|
final String respString = ourCtx.newJsonParser().encodeResourceToString(conf);
|
||||||
|
|
||||||
|
ArgumentCaptor<HttpUriRequest> capt = ArgumentCaptor.forClass(HttpUriRequest.class);
|
||||||
|
when(myHttpClient.execute(capt.capture())).thenReturn(myHttpResponse);
|
||||||
|
when(myHttpResponse.getStatusLine()).thenReturn(new BasicStatusLine(new ProtocolVersion("HTTP", 1, 1), 200, "OK"));
|
||||||
|
when(myHttpResponse.getEntity().getContentType()).thenReturn(new BasicHeader("content-type", Constants.CT_FHIR_JSON + "; charset=UTF-8"));
|
||||||
|
when(myHttpResponse.getEntity().getContent()).then(new Answer<InputStream>() {
|
||||||
|
@Override
|
||||||
|
public InputStream answer(InvocationOnMock theInvocation) throws Throwable {
|
||||||
|
return new ReaderInputStream(new StringReader(respString), Charset.forName("UTF-8"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
IGenericClient client = ourCtx.newRestfulGenericClient("http://example.com/fhir");
|
||||||
|
int idx = 0;
|
||||||
|
|
||||||
|
String patch = "[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]";
|
||||||
|
|
||||||
|
MethodOutcome outcome = client
|
||||||
|
.patch()
|
||||||
|
.withBody(patch)
|
||||||
|
.withId("Patient/123")
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
assertEquals("http://example.com/fhir/Patient/123", UrlUtil.unescape(capt.getAllValues().get(idx).getURI().toString()));
|
||||||
|
assertEquals("PATCH", capt.getAllValues().get(0).getRequestLine().getMethod());
|
||||||
|
assertEquals(patch, extractBodyAsString(capt));
|
||||||
|
assertEquals(Constants.CT_JSON_PATCH, capt.getAllValues().get(idx).getFirstHeader("Content-Type").getValue().replaceAll(";.*", ""));
|
||||||
|
idx++;
|
||||||
|
|
||||||
|
OperationOutcome oo = (OperationOutcome) outcome.getOperationOutcome();
|
||||||
|
assertThat(oo.getText().getDivAsString(), containsString("OK!"));
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testPatchJsonByIdType() throws Exception {
|
||||||
|
OperationOutcome conf = new OperationOutcome();
|
||||||
|
conf.getText().setDivAsString("OK!");
|
||||||
|
final String respString = ourCtx.newJsonParser().encodeResourceToString(conf);
|
||||||
|
|
||||||
|
ArgumentCaptor<HttpUriRequest> capt = ArgumentCaptor.forClass(HttpUriRequest.class);
|
||||||
|
when(myHttpClient.execute(capt.capture())).thenReturn(myHttpResponse);
|
||||||
|
when(myHttpResponse.getStatusLine()).thenReturn(new BasicStatusLine(new ProtocolVersion("HTTP", 1, 1), 200, "OK"));
|
||||||
|
when(myHttpResponse.getEntity().getContentType()).thenReturn(new BasicHeader("content-type", Constants.CT_FHIR_JSON + "; charset=UTF-8"));
|
||||||
|
when(myHttpResponse.getEntity().getContent()).then(new Answer<InputStream>() {
|
||||||
|
@Override
|
||||||
|
public InputStream answer(InvocationOnMock theInvocation) throws Throwable {
|
||||||
|
return new ReaderInputStream(new StringReader(respString), Charset.forName("UTF-8"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
IGenericClient client = ourCtx.newRestfulGenericClient("http://example.com/fhir");
|
||||||
|
int idx = 0;
|
||||||
|
|
||||||
|
String patch = "[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]";
|
||||||
|
|
||||||
|
MethodOutcome outcome = client
|
||||||
|
.patch()
|
||||||
|
.withBody(patch)
|
||||||
|
.withId(new IdType("http://localhost/fhir/Patient/123/_history/234"))
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
assertEquals("http://example.com/fhir/Patient/123", UrlUtil.unescape(capt.getAllValues().get(idx).getURI().toString()));
|
||||||
|
assertEquals("PATCH", capt.getAllValues().get(0).getRequestLine().getMethod());
|
||||||
|
assertEquals(patch, extractBodyAsString(capt));
|
||||||
|
assertEquals(Constants.CT_JSON_PATCH, capt.getAllValues().get(idx).getFirstHeader("Content-Type").getValue().replaceAll(";.*", ""));
|
||||||
|
idx++;
|
||||||
|
|
||||||
|
OperationOutcome oo = (OperationOutcome) outcome.getOperationOutcome();
|
||||||
|
assertThat(oo.getText().getDivAsString(), containsString("OK!"));
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testPatchJsonByConditionalString() throws Exception {
|
||||||
|
OperationOutcome conf = new OperationOutcome();
|
||||||
|
conf.getText().setDivAsString("OK!");
|
||||||
|
final String respString = ourCtx.newJsonParser().encodeResourceToString(conf);
|
||||||
|
|
||||||
|
ArgumentCaptor<HttpUriRequest> capt = ArgumentCaptor.forClass(HttpUriRequest.class);
|
||||||
|
when(myHttpClient.execute(capt.capture())).thenReturn(myHttpResponse);
|
||||||
|
when(myHttpResponse.getStatusLine()).thenReturn(new BasicStatusLine(new ProtocolVersion("HTTP", 1, 1), 200, "OK"));
|
||||||
|
when(myHttpResponse.getEntity().getContentType()).thenReturn(new BasicHeader("content-type", Constants.CT_FHIR_JSON + "; charset=UTF-8"));
|
||||||
|
when(myHttpResponse.getEntity().getContent()).then(new Answer<InputStream>() {
|
||||||
|
@Override
|
||||||
|
public InputStream answer(InvocationOnMock theInvocation) throws Throwable {
|
||||||
|
return new ReaderInputStream(new StringReader(respString), Charset.forName("UTF-8"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
IGenericClient client = ourCtx.newRestfulGenericClient("http://example.com/fhir");
|
||||||
|
int idx = 0;
|
||||||
|
|
||||||
|
String patch = "[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]";
|
||||||
|
|
||||||
|
MethodOutcome outcome = client
|
||||||
|
.patch()
|
||||||
|
.withBody(patch)
|
||||||
|
.conditionalByUrl("Patient?foo=bar")
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
assertEquals("http://example.com/fhir/Patient?foo=bar", UrlUtil.unescape(capt.getAllValues().get(idx).getURI().toString()));
|
||||||
|
assertEquals("PATCH", capt.getAllValues().get(0).getRequestLine().getMethod());
|
||||||
|
assertEquals(patch, extractBodyAsString(capt));
|
||||||
|
assertEquals(Constants.CT_JSON_PATCH, capt.getAllValues().get(idx).getFirstHeader("Content-Type").getValue().replaceAll(";.*", ""));
|
||||||
|
idx++;
|
||||||
|
|
||||||
|
OperationOutcome oo = (OperationOutcome) outcome.getOperationOutcome();
|
||||||
|
assertThat(oo.getText().getDivAsString(), containsString("OK!"));
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testPatchJsonByConditionalParam() throws Exception {
|
||||||
|
OperationOutcome conf = new OperationOutcome();
|
||||||
|
conf.getText().setDivAsString("OK!");
|
||||||
|
final String respString = ourCtx.newJsonParser().encodeResourceToString(conf);
|
||||||
|
|
||||||
|
ArgumentCaptor<HttpUriRequest> capt = ArgumentCaptor.forClass(HttpUriRequest.class);
|
||||||
|
when(myHttpClient.execute(capt.capture())).thenReturn(myHttpResponse);
|
||||||
|
when(myHttpResponse.getStatusLine()).thenReturn(new BasicStatusLine(new ProtocolVersion("HTTP", 1, 1), 200, "OK"));
|
||||||
|
when(myHttpResponse.getEntity().getContentType()).thenReturn(new BasicHeader("content-type", Constants.CT_FHIR_JSON + "; charset=UTF-8"));
|
||||||
|
when(myHttpResponse.getEntity().getContent()).then(new Answer<InputStream>() {
|
||||||
|
@Override
|
||||||
|
public InputStream answer(InvocationOnMock theInvocation) throws Throwable {
|
||||||
|
return new ReaderInputStream(new StringReader(respString), Charset.forName("UTF-8"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
IGenericClient client = ourCtx.newRestfulGenericClient("http://example.com/fhir");
|
||||||
|
int idx = 0;
|
||||||
|
|
||||||
|
String patch = "[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]";
|
||||||
|
|
||||||
|
MethodOutcome outcome = client
|
||||||
|
.patch()
|
||||||
|
.withBody(patch)
|
||||||
|
.conditional("Patient").where(Patient.NAME.matches().value("TEST"))
|
||||||
|
.and(Patient.FAMILY.matches().value("TEST2"))
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
assertEquals("http://example.com/fhir/Patient?name=TEST&family=TEST2", UrlUtil.unescape(capt.getAllValues().get(idx).getURI().toString()));
|
||||||
|
assertEquals("PATCH", capt.getAllValues().get(0).getRequestLine().getMethod());
|
||||||
|
assertEquals(patch, extractBodyAsString(capt));
|
||||||
|
assertEquals(Constants.CT_JSON_PATCH, capt.getAllValues().get(idx).getFirstHeader("Content-Type").getValue().replaceAll(";.*", ""));
|
||||||
|
idx++;
|
||||||
|
|
||||||
|
OperationOutcome oo = (OperationOutcome) outcome.getOperationOutcome();
|
||||||
|
assertThat(oo.getText().getDivAsString(), containsString("OK!"));
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testPatchJsonByConditionalParamResourceType() throws Exception {
|
||||||
|
OperationOutcome conf = new OperationOutcome();
|
||||||
|
conf.getText().setDivAsString("OK!");
|
||||||
|
final String respString = ourCtx.newJsonParser().encodeResourceToString(conf);
|
||||||
|
|
||||||
|
ArgumentCaptor<HttpUriRequest> capt = ArgumentCaptor.forClass(HttpUriRequest.class);
|
||||||
|
when(myHttpClient.execute(capt.capture())).thenReturn(myHttpResponse);
|
||||||
|
when(myHttpResponse.getStatusLine()).thenReturn(new BasicStatusLine(new ProtocolVersion("HTTP", 1, 1), 200, "OK"));
|
||||||
|
when(myHttpResponse.getEntity().getContentType()).thenReturn(new BasicHeader("content-type", Constants.CT_FHIR_JSON + "; charset=UTF-8"));
|
||||||
|
when(myHttpResponse.getEntity().getContent()).then(new Answer<InputStream>() {
|
||||||
|
@Override
|
||||||
|
public InputStream answer(InvocationOnMock theInvocation) throws Throwable {
|
||||||
|
return new ReaderInputStream(new StringReader(respString), Charset.forName("UTF-8"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
IGenericClient client = ourCtx.newRestfulGenericClient("http://example.com/fhir");
|
||||||
|
int idx = 0;
|
||||||
|
|
||||||
|
String patch = "[ { \"op\":\"replace\", \"path\":\"/active\", \"value\":false } ]";
|
||||||
|
|
||||||
|
MethodOutcome outcome = client
|
||||||
|
.patch()
|
||||||
|
.withBody(patch)
|
||||||
|
.conditional(Patient.class).where(Patient.NAME.matches().value("TEST"))
|
||||||
|
.and(Patient.FAMILY.matches().value("TEST2"))
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
assertEquals("http://example.com/fhir/Patient?name=TEST&family=TEST2", UrlUtil.unescape(capt.getAllValues().get(idx).getURI().toString()));
|
||||||
|
assertEquals("PATCH", capt.getAllValues().get(0).getRequestLine().getMethod());
|
||||||
|
assertEquals(patch, extractBodyAsString(capt));
|
||||||
|
assertEquals(Constants.CT_JSON_PATCH, capt.getAllValues().get(idx).getFirstHeader("Content-Type").getValue().replaceAll(";.*", ""));
|
||||||
|
idx++;
|
||||||
|
|
||||||
|
OperationOutcome oo = (OperationOutcome) outcome.getOperationOutcome();
|
||||||
|
assertThat(oo.getText().getDivAsString(), containsString("OK!"));
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testPatchXmlByIdString() throws Exception {
|
||||||
|
OperationOutcome conf = new OperationOutcome();
|
||||||
|
conf.getText().setDivAsString("OK!");
|
||||||
|
final String respString = ourCtx.newJsonParser().encodeResourceToString(conf);
|
||||||
|
|
||||||
|
ArgumentCaptor<HttpUriRequest> capt = ArgumentCaptor.forClass(HttpUriRequest.class);
|
||||||
|
when(myHttpClient.execute(capt.capture())).thenReturn(myHttpResponse);
|
||||||
|
when(myHttpResponse.getStatusLine()).thenReturn(new BasicStatusLine(new ProtocolVersion("HTTP", 1, 1), 200, "OK"));
|
||||||
|
when(myHttpResponse.getEntity().getContentType()).thenReturn(new BasicHeader("content-type", Constants.CT_FHIR_JSON + "; charset=UTF-8"));
|
||||||
|
when(myHttpResponse.getEntity().getContent()).then(new Answer<InputStream>() {
|
||||||
|
@Override
|
||||||
|
public InputStream answer(InvocationOnMock theInvocation) throws Throwable {
|
||||||
|
return new ReaderInputStream(new StringReader(respString), Charset.forName("UTF-8"));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
IGenericClient client = ourCtx.newRestfulGenericClient("http://example.com/fhir");
|
||||||
|
int idx = 0;
|
||||||
|
|
||||||
|
String patch = "<?xml version=\"1.0\" encoding=\"UTF-8\"?><diff xmlns:fhir=\"http://hl7.org/fhir\"><replace sel=\"fhir:Patient/fhir:active/@value\">false</replace></diff>";
|
||||||
|
|
||||||
|
MethodOutcome outcome = client
|
||||||
|
.patch()
|
||||||
|
.withBody(patch)
|
||||||
|
.withId("Patient/123")
|
||||||
|
.execute();
|
||||||
|
|
||||||
|
assertEquals("http://example.com/fhir/Patient/123", UrlUtil.unescape(capt.getAllValues().get(idx).getURI().toString()));
|
||||||
|
assertEquals("PATCH", capt.getAllValues().get(0).getRequestLine().getMethod());
|
||||||
|
assertEquals(patch, extractBodyAsString(capt));
|
||||||
|
assertEquals(Constants.CT_XML_PATCH, capt.getAllValues().get(idx).getFirstHeader("Content-Type").getValue().replaceAll(";.*", ""));
|
||||||
|
idx++;
|
||||||
|
|
||||||
|
OperationOutcome oo = (OperationOutcome) outcome.getOperationOutcome();
|
||||||
|
assertThat(oo.getText().getDivAsString(), containsString("OK!"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPatchInvalid() throws Exception {
|
||||||
|
IGenericClient client = ourCtx.newRestfulGenericClient("http://example.com/fhir");
|
||||||
|
|
||||||
|
try {
|
||||||
|
client
|
||||||
|
.patch()
|
||||||
|
.withBody("AA")
|
||||||
|
.withId("Patient/123")
|
||||||
|
.execute();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
assertEquals("Unable to determine encoding of patch", e.getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testAcceptHeaderWithEncodingSpecified() throws Exception {
|
public void testAcceptHeaderWithEncodingSpecified() throws Exception {
|
||||||
final String msg = "{\"resourceType\":\"Bundle\",\"id\":null,\"base\":\"http://localhost:57931/fhir/contextDev\",\"total\":1,\"link\":[{\"relation\":\"self\",\"url\":\"http://localhost:57931/fhir/contextDev/Patient?identifier=urn%3AMultiFhirVersionTest%7CtestSubmitPatient01&_format=json\"}],\"entry\":[{\"resource\":{\"resourceType\":\"Patient\",\"id\":\"1\",\"meta\":{\"versionId\":\"1\",\"lastUpdated\":\"2014-12-20T18:41:29.706-05:00\"},\"identifier\":[{\"system\":\"urn:MultiFhirVersionTest\",\"value\":\"testSubmitPatient01\"}]}}]}";
|
final String msg = "{\"resourceType\":\"Bundle\",\"id\":null,\"base\":\"http://localhost:57931/fhir/contextDev\",\"total\":1,\"link\":[{\"relation\":\"self\",\"url\":\"http://localhost:57931/fhir/contextDev/Patient?identifier=urn%3AMultiFhirVersionTest%7CtestSubmitPatient01&_format=json\"}],\"entry\":[{\"resource\":{\"resourceType\":\"Patient\",\"id\":\"1\",\"meta\":{\"versionId\":\"1\",\"lastUpdated\":\"2014-12-20T18:41:29.706-05:00\"},\"identifier\":[{\"system\":\"urn:MultiFhirVersionTest\",\"value\":\"testSubmitPatient01\"}]}}]}";
|
||||||
|
|
Loading…
Reference in New Issue