diff --git a/RELEASE_NOTES.md b/RELEASE_NOTES.md
index 8c6f2a408..98dc13da7 100644
--- a/RELEASE_NOTES.md
+++ b/RELEASE_NOTES.md
@@ -8,7 +8,9 @@
* Fix special case r5 loading of terminology to fix validation error on ExampleScenario
* Improve handling of JSON format errors
* Fix bug where extension slices defined in other profiles are not found when processing slices based on extension
-* Validate slice expressions where possible
+* Validate fhirpath expression in slice discriminators
+* Fix slicing by type and profile to allow multiple options per slice
+* List measure choices when a match by version can't be found
## Other code changes
diff --git a/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/ICD11Generator.java b/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/ICD11Generator.java
index 2999ec012..0e7a391d2 100644
--- a/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/ICD11Generator.java
+++ b/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/ICD11Generator.java
@@ -1,6 +1,5 @@
package org.hl7.fhir.convertors.misc;
-import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Date;
import java.util.HashSet;
@@ -395,7 +394,7 @@ public class ICD11Generator {
private JsonObject fetchJson(String source) throws IOException {
- HTTPResult res = ManagedWebAccess.builder().withAccept("application/json").withHeader("API-Version", "v2").withHeader("Accept-Language", "en").get(source);
+ HTTPResult res = ManagedWebAccess.accessor().withHeader("API-Version", "v2").withHeader("Accept-Language", "en").get(source,"application/json");
res.checkThrowException();
return JsonParser.parseObject(res.getContent());
}
diff --git a/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/VSACImporter.java b/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/VSACImporter.java
index cfe79e773..13fb36c81 100644
--- a/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/VSACImporter.java
+++ b/org.hl7.fhir.convertors/src/main/java/org/hl7/fhir/convertors/misc/VSACImporter.java
@@ -32,6 +32,7 @@ import org.hl7.fhir.r4.terminologies.JurisdictionUtilities;
import org.hl7.fhir.utilities.CSVReader;
import org.hl7.fhir.utilities.Utilities;
import org.hl7.fhir.utilities.filesystem.ManagedFileAccess;
+import org.hl7.fhir.utilities.http.ManagedWebAccess;
import org.hl7.fhir.utilities.json.model.JsonArray;
import org.hl7.fhir.utilities.json.model.JsonObject;
import org.hl7.fhir.utilities.json.model.JsonProperty;
@@ -45,17 +46,16 @@ public class VSACImporter extends OIDBasedValueSetImporter {
public static void main(String[] args) throws FHIRException, IOException, ParseException, URISyntaxException {
VSACImporter self = new VSACImporter();
- self.process(args[0], args[1], args[2], "true".equals(args[3]), "true".equals(args[4]));
+ self.process(args[0], args[1], "true".equals(args[2]), "true".equals(args[3]));
}
- private void process(String source, String dest, String apiKey, boolean onlyNew, boolean onlyActive) throws FHIRException, IOException, URISyntaxException {
+ private void process(String source, String dest, boolean onlyNew, boolean onlyActive) throws FHIRException, IOException, URISyntaxException {
CSVReader csv = new CSVReader(ManagedFileAccess.inStream(source));
csv.readHeaders();
Map errs = new HashMap<>();
+ ManagedWebAccess.loadFromFHIRSettings();
FHIRToolingClient fhirToolingClient = new FHIRToolingClient("https://cts.nlm.nih.gov/fhir", "fhir/vsac");
- fhirToolingClient.setUsername("apikey");
- fhirToolingClient.setPassword(apiKey);
fhirToolingClient.setTimeoutNormal(30000);
fhirToolingClient.setTimeoutExpand(30000);
@@ -121,6 +121,7 @@ public class VSACImporter extends OIDBasedValueSetImporter {
oo.addIssue().setSeverity(IssueSeverity.ERROR).setCode(IssueType.EXCEPTION).setDiagnostics(errs.get(oid)).addLocation(oid);
}
new JsonParser().setOutputStyle(OutputStyle.PRETTY).compose(ManagedFileAccess.outStream(Utilities.path(dest, "other", "OperationOutcome-vsac-errors.json")), oo);
+ System.out.println();
System.out.println("Done. " + i + " ValueSets in "+Utilities.describeDuration(System.currentTimeMillis() - tt));
}
diff --git a/org.hl7.fhir.dstu2/pom.xml b/org.hl7.fhir.dstu2/pom.xml
index 1dc4fd998..881479b40 100644
--- a/org.hl7.fhir.dstu2/pom.xml
+++ b/org.hl7.fhir.dstu2/pom.xml
@@ -28,6 +28,12 @@
org.hl7.fhir.utilities
+
+ org.projectlombok
+ lombok
+ provided
+
+
org.fhir
@@ -81,6 +87,24 @@
+
+ org.assertj
+ assertj-core
+ test
+
+
+
+ com.squareup.okhttp3
+ mockwebserver
+ true
+ test
+
+
+ com.squareup.okio
+ okio
+ true
+ test
+
com.fasterxml.jackson.core
jackson-databind
diff --git a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ClientUtils.java b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ClientUtils.java
index e4261fd9c..c8dcb0bfc 100644
--- a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ClientUtils.java
+++ b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ClientUtils.java
@@ -33,42 +33,21 @@ import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStreamWriter;
-import java.io.UnsupportedEncodingException;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URLConnection;
-import java.nio.charset.StandardCharsets;
import java.text.ParseException;
import java.text.SimpleDateFormat;
-import java.util.ArrayList;
-import java.util.Calendar;
-import java.util.Date;
-import java.util.List;
-import java.util.Locale;
-import java.util.Map;
+import java.util.*;
+import java.util.concurrent.TimeUnit;
+
+import lombok.Getter;
+import lombok.Setter;
-import org.apache.commons.codec.binary.Base64;
import org.apache.commons.io.IOUtils;
import org.apache.commons.lang3.StringUtils;
-import org.apache.http.Header;
-import org.apache.http.HttpEntityEnclosingRequest;
-import org.apache.http.HttpHost;
-import org.apache.http.HttpRequest;
-import org.apache.http.HttpResponse;
-import org.apache.http.client.HttpClient;
-import org.apache.http.client.methods.HttpDelete;
-import org.apache.http.client.methods.HttpEntityEnclosingRequestBase;
-import org.apache.http.client.methods.HttpGet;
-import org.apache.http.client.methods.HttpOptions;
-import org.apache.http.client.methods.HttpPost;
-import org.apache.http.client.methods.HttpPut;
-import org.apache.http.client.methods.HttpUriRequest;
-import org.apache.http.conn.params.ConnRoutePNames;
-import org.apache.http.entity.ByteArrayEntity;
-import org.apache.http.impl.client.DefaultHttpClient;
-import org.apache.http.params.HttpConnectionParams;
-import org.apache.http.params.HttpParams;
+
import org.hl7.fhir.dstu2.formats.IParser;
import org.hl7.fhir.dstu2.formats.IParser.OutputStyle;
import org.hl7.fhir.dstu2.formats.JsonParser;
@@ -84,8 +63,11 @@ import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.utilities.MimeType;
import org.hl7.fhir.utilities.ToolingClientLogger;
import org.hl7.fhir.utilities.Utilities;
+import org.hl7.fhir.utilities.http.*;
import org.hl7.fhir.utilities.settings.FhirSettings;
+import javax.annotation.Nonnull;
+
/**
* Helper class handling lower level HTTP transport concerns. TODO Document
* methods.
@@ -93,51 +75,35 @@ import org.hl7.fhir.utilities.settings.FhirSettings;
* @author Claude Nanjo
*/
public class ClientUtils {
-
+ protected static final String LOCATION_HEADER = "location";
+ protected static final String CONTENT_LOCATION_HEADER = "content-location";
public static final String DEFAULT_CHARSET = "UTF-8";
- public static final String HEADER_LOCATION = "location";
+
private static boolean debugging = false;
- private HttpHost proxy;
+ @Getter
+ @Setter
private int timeout = 5000;
- private String username;
- private String password;
+
+ @Setter
+ @Getter
private ToolingClientLogger logger;
+
+ @Setter
+ @Getter
private int retryCount;
+
+ @Getter
+ @Setter
private String userAgent;
- private String acceptLang;
- private String contentLang;
+ @Setter
+ private String acceptLanguage;
+ @Setter
+ private String contentLanguage;
+ private final TimeUnit timeoutUnit = TimeUnit.MILLISECONDS;
- public HttpHost getProxy() {
- return proxy;
- }
-
- public void setProxy(HttpHost proxy) {
- this.proxy = proxy;
- }
-
- public int getTimeout() {
- return timeout;
- }
-
- public void setTimeout(int timeout) {
- this.timeout = timeout;
- }
-
- public String getUsername() {
- return username;
- }
-
- public void setUsername(String username) {
- this.username = username;
- }
-
- public String getPassword() {
- return password;
- }
-
- public void setPassword(String password) {
- this.password = password;
+ protected ManagedFhirWebAccessor getManagedWebAccessor() {
+ return ManagedWebAccess.fhirAccessor().withRetries(retryCount).withTimeout(timeout, timeoutUnit).withLogger(logger);
}
public ResourceRequest issueOptionsRequest(URI optionsUri, String resourceFormat,
@@ -146,8 +112,10 @@ public class ClientUtils {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpOptions options = new HttpOptions(optionsUri);
- return issueResourceRequest(resourceFormat, options, timeoutLoading);
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.OPTIONS)
+ .withUrl(optionsUri.toString());
+ return issueResourceRequest(resourceFormat, httpRequest, timeoutLoading);
}
public ResourceRequest issueGetResourceRequest(URI resourceUri, String resourceFormat,
@@ -155,17 +123,23 @@ public class ClientUtils {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpGet httpget = new HttpGet(resourceUri);
- return issueResourceRequest(resourceFormat, httpget, timeoutLoading);
+
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.GET)
+ .withUrl(resourceUri.toString());
+ return issueResourceRequest(resourceFormat, httpRequest, timeoutLoading);
}
public ResourceRequest issuePutRequest(URI resourceUri, byte[] payload, String resourceFormat,
- List headers, int timeoutLoading) {
+ Iterable headers, int timeoutLoading) {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpPut httpPut = new HttpPut(resourceUri);
- return issueResourceRequest(resourceFormat, httpPut, payload, headers, timeoutLoading);
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.PUT)
+ .withUrl(resourceUri.toString())
+ .withBody(payload);
+ return issueResourceRequest(resourceFormat, httpRequest, headers, timeoutLoading);
}
public ResourceRequest issuePutRequest(URI resourceUri, byte[] payload, String resourceFormat,
@@ -173,17 +147,25 @@ public class ClientUtils {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpPut httpPut = new HttpPut(resourceUri);
- return issueResourceRequest(resourceFormat, httpPut, payload, null, timeoutLoading);
+
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.PUT)
+ .withUrl(resourceUri.toString())
+ .withBody(payload);
+ return issueResourceRequest(resourceFormat, httpRequest, timeoutLoading);
}
public ResourceRequest issuePostRequest(URI resourceUri, byte[] payload,
- String resourceFormat, List headers, int timeoutLoading) {
+ String resourceFormat, Iterable headers, int timeoutLoading) {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpPost httpPost = new HttpPost(resourceUri);
- return issueResourceRequest(resourceFormat, httpPost, payload, headers, timeoutLoading);
+
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withUrl(resourceUri.toString())
+ .withBody(payload);
+ return issueResourceRequest(resourceFormat, httpRequest, headers, timeoutLoading);
}
public ResourceRequest issuePostRequest(URI resourceUri, byte[] payload,
@@ -195,30 +177,26 @@ public class ClientUtils {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpGet httpget = new HttpGet(resourceUri);
- configureFhirRequest(httpget, resourceFormat);
- HttpResponse response = sendRequest(httpget);
- return unmarshalReference(response, resourceFormat);
- }
- private void setAuth(HttpRequest httpget) {
- if (password != null) {
- try {
- byte[] b = Base64.encodeBase64((username + ":" + password).getBytes("ASCII"));
- String b64 = new String(b, StandardCharsets.US_ASCII);
- httpget.setHeader("Authorization", "Basic " + b64);
- } catch (UnsupportedEncodingException e) {
- }
- }
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.GET)
+ .withUrl(resourceUri.toString());
+ Iterable headers = getFhirHeaders(httpRequest, resourceFormat);
+ HTTPResult response = sendRequest(httpRequest.withHeaders(headers));
+ return unmarshalReference(response, resourceFormat);
}
public Bundle postBatchRequest(URI resourceUri, byte[] payload, String resourceFormat, int timeoutLoading) {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpPost httpPost = new HttpPost(resourceUri);
- configureFhirRequest(httpPost, resourceFormat);
- HttpResponse response = sendPayload(httpPost, payload, proxy, timeoutLoading);
+
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withUrl(resourceUri.toString())
+ .withBody(payload);
+ Iterable headers = getFhirHeaders(httpRequest, resourceFormat);
+ HTTPResult response = sendPayload(httpRequest.withHeaders(headers));
return unmarshalFeed(response, resourceFormat);
}
@@ -226,9 +204,12 @@ public class ClientUtils {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- HttpDelete deleteRequest = new HttpDelete(resourceUri);
- HttpResponse response = sendRequest(deleteRequest);
- int responseStatusCode = response.getStatusLine().getStatusCode();
+
+ HTTPRequest request = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.DELETE)
+ .withUrl(resourceUri.toString());
+ HTTPResult response = sendRequest(request);
+ int responseStatusCode = response.getCode();
boolean deletionSuccessful = false;
if (responseStatusCode == 204) {
deletionSuccessful = true;
@@ -240,171 +221,120 @@ public class ClientUtils {
* Request/Response Helper methods
***********************************************************/
- protected ResourceRequest issueResourceRequest(String resourceFormat, HttpUriRequest request,
+ protected ResourceRequest issueResourceRequest(String resourceFormat, HTTPRequest request,
int timeoutLoading) {
- return issueResourceRequest(resourceFormat, request, null, timeoutLoading);
+ return issueResourceRequest(resourceFormat, request, Collections.emptyList(), timeoutLoading);
}
-
/**
- * @param resourceFormat
- * @param options
- * @return
+ * Issue a resource request.
+ * @param resourceFormat the expected FHIR format
+ * @param request the request to be sent
+ * @param headers any additional headers to add
+ *
+ * @return A ResourceRequest object containing the requested resource
*/
- protected ResourceRequest issueResourceRequest(String resourceFormat, HttpUriRequest request,
- byte[] payload, int timeoutLoading) {
- return issueResourceRequest(resourceFormat, request, payload, null, timeoutLoading);
- }
-
- /**
- * @param resourceFormat
- * @param options
- * @return
- */
- protected ResourceRequest issueResourceRequest(String resourceFormat, HttpUriRequest request,
- byte[] payload, List headers, int timeoutLoading) {
+ protected ResourceRequest issueResourceRequest(String resourceFormat, HTTPRequest request,
+ @Nonnull Iterable headers, int timeoutLoading) {
if (FhirSettings.isProhibitNetworkAccess()) {
throw new FHIRException("Network Access is prohibited in this context");
}
- configureFhirRequest(request, resourceFormat, headers);
- HttpResponse response = null;
- if (request instanceof HttpEntityEnclosingRequest && payload != null) {
- response = sendPayload((HttpEntityEnclosingRequestBase) request, payload, proxy, timeoutLoading);
- } else if (request instanceof HttpEntityEnclosingRequest && payload == null) {
- throw new EFhirClientException("PUT and POST requests require a non-null payload");
- } else {
- response = sendRequest(request);
- }
- T resource = unmarshalReference(response, resourceFormat);
- return new ResourceRequest(resource, response.getStatusLine().getStatusCode(), getLocationHeader(response));
- }
-
- /**
- * Method adds required request headers. TODO handle JSON request as well.
- *
- * @param request
- */
- protected void configureFhirRequest(HttpRequest request, String format) {
- configureFhirRequest(request, format, null);
- }
-
- /**
- * Method adds required request headers. TODO handle JSON request as well.
- *
- * @param request
- */
- protected void configureFhirRequest(HttpRequest request, String format, List headers) {
- if (!Utilities.noString(userAgent)) {
- request.addHeader("User-Agent", userAgent);
- }
- if (!Utilities.noString(acceptLang)) {
- request.addHeader("Accept-Language", acceptLang);
- }
- if (!Utilities.noString(contentLang)) {
- request.addHeader("Content-Language", acceptLang);
- }
-
- if (format != null) {
- request.addHeader("Accept", format);
- request.addHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET);
- }
- if (headers != null) {
- for (Header header : headers) {
- request.addHeader(header);
- }
- }
- setAuth(request);
- }
-
- /**
- * Method posts request payload
- *
- * @param request
- * @param payload
- * @return
- */
- @SuppressWarnings({ "resource", "deprecation" })
- protected HttpResponse sendPayload(HttpEntityEnclosingRequestBase request, byte[] payload, HttpHost proxy,
- int timeoutLoading) {
- if (FhirSettings.isProhibitNetworkAccess()) {
- throw new FHIRException("Network Access is prohibited in this context");
- }
- HttpResponse response = null;
- boolean ok = false;
- long t = System.currentTimeMillis();
- int tryCount = 0;
- while (!ok) {
- try {
- tryCount++;
- HttpClient httpclient = new DefaultHttpClient();
- HttpParams params = httpclient.getParams();
- HttpConnectionParams.setConnectionTimeout(params, timeout);
- HttpConnectionParams.setSoTimeout(params, timeout * timeoutLoading);
-
- if (proxy != null) {
- httpclient.getParams().setParameter(ConnRoutePNames.DEFAULT_PROXY, proxy);
- }
- request.setEntity(new ByteArrayEntity(payload));
- log(request);
- response = httpclient.execute(request);
- ok = true;
- } catch (IOException ioe) {
- System.out.println(ioe.getMessage() + " (" + (System.currentTimeMillis() - t) + "ms / "
- + Utilities.describeSize(payload.length) + ")");
- if (tryCount <= retryCount || (tryCount < 3 && ioe instanceof org.apache.http.conn.ConnectTimeoutException)) {
- ok = false;
- } else {
- throw new EFhirClientException("Error sending HTTP Post/Put Payload to " + "??" + ": " + ioe.getMessage(),
- ioe);
- }
- }
- }
- return response;
- }
-
- /**
- *
- * @param request
- * @param payload
- * @return
- */
- protected HttpResponse sendRequest(HttpUriRequest request) {
- if (FhirSettings.isProhibitNetworkAccess()) {
- throw new FHIRException("Network Access is prohibited in this context");
- }
- HttpResponse response = null;
+ Iterable configuredHeaders = getFhirHeaders(request, resourceFormat, headers);
try {
- HttpClient httpclient = new DefaultHttpClient();
- log(request);
- HttpParams params = httpclient.getParams();
- HttpConnectionParams.setConnectionTimeout(params, timeout);
- HttpConnectionParams.setSoTimeout(params, timeout);
- if (proxy != null) {
- httpclient.getParams().setParameter(ConnRoutePNames.DEFAULT_PROXY, proxy);
- }
- response = httpclient.execute(request);
+
+ HTTPResult response = getManagedWebAccessor().httpCall(request.withHeaders(configuredHeaders));
+ T resource = unmarshalReference(response, resourceFormat);
+ return new ResourceRequest(resource, response.getCode(), getLocationHeader(response.getHeaders()));
+ } catch (IOException ioe) {
+ throw new EFhirClientException("Error sending HTTP Post/Put Payload to " + "??" + ": " + ioe.getMessage(),
+ ioe);
+ }
+ }
+
+ /**
+ * Get required headers for FHIR requests.
+ *
+ * @param httpRequest the request
+ * @param format the expected format
+ */
+ protected Iterable getFhirHeaders(HTTPRequest httpRequest, String format) {
+ return getFhirHeaders(httpRequest, format, null);
+ }
+
+ /**
+ * Get required headers for FHIR requests.
+ *
+ * @param httpRequest the request
+ * @param format the expected format
+ * @param headers any additional headers to add
+ */
+ protected Iterable getFhirHeaders(HTTPRequest httpRequest, String format, Iterable headers) {
+ List configuredHeaders = new ArrayList<>();
+ if (!Utilities.noString(userAgent)) {
+ configuredHeaders.add(new HTTPHeader("User-Agent", userAgent));
+ }
+ if (!Utilities.noString(acceptLanguage)) {
+ configuredHeaders.add(new HTTPHeader("Accept-Language", acceptLanguage));
+ }
+ if (!Utilities.noString(contentLanguage)) {
+ configuredHeaders.add(new HTTPHeader("Content-Language", acceptLanguage));
+ }
+
+ Iterable resourceFormatHeaders = getResourceFormatHeaders(httpRequest, format);
+ resourceFormatHeaders.forEach(configuredHeaders::add);
+
+ if (headers != null) {
+ headers.forEach(configuredHeaders::add);
+ }
+ return configuredHeaders;
+ }
+
+ protected static List getResourceFormatHeaders(HTTPRequest httpRequest, String format) {
+ List headers = new ArrayList<>();
+ headers.add(new HTTPHeader("Accept", format));
+ if (httpRequest.getMethod() == HTTPRequest.HttpMethod.PUT
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.POST
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.PATCH
+ ) {
+ headers.add(new HTTPHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET));
+ }
+ return headers;
+ }
+
+ /**
+ *
+ * @param request The request to be sent
+ * @return The response from the server
+ */
+ protected HTTPResult sendRequest(HTTPRequest request) {
+ if (FhirSettings.isProhibitNetworkAccess()) {
+ throw new FHIRException("Network Access is prohibited in this context");
+ }
+ HTTPResult response = null;
+ try {
+
+ response = getManagedWebAccessor().httpCall(request);
+ return response;
} catch (IOException ioe) {
if (ClientUtils.debugging) {
ioe.printStackTrace();
}
throw new EFhirClientException("Error sending Http Request: " + ioe.getMessage(), ioe);
}
- return response;
}
/**
* Unmarshals a resource from the response stream.
*
- * @param response
- * @return
+ * @param response The response from the server
+ * @return The unmarshalled resource
*/
@SuppressWarnings("unchecked")
- protected T unmarshalReference(HttpResponse response, String format) {
+ protected T unmarshalReference(HTTPResult response, String format) {
T resource = null;
OperationOutcome error = null;
- byte[] cnt = log(response);
- if (cnt != null) {
+ if (response.getContent() != null) {
try {
- resource = (T) getParser(format).parse(cnt);
+ resource = (T) getParser(format).parse(response.getContent());
if (resource instanceof OperationOutcome && hasError((OperationOutcome) resource)) {
error = (OperationOutcome) resource;
}
@@ -423,18 +353,18 @@ public class ClientUtils {
/**
* Unmarshals Bundle from response stream.
*
- * @param response
- * @return
+ * @param response The response from the server
+ * @return The unmarshalled Bundle
*/
- protected Bundle unmarshalFeed(HttpResponse response, String format) {
+ protected Bundle unmarshalFeed(HTTPResult response, String format) {
Bundle feed = null;
- byte[] cnt = log(response);
- String contentType = response.getHeaders("Content-Type")[0].getValue();
+
+ String contentType = HTTPHeaderUtil.getSingleHeader(response.getHeaders(), "Content-Type");
OperationOutcome error = null;
try {
- if (cnt != null) {
+ if (response.getContent() != null) {
if (contentType.contains(ResourceFormat.RESOURCE_XML.getHeader()) || contentType.contains("text/xml+fhir")) {
- Resource rf = getParser(format).parse(cnt);
+ Resource rf = getParser(format).parse(response.getContent());
if (rf instanceof Bundle)
feed = (Bundle) rf;
else if (rf instanceof OperationOutcome && hasError((OperationOutcome) rf)) {
@@ -455,21 +385,20 @@ public class ClientUtils {
return feed;
}
- private boolean hasError(OperationOutcome oo) {
+ protected boolean hasError(OperationOutcome oo) {
for (OperationOutcomeIssueComponent t : oo.getIssue())
if (t.getSeverity() == IssueSeverity.ERROR || t.getSeverity() == IssueSeverity.FATAL)
return true;
return false;
}
- protected String getLocationHeader(HttpResponse response) {
- String location = null;
- if (response.getHeaders("location").length > 0) {// TODO Distinguish between both cases if necessary
- location = response.getHeaders("location")[0].getValue();
- } else if (response.getHeaders("content-location").length > 0) {
- location = response.getHeaders("content-location")[0].getValue();
+ protected static String getLocationHeader(Iterable headers) {
+ String locationHeader = HTTPHeaderUtil.getSingleHeader(headers, LOCATION_HEADER);
+
+ if (locationHeader != null) {
+ return locationHeader;
}
- return location;
+ return HTTPHeaderUtil.getSingleHeader(headers, CONTENT_LOCATION_HEADER);
}
/*****************************************************************
@@ -586,12 +515,17 @@ public class ClientUtils {
public Bundle issuePostFeedRequest(URI resourceUri, Map parameters, String resourceName,
Resource resource, String resourceFormat) throws IOException {
- HttpPost httppost = new HttpPost(resourceUri);
+
+ HTTPRequest httpRequest = new HTTPRequest()
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withUrl(resourceUri.toString());
String boundary = "----WebKitFormBoundarykbMUo6H8QaUnYtRy";
- httppost.addHeader("Content-Type", "multipart/form-data; boundary=" + boundary);
- httppost.addHeader("Accept", resourceFormat);
- configureFhirRequest(httppost, null);
- HttpResponse response = sendPayload(httppost, encodeFormSubmission(parameters, resourceName, resource, boundary));
+ List headers = new ArrayList<>();
+ headers.add(new HTTPHeader("Content-Type", "multipart/form-data; boundary=" + boundary));
+ headers.add(new HTTPHeader("Accept", resourceFormat));
+ this.getFhirHeaders(httpRequest, null).forEach(headers::add);
+
+ HTTPResult response = sendPayload(httpRequest.withBody(encodeFormSubmission(parameters, resourceName, resource, boundary)).withHeaders(headers));
return unmarshalFeed(response, resourceFormat);
}
@@ -622,80 +556,22 @@ public class ClientUtils {
}
/**
- * Method posts request payload
+ * Send an HTTP Post/Put Payload
*
- * @param request
- * @param payload
- * @return
+ * @param request The request to be sent
+ * @return The response from the server
*/
- protected HttpResponse sendPayload(HttpEntityEnclosingRequestBase request, byte[] payload) {
- HttpResponse response = null;
+ protected HTTPResult sendPayload(HTTPRequest request) {
+ HTTPResult response = null;
try {
- log(request);
- HttpClient httpclient = new DefaultHttpClient();
- request.setEntity(new ByteArrayEntity(payload));
- response = httpclient.execute(request);
- log(response);
+
+ response = getManagedWebAccessor().httpCall(request);
} catch (IOException ioe) {
throw new EFhirClientException("Error sending HTTP Post/Put Payload: " + ioe.getMessage(), ioe);
}
return response;
}
- private void log(HttpUriRequest request) {
- if (logger != null) {
- List headers = new ArrayList<>();
- for (Header h : request.getAllHeaders()) {
- headers.add(h.toString());
- }
- logger.logRequest(request.getMethod(), request.getURI().toString(), headers, null);
- }
- }
-
- private void log(HttpEntityEnclosingRequestBase request) {
- if (logger != null) {
- List headers = new ArrayList<>();
- for (Header h : request.getAllHeaders()) {
- headers.add(h.toString());
- }
- byte[] cnt = null;
- InputStream s;
- try {
- s = request.getEntity().getContent();
- cnt = IOUtils.toByteArray(s);
- s.close();
- } catch (Exception e) {
- }
- logger.logRequest(request.getMethod(), request.getURI().toString(), headers, cnt);
- }
- }
-
- private byte[] log(HttpResponse response) {
- byte[] cnt = null;
- try {
- InputStream s = response.getEntity().getContent();
- cnt = IOUtils.toByteArray(s);
- s.close();
- } catch (Exception e) {
- }
- if (logger != null) {
- List headers = new ArrayList<>();
- for (Header h : response.getAllHeaders()) {
- headers.add(h.toString());
- }
- logger.logResponse(response.getStatusLine().toString(), headers, cnt, 0);
- }
- return cnt;
- }
-
- public ToolingClientLogger getLogger() {
- return logger;
- }
-
- public void setLogger(ToolingClientLogger logger) {
- this.logger = logger;
- }
-
/**
* Used for debugging
*
@@ -714,26 +590,5 @@ public class ClientUtils {
return value;
}
- public int getRetryCount() {
- return retryCount;
- }
- public void setRetryCount(int retryCount) {
- this.retryCount = retryCount;
- }
-
- public String getUserAgent() {
- return userAgent;
- }
-
- public void setUserAgent(String userAgent) {
- this.userAgent = userAgent;
- }
-
- public void setAcceptLanguage(String language) {
- this.acceptLang = language;
- }
- public void setContentLanguage(String language) {
- this.contentLang = language;
- }
}
\ No newline at end of file
diff --git a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/FHIRToolingClient.java b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/FHIRToolingClient.java
index ff5900da4..5f42d3d2d 100644
--- a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/FHIRToolingClient.java
+++ b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/FHIRToolingClient.java
@@ -95,48 +95,21 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
private HttpHost proxy;
private int maxResultSetSize = -1;// _count
private Conformance conf;
- private ClientUtils utils = new ClientUtils();
+ private ClientUtils utils = null;
private int useCount;
+ protected ClientUtils getClientUtils() {
+ return new ClientUtils();
+ }
+
// Pass enpoint for client - URI
public FHIRToolingClient(String baseServiceUrl, String userAgent) throws URISyntaxException {
preferredResourceFormat = ResourceFormat.RESOURCE_XML;
+ utils = getClientUtils();
utils.setUserAgent(userAgent);
- detectProxy();
initialize(baseServiceUrl);
}
- public FHIRToolingClient(String baseServiceUrl, String userAgent, String username, String password)
- throws URISyntaxException {
- preferredResourceFormat = ResourceFormat.RESOURCE_XML;
- utils.setUserAgent(userAgent);
- utils.setUsername(username);
- utils.setPassword(password);
- detectProxy();
- initialize(baseServiceUrl);
- }
-
- public void configureProxy(String proxyHost, int proxyPort) {
- utils.setProxy(new HttpHost(proxyHost, proxyPort));
- }
-
- public void detectProxy() {
- String host = System.getenv(hostKey);
- String port = System.getenv(portKey);
-
- if (host == null) {
- host = System.getProperty(hostKey);
- }
-
- if (port == null) {
- port = System.getProperty(portKey);
- }
-
- if (host != null && port != null) {
- this.configureProxy(host, Integer.parseInt(port));
- }
- }
-
public void initialize(String baseServiceUrl) throws URISyntaxException {
base = baseServiceUrl;
resourceAddress = new ResourceAddress(baseServiceUrl);
@@ -286,11 +259,11 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
recordUse();
ResourceRequest result = null;
try {
- List headers = null;
+
result = utils.issuePutRequest(
resourceAddress.resolveGetUriFromResourceClassAndId(resource.getClass(), resource.getId()),
utils.getResourceAsByteArray(resource, false, isJson(getPreferredResourceFormat())),
- withVer(getPreferredResourceFormat(), "1.0"), headers, timeoutOperation);
+ withVer(getPreferredResourceFormat(), "1.0"), null, timeoutOperation);
result.addErrorStatus(410);// gone
result.addErrorStatus(404);// unknown
result.addErrorStatus(405);
@@ -323,10 +296,12 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
recordUse();
ResourceRequest result = null;
try {
- List headers = null;
- result = utils.issuePutRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
+ result = utils.issuePutRequest(
+ resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
utils.getResourceAsByteArray(resource, false, isJson(getPreferredResourceFormat())),
- withVer(getPreferredResourceFormat(), "1.0"), headers, timeoutOperation);
+ withVer(getPreferredResourceFormat(), "1.0"),
+ null,
+ timeoutOperation);
result.addErrorStatus(410);// gone
result.addErrorStatus(404);// unknown
result.addErrorStatus(405);
@@ -355,170 +330,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return result.getPayload();
}
-//
-// public boolean delete(Class resourceClass, String id) {
-// try {
-// return utils.issueDeleteRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id), proxy);
-// } catch(Exception e) {
-// throw new EFhirClientException("An error has occurred while trying to delete this resource", e);
-// }
-//
-// }
-//
-// public OperationOutcome create(Class resourceClass, T resource) {
-// ResourceRequest resourceRequest = null;
-// try {
-// List headers = null;
-// resourceRequest = utils.issuePostRequest(resourceAddress.resolveGetUriFromResourceClass(resourceClass),utils.getResourceAsByteArray(resource, false, isJson(getPreferredResourceFormat())), withVer(getPreferredResourceFormat(), "1.0"), headers, proxy);
-// resourceRequest.addSuccessStatus(201);
-// if(resourceRequest.isUnsuccessfulRequest()) {
-// throw new EFhirClientException("Server responded with HTTP error code " + resourceRequest.getHttpStatus(), (OperationOutcome)resourceRequest.getPayload());
-// }
-// } catch(Exception e) {
-// handleException("An error has occurred while trying to create this resource", e);
-// }
-// OperationOutcome operationOutcome = null;;
-// try {
-// operationOutcome = (OperationOutcome)resourceRequest.getPayload();
-// ResourceAddress.ResourceVersionedIdentifier resVersionedIdentifier =
-// ResourceAddress.parseCreateLocation(resourceRequest.getLocation());
-// OperationOutcomeIssueComponent issue = operationOutcome.addIssue();
-// issue.setSeverity(IssueSeverity.INFORMATION);
-// issue.setUserData(ResourceAddress.ResourceVersionedIdentifier.class.toString(),
-// resVersionedIdentifier);
-// return operationOutcome;
-// } catch(ClassCastException e) {
-// // some server (e.g. grahams) returns the resource directly
-// operationOutcome = new OperationOutcome();
-// OperationOutcomeIssueComponent issue = operationOutcome.addIssue();
-// issue.setSeverity(IssueSeverity.INFORMATION);
-// issue.setUserData(ResourceRequest.class.toString(),
-// resourceRequest.getPayload());
-// return operationOutcome;
-// }
-// }
-
-//
-// public Bundle history(Calendar lastUpdate, Class resourceClass, String id) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForResourceId(resourceClass, id, lastUpdate, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history information for this resource", e);
-// }
-// return history;
-// }
-
-//
-// public Bundle history(Date lastUpdate, Class resourceClass, String id) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForResourceId(resourceClass, id, lastUpdate, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history information for this resource", e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle history(Calendar lastUpdate, Class resourceClass) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForResourceType(resourceClass, lastUpdate, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history information for this resource type", e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle history(Date lastUpdate, Class resourceClass) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForResourceType(resourceClass, lastUpdate, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history information for this resource type", e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle history(Class resourceClass) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForResourceType(resourceClass, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history information for this resource type", e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle history(Class resourceClass, String id) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForResourceId(resourceClass, id, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history information for this resource", e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle history(Date lastUpdate) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForAllResources(lastUpdate, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history since last update",e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle history(Calendar lastUpdate) {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForAllResources(lastUpdate, maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history since last update",e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle history() {
-// Bundle history = null;
-// try {
-// history = utils.issueGetFeedRequest(resourceAddress.resolveGetHistoryForAllResources(maxResultSetSize), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("An error has occurred while trying to retrieve history since last update",e);
-// }
-// return history;
-// }
-//
-//
-// public Bundle search(Class resourceClass, Map parameters) {
-// Bundle searchResults = null;
-// try {
-// searchResults = utils.issueGetFeedRequest(resourceAddress.resolveSearchUri(resourceClass, parameters), withVer(getPreferredResourceFormat(), "1.0"), proxy);
-// } catch (Exception e) {
-// handleException("Error performing search with parameters " + parameters, e);
-// }
-// return searchResults;
-// }
-//
-//
-// public Bundle searchPost(Class resourceClass, T resource, Map parameters) {
-// Bundle searchResults = null;
-// try {
-// searchResults = utils.issuePostFeedRequest(resourceAddress.resolveSearchUri(resourceClass, new HashMap()), parameters, "src", resource, getPreferredResourceFormat());
-// } catch (Exception e) {
-// handleException("Error performing search with parameters " + parameters, e);
-// }
-// return searchResults;
-// }
public Parameters operateType(Class resourceClass, String name, Parameters params) {
recordUse();
@@ -588,107 +400,15 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return (OperationOutcome) result.getPayload();
}
- /*
- * change to meta operations
- *
- * public List getAllTags() { TagListRequest result = null; try { result
- * = utils.issueGetRequestForTagList(resourceAddress.resolveGetAllTags(),
- * withVer(getPreferredResourceFormat(), "1.0"), null, proxy); } catch (Exception e) {
- * handleException("An error has occurred while trying to retrieve all tags",
- * e); } return result.getPayload(); }
- *
- *
- * public List getAllTagsForResourceType(Class
- * resourceClass) { TagListRequest result = null; try { result =
- * utils.issueGetRequestForTagList(resourceAddress.
- * resolveGetAllTagsForResourceType(resourceClass),
- * withVer(getPreferredResourceFormat(), "1.0"), null, proxy); } catch (Exception e) {
- * handleException("An error has occurred while trying to retrieve tags for this resource type"
- * , e); } return result.getPayload(); }
- *
- *
- * public List getTagsForReference(Class
- * resource, String id) { TagListRequest result = null; try { result =
- * utils.issueGetRequestForTagList(resourceAddress.resolveGetTagsForReference(
- * resource, id), withVer(getPreferredResourceFormat(), "1.0"), null, proxy); } catch (Exception
- * e) {
- * handleException("An error has occurred while trying to retrieve tags for this resource"
- * , e); } return result.getPayload(); }
- *
- *
- * public List getTagsForResourceVersion(Class
- * resource, String id, String versionId) { TagListRequest result = null; try {
- * result = utils.issueGetRequestForTagList(resourceAddress.
- * resolveGetTagsForResourceVersion(resource, id, versionId),
- * withVer(getPreferredResourceFormat(), "1.0"), null, proxy); } catch (Exception e) {
- * handleException("An error has occurred while trying to retrieve tags for this resource version"
- * , e); } return result.getPayload(); }
- *
- * // // public boolean deleteTagsForReference(Class
- * resourceClass, String id) { // try { // return
- * utils.issueDeleteRequest(resourceAddress.resolveGetTagsForReference(
- * resourceClass, id), proxy); // } catch(Exception e) { //
- * handleException("An error has occurred while trying to retrieve tags for this resource version"
- * , e); // throw new
- * EFhirClientException("An error has occurred while trying to delete this resource"
- * , e); // } // // } // // // public boolean
- * deleteTagsForResourceVersion(Class resourceClass, String id, List
- * tags, String version) { // try { // return
- * utils.issueDeleteRequest(resourceAddress.resolveGetTagsForResourceVersion(
- * resourceClass, id, version), proxy); // } catch(Exception e) { //
- * handleException("An error has occurred while trying to retrieve tags for this resource version"
- * , e); // throw new
- * EFhirClientException("An error has occurred while trying to delete this resource"
- * , e); // } // }
- *
- *
- * public List createTags(List tags,
- * Class resourceClass, String id) { TagListRequest request = null; try {
- * request =
- * utils.issuePostRequestForTagList(resourceAddress.resolveGetTagsForReference(
- * resourceClass, id),utils.getTagListAsByteArray(tags, false,
- * isJson(getPreferredResourceFormat())), withVer(getPreferredResourceFormat(), "1.0"), null,
- * proxy); request.addSuccessStatus(201); request.addSuccessStatus(200);
- * if(request.isUnsuccessfulRequest()) { throw new
- * EFhirClientException("Server responded with HTTP error code " +
- * request.getHttpStatus()); } } catch(Exception e) {
- * handleException("An error has occurred while trying to set tags for this resource"
- * , e); } return request.getPayload(); }
- *
- *
- * public List createTags(List tags,
- * Class resourceClass, String id, String version) { TagListRequest request =
- * null; try { request = utils.issuePostRequestForTagList(resourceAddress.
- * resolveGetTagsForResourceVersion(resourceClass, id,
- * version),utils.getTagListAsByteArray(tags, false,
- * isJson(getPreferredResourceFormat())), withVer(getPreferredResourceFormat(), "1.0"), null,
- * proxy); request.addSuccessStatus(201); request.addSuccessStatus(200);
- * if(request.isUnsuccessfulRequest()) { throw new
- * EFhirClientException("Server responded with HTTP error code " +
- * request.getHttpStatus()); } } catch(Exception e) {
- * handleException("An error has occurred while trying to set the tags for this resource version"
- * , e); } return request.getPayload(); }
- *
- *
- * public List deleteTags(List tags,
- * Class resourceClass, String id, String version) { TagListRequest request =
- * null; try { request = utils.issuePostRequestForTagList(resourceAddress.
- * resolveDeleteTagsForResourceVersion(resourceClass, id,
- * version),utils.getTagListAsByteArray(tags, false,
- * isJson(getPreferredResourceFormat())), withVer(getPreferredResourceFormat(), "1.0"), null,
- * proxy); request.addSuccessStatus(201); request.addSuccessStatus(200);
- * if(request.isUnsuccessfulRequest()) { throw new
- * EFhirClientException("Server responded with HTTP error code " +
- * request.getHttpStatus()); } } catch(Exception e) {
- * handleException("An error has occurred while trying to delete the tags for this resource version"
- * , e); } return request.getPayload(); }
- */
/**
- * Helper method to prevent nesting of previously thrown EFhirClientExceptions
- *
- * @param e
- * @throws EFhirClientException
+ * Helper method to prevent nesting of previously thrown EFhirClientExceptions. If the e param is an instance of
+ * EFhirClientException, it will be rethrown. Otherwise, a new EFhirClientException will be thrown with e as the
+ * cause.
+ *
+ * @param message The EFhirClientException message.
+ * @param e The exception.
+ * @throws EFhirClientException representing the exception.
*/
protected void handleException(String message, Exception e) throws EFhirClientException {
if (e instanceof EFhirClientException) {
@@ -702,8 +422,8 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
* Helper method to determine whether desired resource representation is Json or
* XML.
*
- * @param format
- * @return
+ * @param format the format to check
+ * @return true if JSON, false if XML
*/
protected boolean isJson(String format) {
boolean isJson = false;
@@ -784,13 +504,13 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
public ValueSet expandValueset(ValueSet source, Parameters expParams) {
recordUse();
- List headers = null;
+
Parameters p = expParams == null ? new Parameters() : expParams.copy();
p.addParameter().setName("valueSet").setResource(source);
ResourceRequest result = utils.issuePostRequest(
resourceAddress.resolveOperationUri(ValueSet.class, "expand"),
utils.getResourceAsByteArray(p, false, isJson(getPreferredResourceFormat())), withVer(getPreferredResourceFormat(), "1.0"),
- headers, 4);
+ null, 4);
result.addErrorStatus(410); // gone
result.addErrorStatus(404); // unknown
result.addErrorStatus(405);
@@ -812,11 +532,11 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
recordUse();
Parameters params = new Parameters();
params.addParameter().setName("name").setValue(new StringType(name));
- List headers = null;
+
ResourceRequest result = utils.issuePostRequest(
resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
utils.getResourceAsByteArray(params, false, isJson(getPreferredResourceFormat())), withVer(getPreferredResourceFormat(), "1.0"),
- headers, timeoutNormal);
+ null, timeoutNormal);
result.addErrorStatus(410);// gone
result.addErrorStatus(404);// unknown
result.addErrorStatus(405);
@@ -835,11 +555,11 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
Parameters params = new Parameters();
params.addParameter().setName("name").setValue(new StringType(name));
params.addParameter().setName("concept").setValue(coding);
- List headers = null;
+
ResourceRequest result = utils.issuePostRequest(
resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
utils.getResourceAsByteArray(params, false, isJson(getPreferredResourceFormat())), withVer(getPreferredResourceFormat(), "1.0"),
- headers, timeoutOperation);
+ null, timeoutOperation);
result.addErrorStatus(410);// gone
result.addErrorStatus(404);// unknown
result.addErrorStatus(405);
@@ -861,22 +581,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
utils.setTimeout(timeout);
}
- public String getUsername() {
- return utils.getUsername();
- }
-
- public void setUsername(String username) {
- utils.setUsername(username);
- }
-
- public String getPassword() {
- return utils.getPassword();
- }
-
- public void setPassword(String password) {
- utils.setPassword(password);
- }
-
public Parameters getTerminologyCapabilities() {
return (Parameters) utils
.issueGetResourceRequest(resourceAddress.resolveMetadataTxCaps(), withVer(getPreferredResourceFormat(), "1.0"), timeoutNormal)
diff --git a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceAddress.java b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceAddress.java
index 7ff64f2bc..e58b6fdaa 100644
--- a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceAddress.java
+++ b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceAddress.java
@@ -233,9 +233,8 @@ public class ResourceAddress {
/**
* For now, assume this type of location header structure. Generalize later:
* http://hl7connect.healthintersections.com.au/svc/fhir/318/_history/1
- *
- * @param serviceBase
- * @param locationHeader
+ *
+ * @param locationResponseHeader
*/
public static ResourceAddress.ResourceVersionedIdentifier parseCreateLocation(String locationResponseHeader) {
Pattern pattern = Pattern.compile(REGEX_ID_WITH_HISTORY);
diff --git a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceRequest.java b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceRequest.java
index 3b337373d..885112ba4 100644
--- a/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceRequest.java
+++ b/org.hl7.fhir.dstu2/src/main/java/org/hl7/fhir/dstu2/utils/client/ResourceRequest.java
@@ -32,11 +32,15 @@ package org.hl7.fhir.dstu2.utils.client;
import java.util.ArrayList;
import java.util.List;
+import lombok.Getter;
import org.hl7.fhir.dstu2.model.Resource;
public class ResourceRequest {
+ @Getter
private T payload;
+ @Getter
private int httpStatus = -1;
+ @Getter
private String location;
private List successfulStatuses = new ArrayList();
private List errorStatuses = new ArrayList();
@@ -67,14 +71,6 @@ public class ResourceRequest {
this.location = location;
}
- public int getHttpStatus() {
- return httpStatus;
- }
-
- public T getPayload() {
- return payload;
- }
-
public T getReference() {
T payloadResource = null;
if (payload != null) {
@@ -99,7 +95,4 @@ public class ResourceRequest {
this.errorStatuses.add(status);
}
- public String getLocation() {
- return location;
- }
}
\ No newline at end of file
diff --git a/org.hl7.fhir.dstu2/src/test/java/org/hl7/fhir/dstu2/utils/client/ClientUtilsTest.java b/org.hl7.fhir.dstu2/src/test/java/org/hl7/fhir/dstu2/utils/client/ClientUtilsTest.java
new file mode 100644
index 000000000..f659af312
--- /dev/null
+++ b/org.hl7.fhir.dstu2/src/test/java/org/hl7/fhir/dstu2/utils/client/ClientUtilsTest.java
@@ -0,0 +1,350 @@
+package org.hl7.fhir.dstu2.utils.client;
+
+
+import okhttp3.HttpUrl;
+import okhttp3.mockwebserver.MockResponse;
+import okhttp3.mockwebserver.MockWebServer;
+import okhttp3.mockwebserver.RecordedRequest;
+import org.hl7.fhir.dstu2.formats.JsonParser;
+import org.hl7.fhir.dstu2.model.*;
+import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPHeaderUtil;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Test;
+import org.mockito.Mockito;
+
+import java.io.IOException;
+import java.net.URI;
+import java.net.URISyntaxException;
+import java.util.List;
+import java.util.Map;
+import java.util.Random;
+import java.util.concurrent.TimeUnit;
+
+import static org.assertj.core.api.Assertions.assertThat;
+import static org.junit.jupiter.api.Assertions.*;
+
+public class ClientUtilsTest {
+
+ public static final String DUMMY_LOCATION = "http://myhost/Patient/1";
+ public static final int TIMEOUT = 5000;
+ private MockWebServer server;
+ private HttpUrl serverUrl;
+ private ClientUtils clientUtils;
+
+ private final Address address = new Address()
+ .setCity("Toronto")
+ .setState("Ontario")
+ .setCountry("Canada");
+ private final HumanName humanName = new HumanName()
+ .addGiven("Mark")
+ .addFamily("Iantorno");
+ private final Patient patient = new Patient()
+ .addName(humanName)
+ .addAddress(address)
+ .setGender(Enumerations.AdministrativeGender.MALE);
+
+ @BeforeEach
+ void setup() {
+ setupMockServer();
+ clientUtils = new ClientUtils();
+ }
+
+ void setupMockServer() {
+ server = new MockWebServer();
+ serverUrl = server.url("/v1/endpoint");
+ }
+
+ @Test
+ @DisplayName("Test resource format headers are added correctly.")
+ void addResourceFormatHeadersGET() {
+
+ String testFormat = "yaml";
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.GET);
+
+ Iterable headers = ClientUtils.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
+ Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
+ Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
+ "Accept header not populated with expected value " + testFormat + ".");
+
+ Assertions.assertNull(headersMap.get("Content-Type"), "Content-Type header null.");
+
+
+ }
+
+ @Test
+ @DisplayName("Test resource format headers are added correctly (POST).")
+ void addResourceFormatHeadersPOST() {
+
+ String testFormat = "yaml";
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.POST);
+
+ Iterable headers = ClientUtils.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
+ Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
+ Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
+ "Accept header not populated with expected value " + testFormat + ".");
+
+ Assertions.assertNotNull(headersMap.get("Content-Type"), "Content-Type header null.");
+ Assertions.assertEquals(testFormat + ";charset=" + ClientUtils.DEFAULT_CHARSET, headersMap.get("Content-Type").get(0),
+ "Content-Type header not populated with expected value \"" + testFormat + ";charset=" + ClientUtils.DEFAULT_CHARSET + "\".");
+
+
+ }
+
+ @Test
+ public void testResourceFormatHeaders_GET() throws IOException, InterruptedException {
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(generateResourceBytes(patient)))
+ .addHeader("Content-Type", ResourceFormat.RESOURCE_JSON)
+ );
+ ResourceRequest resourceRequest = clientUtils.issueGetResourceRequest(serverUrl.uri(), "application/json+fhir", TIMEOUT);
+ RecordedRequest recordedRequest = server.takeRequest();
+ assertThat(recordedRequest.getHeader("Accept")).isEqualTo("application/json+fhir");
+ }
+
+ @Test
+ public void testResourceFormatHeaders_POST() throws IOException, InterruptedException {
+ byte[] payload = generateResourceBytes(patient);
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(payload))
+ .addHeader("Content-Type", "application/json+fhir")
+ .setResponseCode(201)
+ );
+ ResourceRequest resourceRequest = clientUtils.issuePostRequest(serverUrl.uri(), payload, "application/json+fhir", TIMEOUT);
+ RecordedRequest recordedRequest = server.takeRequest();
+ assertThat(recordedRequest.getHeader("Accept")).isEqualTo("application/json+fhir");
+ }
+
+ @Test
+ public void testResourceRequest() {
+ ResourceRequest request = new ResourceRequest<>(new Patient(), 200, "location");
+ request.addSuccessStatus(200);
+ assertTrue(request.getPayload().equalsDeep(new Patient()));
+ assertThat(request.getHttpStatus()).isEqualTo(200);
+ assertThat(request.getLocation()).isEqualTo("location");
+ assertTrue(request.isSuccessfulRequest());
+ assertFalse(request.isUnsuccessfulRequest());
+ }
+
+ @Test
+ public void testIssueGetResourceRequest() throws IOException {
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(generateResourceBytes(patient)))
+ .addHeader("Content-Type", "application/json+fhir")
+ .addHeader("Location", DUMMY_LOCATION)
+ );
+ ResourceRequest resourceRequest = clientUtils.issueGetResourceRequest(serverUrl.uri(), "application/json+fhir", TIMEOUT);
+ resourceRequest.addSuccessStatus(200);
+ assertThat(resourceRequest.getLocation()).isEqualTo(DUMMY_LOCATION);
+ assertTrue(resourceRequest.isSuccessfulRequest());
+ assertTrue(patient.equalsDeep(resourceRequest.getPayload()));
+ }
+
+ @Test
+ void testIssueGetResourceRequest_withContentLocation() throws IOException {
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(generateResourceBytes(patient)))
+ .addHeader("Content-Type", "application/json+fhir")
+ .addHeader("Content-Location", DUMMY_LOCATION)
+ );
+ ResourceRequest resourceRequest = clientUtils.issueGetResourceRequest(serverUrl.uri(), "application/json+fhir", TIMEOUT);
+
+ assertThat(resourceRequest.getLocation()).isEqualTo(DUMMY_LOCATION);
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns 'location' header when both 'location' and 'content-location' are set.")
+ void testIssueGetResourceRequest_ReturnsLocationHeaderWhenBothSet() throws IOException {
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(generateResourceBytes(patient)))
+ .addHeader("Content-Type", "application/json+fhir")
+ .addHeader("Location", DUMMY_LOCATION)
+ .addHeader("Content-Location", "Wrong wrong wrong")
+ );
+ ResourceRequest resourceRequest = clientUtils.issueGetResourceRequest(serverUrl.uri(), "application/json+fhir", TIMEOUT);
+ assertThat(resourceRequest.getLocation()).isEqualTo(DUMMY_LOCATION);
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns 'null' header when neither 'location' or 'content-location' are set.")
+ void testIssueGetResourceRequest_ReturnsNullWhenNoHeadersSet() throws IOException {
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(generateResourceBytes(patient)))
+ .addHeader("Content-Type", "application/json+fhir")
+ );
+ ResourceRequest resourceRequest = clientUtils.issueGetResourceRequest(serverUrl.uri(), "application/json+fhir", TIMEOUT);
+ assertThat(resourceRequest.getLocation()).isNull();
+ }
+
+ @Test
+ public void testIssuePostRequest() throws IOException, InterruptedException {
+ byte[] payload = generateResourceBytes(patient);
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(payload))
+ .addHeader("Content-Type", "application/json+fhir")
+ .addHeader("Location", DUMMY_LOCATION)
+ .setResponseCode(201)
+ );
+ ResourceRequest resourceRequest = clientUtils.issuePostRequest(serverUrl.uri(), payload, "application/json+fhir", TIMEOUT);
+ resourceRequest.addSuccessStatus(201);
+
+ RecordedRequest recordedRequest = server.takeRequest();
+ assertArrayEquals(payload, recordedRequest.getBody().readByteArray(),
+ "PUT request payload does not match send data.");
+ assertThat(resourceRequest.getLocation()).isEqualTo(DUMMY_LOCATION);
+ assertTrue(resourceRequest.isSuccessfulRequest());
+ assertTrue(patient.equalsDeep(resourceRequest.getPayload()));
+ }
+
+ @Test
+ public void testIssuePutRequest() throws IOException, InterruptedException {
+ byte[] payload = generateResourceBytes(patient);
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(payload))
+ .addHeader("Content-Type", "application/json+fhir")
+ .addHeader("Location", DUMMY_LOCATION)
+ .setResponseCode(200)
+ );
+ ResourceRequest resourceRequest = clientUtils.issuePutRequest(serverUrl.uri(), payload, "application/json+fhir", TIMEOUT);
+ resourceRequest.addSuccessStatus(200);
+ RecordedRequest recordedRequest = server.takeRequest();
+ assertArrayEquals(payload, recordedRequest.getBody().readByteArray(),
+ "PUT request payload does not match send data.");
+
+ assertThat(resourceRequest.getLocation()).isEqualTo(DUMMY_LOCATION);
+ assertTrue(resourceRequest.isSuccessfulRequest());
+ assertTrue(patient.equalsDeep(resourceRequest.getPayload()));
+ }
+
+ @Test
+ public void testIssueDeleteRequest() throws IOException {
+ server.enqueue(
+ new MockResponse().addHeader("Location", DUMMY_LOCATION).setResponseCode(204)
+ );
+ boolean success = clientUtils.issueDeleteRequest(serverUrl.uri());
+ assertTrue(success);
+ }
+
+ @Test
+ public void testIssueDeleteRequest_fail() throws IOException, InterruptedException {
+ server.enqueue(
+ new MockResponse().addHeader("Location", DUMMY_LOCATION).setResponseCode(500)
+ );
+ boolean success = clientUtils.issueDeleteRequest(serverUrl.uri());
+ RecordedRequest recordedRequest = server.takeRequest();
+ assertThat(recordedRequest.getMethod()).isEqualTo("DELETE");
+ assertThat(recordedRequest.getRequestUrl().uri()).isEqualTo(serverUrl.uri());
+ assertFalse(success);
+ }
+
+ @Test
+ @DisplayName("test logger works")
+ public void testLogger() throws IOException, InterruptedException {
+ byte[] payload = generateResourceBytes(patient);
+ server.enqueue(
+ new MockResponse()
+ .setResponseCode(200)
+ .setBody(new String(payload))
+ );
+ ToolingClientLogger mockLogger = Mockito.mock(ToolingClientLogger.class);
+ clientUtils.setLogger(mockLogger);
+ clientUtils.issuePostRequest(serverUrl.uri(), payload, "application/json+fhir"
+ , null, TIMEOUT);
+ server.takeRequest();
+ Mockito.verify(mockLogger, Mockito.times(1))
+ .logRequest(Mockito.anyString(), Mockito.anyString(), Mockito.anyList(), Mockito.any());
+ Mockito.verify(mockLogger, Mockito.times(1))
+ .logResponse(Mockito.anyString(), Mockito.anyList(), Mockito.any(), Mockito.anyLong());
+ }
+
+ @Test
+ @DisplayName("Test that FATAL issue severity triggers error.")
+ void hasErrorTestFatal() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.FATAL));
+ Assertions.assertTrue(clientUtils.hasError(outcome), "Error check not triggered for FATAL issue severity.");
+ }
+
+ @Test
+ @DisplayName("Test that ERROR issue severity triggers error.")
+ void hasErrorTestError() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.ERROR));
+ Assertions.assertTrue(clientUtils.hasError(outcome), "Error check not triggered for ERROR issue severity.");
+ }
+
+ @Test
+ @DisplayName("Test that no FATAL or ERROR issue severity does not trigger error.")
+ void hasErrorTestNoErrors() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ Assertions.assertFalse(clientUtils.hasError(outcome), "Error check triggered unexpectedly.");
+ }
+
+ @Test
+ @DisplayName("GET request, test client retries after timeout failure.")
+ void test_get_retries_with_timeout() throws IOException, URISyntaxException {
+ int failedAttempts = new Random().nextInt(5) + 1;
+ System.out.println("Simulating <" + failedAttempts + "> failed connections (timeouts) before success.");
+ for (int i = 0; i < failedAttempts; i++) {
+ server.enqueue(new MockResponse().setHeadersDelay(TIMEOUT * 10, TimeUnit.MILLISECONDS)
+ .setBody(new String(generateResourceBytes(patient))));
+ }
+ server.enqueue(new MockResponse().setBody(new String(generateResourceBytes(patient))));
+ clientUtils.setRetryCount(failedAttempts + 1);
+
+ ResourceRequest resourceRequest = clientUtils.issueGetResourceRequest(serverUrl.uri(), "application/json+fhir"
+ , TIMEOUT);
+ resourceRequest.addSuccessStatus(200);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+ @Test
+ @DisplayName("GET request, test client retries after bad response.")
+ void test_get_retries_with_unsuccessful_response() throws IOException, URISyntaxException {
+ int failedAttempts = new Random().nextInt(5) + 1;
+ System.out.println("Simulating <" + failedAttempts + "> failed connections (bad response codes) before success.");
+ for (int i = 0; i < failedAttempts; i++) {
+ server.enqueue(new MockResponse().setResponseCode(400 + i).setBody(new String(generateResourceBytes(patient))));
+ }
+ server.enqueue(new MockResponse().setBody(new String(generateResourceBytes(patient))));
+ clientUtils.setRetryCount(failedAttempts + 1);
+
+ ResourceRequest resourceRequest = clientUtils.issueGetResourceRequest(serverUrl.uri(), "application/json+fhir", TIMEOUT);
+ resourceRequest.addSuccessStatus(200);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+
+ byte[] generateResourceBytes(Resource resource) throws IOException {
+ return new JsonParser().composeBytes(resource);
+ }
+}
diff --git a/org.hl7.fhir.dstu2/src/test/java/org/hl7/fhir/dstu2/utils/client/FHIRToolingClientTest.java b/org.hl7.fhir.dstu2/src/test/java/org/hl7/fhir/dstu2/utils/client/FHIRToolingClientTest.java
new file mode 100644
index 000000000..587fdb9f2
--- /dev/null
+++ b/org.hl7.fhir.dstu2/src/test/java/org/hl7/fhir/dstu2/utils/client/FHIRToolingClientTest.java
@@ -0,0 +1,166 @@
+package org.hl7.fhir.dstu2.utils.client;
+
+import org.hl7.fhir.dstu2.model.*;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+import org.mockito.*;
+
+import java.net.URI;
+import java.net.URISyntaxException;
+
+import static org.assertj.core.api.AssertionsForClassTypes.assertThat;
+import static org.mockito.Mockito.*;
+
+public class FHIRToolingClientTest {
+
+ private final Address address = new Address()
+ .setCity("Toronto")
+ .setState("Ontario")
+ .setCountry("Canada");
+ private final HumanName humanName = new HumanName()
+ .addGiven("Mark")
+ .addFamily("Iantorno");
+ private final Patient patient = new Patient()
+ .addName(humanName)
+ .addAddress(address)
+ .setGender(Enumerations.AdministrativeGender.MALE);
+
+ private ClientUtils mockClientUtils;
+ private FHIRToolingClient toolingClient;
+
+ @Captor
+ private ArgumentCaptor uriArgumentCaptor;
+
+ @BeforeEach
+ public void beforeEach() throws URISyntaxException {
+ MockitoAnnotations.openMocks(this);
+ mockClientUtils = Mockito.mock(ClientUtils.class);
+
+ toolingClient = new FHIRToolingClient("http://dummy-base-url.com", "dummy-user-agent") {
+ @Override
+ protected ClientUtils getClientUtils() {
+ return mockClientUtils;
+ }
+ };
+
+ /*
+ Need to reset here. When initialized, the client makes a call to getConformanceStatementQuick, which messes with
+ our expected calls.
+ */
+ reset(mockClientUtils);
+ }
+
+ @Test
+ public void testGetTerminologyCapabilities() throws URISyntaxException {
+
+ Parameters expectedCapabilities = new Parameters();
+ expectedCapabilities.addParameter().setName("name").setValue(new StringType("dummyValue"));
+
+ when(mockClientUtils.issueGetResourceRequest(uriArgumentCaptor.capture(), Mockito.anyString(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(expectedCapabilities, 200, "location"));
+ Parameters actualCapabilities = toolingClient.getTerminologyCapabilities();
+
+ Mockito.verify(mockClientUtils).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(), ArgumentMatchers.anyInt());
+ assertThat(actualCapabilities).isEqualTo(expectedCapabilities);
+
+ assertThat(uriArgumentCaptor.getValue()).isEqualTo(new URI("http://dummy-base-url.com/metadata?mode=terminology"));
+ }
+
+ @Test
+ public void testGetConformanceStatement() throws URISyntaxException {
+
+ Conformance expectedConformance = new Conformance();
+ expectedConformance.setCopyright("dummyCopyright");
+ when(mockClientUtils.issueGetResourceRequest(uriArgumentCaptor.capture(), Mockito.anyString(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(expectedConformance, 200, "location"));
+ Conformance actualConformance = toolingClient.getConformanceStatement();
+
+ Mockito.verify(mockClientUtils).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(), ArgumentMatchers.anyInt());
+ assertThat(actualConformance).isEqualTo(expectedConformance);
+
+ assertThat(uriArgumentCaptor.getValue()).isEqualTo(new URI("http://dummy-base-url.com/metadata"));
+ }
+
+ @Test
+ public void testGetConformanceStatementQuick() throws URISyntaxException {
+
+ Conformance expectedConformance = new Conformance();
+ expectedConformance.setCopyright("dummyCopyright");
+ when(mockClientUtils.issueGetResourceRequest(uriArgumentCaptor.capture(), Mockito.anyString(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(expectedConformance, 200, "location"));
+ Conformance actualConformance = toolingClient.getConformanceStatementQuick();
+
+ Mockito.verify(mockClientUtils).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(), ArgumentMatchers.anyInt());
+ assertThat(actualConformance).isEqualTo(expectedConformance);
+
+ assertThat(uriArgumentCaptor.getValue()).isEqualTo(new URI("http://dummy-base-url.com/metadata?_summary=true"));
+ }
+
+ @Test
+ void testRead() {
+ when(mockClientUtils.issueGetResourceRequest(uriArgumentCaptor.capture(), Mockito.anyString(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(patient, 200, "location"));
+ Patient actualPatient = toolingClient.read(Patient.class, "id");
+
+ Mockito.verify(mockClientUtils).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(), ArgumentMatchers.anyInt());
+ assertThat(actualPatient).isEqualTo(patient);
+
+ assertThat(uriArgumentCaptor.getValue().toString()).isEqualTo("http://dummy-base-url.com/Patient/id");
+ }
+
+ @Test
+ void testVRead() {
+ when(mockClientUtils.issueGetResourceRequest(uriArgumentCaptor.capture(), Mockito.anyString(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(patient, 200, "location"));
+ Patient actualPatient = toolingClient.vread(Patient.class, "id", "version");
+
+ Mockito.verify(mockClientUtils).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(), ArgumentMatchers.anyInt());
+ assertThat(actualPatient).isEqualTo(patient);
+
+ assertThat(uriArgumentCaptor.getValue().toString()).isEqualTo("http://dummy-base-url.com/Patient/id/_history/version");
+ }
+
+ @Test
+ void testCanonical() {
+ Bundle bundle = new Bundle();
+ bundle.addEntry().setResource(patient);
+ when(mockClientUtils.issueGetResourceRequest(uriArgumentCaptor.capture(), Mockito.anyString(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(bundle, 200, "location"));
+ Patient actualPatient = toolingClient.getCanonical(Patient.class, "canonicalURL");
+
+ Mockito.verify(mockClientUtils).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(), ArgumentMatchers.anyInt());
+ assertThat(actualPatient).isEqualTo(patient);
+
+ assertThat(uriArgumentCaptor.getValue().toString()).isEqualTo("http://dummy-base-url.com/Patient?url=canonicalURL");
+ }
+
+ @Test
+ void testUpdate() {
+ final byte[] dummyBytes = "dummyBytes".getBytes();
+ when(mockClientUtils.getResourceAsByteArray(any(Patient.class), anyBoolean(), anyBoolean())).thenReturn(dummyBytes);
+ when(mockClientUtils.issuePutRequest(uriArgumentCaptor.capture(), Mockito.any(byte[].class), Mockito.anyString(), Mockito.isNull(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(patient, 200, "location"));
+ Patient actualPatient = toolingClient.update(Patient.class, patient, "id");
+
+ Mockito.verify(mockClientUtils).issuePutRequest(ArgumentMatchers.any(URI.class), Mockito.any(byte[].class), ArgumentMatchers.anyString(), ArgumentMatchers.isNull(),ArgumentMatchers.anyInt());
+ assertThat(actualPatient).isEqualTo(patient);
+
+ assertThat(uriArgumentCaptor.getValue().toString()).isEqualTo("http://dummy-base-url.com/Patient/id");
+ }
+
+ @Test
+ void testValidate() {
+ final byte[] dummyBytes = "dummyBytes".getBytes();
+ final OperationOutcome expectedOutcome = new OperationOutcome();
+ OperationOutcome.OperationOutcomeIssueComponent issueComponent = expectedOutcome.addIssue();
+ issueComponent.setSeverity(OperationOutcome.IssueSeverity.ERROR);
+ when(mockClientUtils.getResourceAsByteArray(any(Patient.class), anyBoolean(), anyBoolean())).thenReturn(dummyBytes);
+ when(mockClientUtils.issuePostRequest(uriArgumentCaptor.capture(), Mockito.any(byte[].class), Mockito.anyString(), Mockito.anyInt()))
+ .thenReturn(new ResourceRequest<>(expectedOutcome, 200, "location"));
+
+ OperationOutcome actualOutcome = toolingClient.validate(Patient.class, patient, "id");
+ assertThat(actualOutcome).isEqualTo(expectedOutcome);
+
+ assertThat(uriArgumentCaptor.getValue().toString()).isEqualTo("http://dummy-base-url.com/Patient/$validate/id");
+ }
+}
diff --git a/org.hl7.fhir.dstu3/pom.xml b/org.hl7.fhir.dstu3/pom.xml
index 8f61071b8..aef8d1dcc 100644
--- a/org.hl7.fhir.dstu3/pom.xml
+++ b/org.hl7.fhir.dstu3/pom.xml
@@ -28,6 +28,12 @@
hapi-fhir-base
+
+ org.projectlombok
+ lombok
+ provided
+
+
org.fhir
@@ -99,6 +105,13 @@
+
+ com.squareup.okhttp3
+ mockwebserver
+ true
+ test
+
+
com.fasterxml.jackson.core
jackson-databind
diff --git a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/FHIRToolingClient.java b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/FHIRToolingClient.java
index fe6aec345..220c97cd6 100644
--- a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/FHIRToolingClient.java
+++ b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/FHIRToolingClient.java
@@ -3,18 +3,15 @@ package org.hl7.fhir.dstu3.utils.client;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
-import java.util.ArrayList;
-import java.util.Base64;
-import java.util.EnumSet;
-import java.util.HashMap;
-import java.util.Map;
+import java.util.*;
+import lombok.Getter;
+import lombok.Setter;
import org.hl7.fhir.dstu3.model.Bundle;
import org.hl7.fhir.dstu3.model.CapabilityStatement;
import org.hl7.fhir.dstu3.model.CodeSystem;
import org.hl7.fhir.dstu3.model.Coding;
import org.hl7.fhir.dstu3.model.ConceptMap;
-import org.hl7.fhir.dstu3.model.ExpansionProfile;
import org.hl7.fhir.dstu3.model.OperationOutcome;
import org.hl7.fhir.dstu3.model.Parameters;
import org.hl7.fhir.dstu3.model.Parameters.ParametersParameterComponent;
@@ -31,8 +28,7 @@ import org.hl7.fhir.utilities.FhirPublication;
import org.hl7.fhir.utilities.ToolingClientLogger;
import org.hl7.fhir.utilities.Utilities;
-import okhttp3.Headers;
-import okhttp3.internal.http2.Header;
+import org.hl7.fhir.utilities.http.HTTPHeader;
/**
* Very Simple RESTful client. This is purely for use in the standalone
@@ -72,13 +68,17 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
private int maxResultSetSize = -1;//_count
private CapabilityStatement capabilities;
private Client client = new Client();
- private ArrayList headers = new ArrayList<>();
- private String username;
- private String password;
+ private List headers = new ArrayList<>();
+ @Setter
+ @Getter
private String userAgent;
private EnumSet allowedVersions;
- private String acceptLang;
- private String contentLang;
+ @Setter
+ @Getter
+ private String acceptLanguage;
+ @Setter
+ private String contentLanguage;
+ @Getter
private int useCount;
//Pass endpoint for client - URI
@@ -150,7 +150,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
capabilities = (Parameters) client.issueGetResourceRequest(resourceAddress.resolveMetadataTxCaps(),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(false),
"TerminologyCapabilities",
timeoutNormal).getReference();
} catch (Exception e) {
@@ -164,7 +164,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
conformance = (CapabilityStatement) client.issueGetResourceRequest(resourceAddress.resolveMetadataUri(false),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(false),
"CapabilitiesStatement",
timeoutNormal).getReference();
} catch (Exception e) {
@@ -178,7 +178,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
capabilities = (CapabilityStatement) client.issueGetResourceRequest(resourceAddress.resolveMetadataUri(true),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(false),
"CapabilitiesStatement-Quick",
timeoutNormal).getReference();
} catch (Exception e) {
@@ -193,7 +193,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issueGetResourceRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(false),
"Read " + resourceClass.getName() + "/" + id,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
@@ -211,7 +211,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issueGetResourceRequest(resourceAddress.resolveGetUriFromResourceClassAndIdAndVersion(resourceClass, id, version),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(false),
"VRead " + resourceClass.getName() + "/" + id + "/?_history/" + version,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
@@ -229,7 +229,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issueGetResourceRequest(resourceAddress.resolveGetUriFromResourceClassAndCanonical(resourceClass, canonicalURL),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(false),
"Read " + resourceClass.getName() + "?url=" + canonicalURL,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
@@ -253,7 +253,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePutRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resource.getClass(), resource.getId()),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(true),
"Update " + resource.fhirType() + "/" + resource.getId(),
timeoutOperation);
if (result.isUnsuccessfulRequest()) {
@@ -281,7 +281,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePutRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(true),
"Update " + resource.fhirType() + "/" + id,
timeoutOperation);
if (result.isUnsuccessfulRequest()) {
@@ -320,13 +320,13 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
if (client.getLogger() != null) {
client.getLogger().logRequest("POST", url.toString(), null, body);
}
- result = client.issuePostRequest(url, body, withVer(getPreferredResourceFormat(), "3.0"), generateHeaders(),
+ result = client.issuePostRequest(url, body, withVer(getPreferredResourceFormat(), "3.0"), generateHeaders(true),
"POST " + resourceClass.getName() + "/$" + name, timeoutLong);
} else {
if (client.getLogger() != null) {
client.getLogger().logRequest("GET", url.toString(), null, null);
}
- result = client.issueGetResourceRequest(url, withVer(getPreferredResourceFormat(), "3.0"), generateHeaders(), "GET " + resourceClass.getName() + "/$" + name, timeoutLong);
+ result = client.issueGetResourceRequest(url, withVer(getPreferredResourceFormat(), "3.0"), generateHeaders(false), "GET " + resourceClass.getName() + "/$" + name, timeoutLong);
}
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(), (OperationOutcome) result.getPayload());
@@ -363,7 +363,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issuePostRequest(resourceAddress.resolveValidateUri(resourceClass, id),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
- withVer(getPreferredResourceFormat(), "3.0"), generateHeaders(),
+ withVer(getPreferredResourceFormat(), "3.0"), generateHeaders(true),
"POST " + resourceClass.getName() + (id != null ? "/" + id : "") + "/$validate", timeoutLong);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(), (OperationOutcome) result.getPayload());
@@ -375,10 +375,14 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
}
/**
- * Helper method to prevent nesting of previously thrown EFhirClientExceptions
+ * Helper method to prevent nesting of previously thrown EFhirClientExceptions. If the e param is an instance of
+ * EFhirClientException, it will be rethrown. Otherwise, a new EFhirClientException will be thrown with e as the
+ * cause.
*
- * @param e
- * @throws EFhirClientException
+
+ * @param message The EFhirClientException message.
+ * @param e The exception
+ * @throws EFhirClientException EFhirClientException representing the exception.
*/
protected void handleException(String message, Exception e) throws EFhirClientException {
if (e instanceof EFhirClientException) {
@@ -389,11 +393,11 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
}
/**
- * Helper method to determine whether desired resource representation
- * is Json or XML.
+ * Helper method to determine whether desired resource representation is Json or
+ * XML.
*
- * @param format
- * @return
+ * @param format the format
+ * @return true if the format is JSON, false otherwise
*/
protected boolean isJson(String format) {
boolean isJson = false;
@@ -420,7 +424,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issueGetResourceRequest(resourceAddress.resolveOperationUri(CodeSystem.class, "lookup", params),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(false),
"CodeSystem/$lookup",
timeoutNormal);
} catch (IOException e) {
@@ -439,7 +443,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(CodeSystem.class, "lookup"),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(true),
"CodeSystem/$lookup",
timeoutNormal);
} catch (IOException e) {
@@ -458,7 +462,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(ConceptMap.class, "transform"),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(true),
"ConceptMap/$transform",
timeoutNormal);
} catch (IOException e) {
@@ -479,7 +483,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(ValueSet.class, "expand"),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(true),
"ValueSet/$expand?url=" + source.getUrl(),
timeoutExpand);
if (result.isUnsuccessfulRequest()) {
@@ -504,7 +508,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(true),
"Closure?name=" + name,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
@@ -526,7 +530,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true),
withVer(getPreferredResourceFormat(), "3.0"),
- generateHeaders(),
+ generateHeaders(true),
"UpdateClosure?name=" + name,
timeoutOperation);
if (result.isUnsuccessfulRequest()) {
@@ -538,21 +542,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return result == null ? null : (ConceptMap) result.getPayload();
}
- public String getUsername() {
- return username;
- }
-
- public void setUsername(String username) {
- this.username = username;
- }
-
- public String getPassword() {
- return password;
- }
-
- public void setPassword(String password) {
- this.password = password;
- }
public long getTimeout() {
return client.getTimeout();
@@ -578,48 +567,27 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
client.setRetryCount(retryCount);
}
- public void setClientHeaders(ArrayList headers) {
- this.headers = headers;
+ public void setClientHeaders(Iterable headers) {
+ this.headers =new ArrayList<>();
+ headers.forEach(this.headers::add);
}
- private Headers generateHeaders() {
- Headers.Builder builder = new Headers.Builder();
- // Add basic auth header if it exists
- if (basicAuthHeaderExists()) {
- builder.add(getAuthorizationHeader().toString());
- }
+ private Iterable generateHeaders(boolean hasBody) {
// Add any other headers
- if(this.headers != null) {
- this.headers.forEach(header -> builder.add(header.toString()));
- }
+ List headers = new ArrayList<>(this.headers);
if (!Utilities.noString(userAgent)) {
- builder.add("User-Agent: "+userAgent);
- }
- if (!Utilities.noString(acceptLang)) {
- builder.add("Accept-Language: "+acceptLang);
+ headers.add(new HTTPHeader("User-Agent",userAgent));
}
- if (!Utilities.noString(contentLang)) {
- builder.add("Content-Language: "+contentLang);
+
+ if (!Utilities.noString(acceptLanguage)) {
+ headers.add(new HTTPHeader("Accept-Language", acceptLanguage));
}
- return builder.build();
- }
- public boolean basicAuthHeaderExists() {
- return (username != null) && (password != null);
- }
+ if (hasBody && !Utilities.noString(contentLanguage)) {
+ headers.add(new HTTPHeader("Content-Language",contentLanguage));
+ }
- public Header getAuthorizationHeader() {
- String usernamePassword = username + ":" + password;
- String base64usernamePassword = Base64.getEncoder().encodeToString(usernamePassword.getBytes());
- return new Header("Authorization", "Basic " + base64usernamePassword);
- }
-
- public String getUserAgent() {
- return userAgent;
- }
-
- public void setUserAgent(String userAgent) {
- this.userAgent = userAgent;
+ return headers;
}
public String getServerVersion() {
@@ -627,17 +595,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return capabilities == null ? null : capabilities.getSoftware().getVersion();
}
- public void setAcceptLanguage(String lang) {
- this.acceptLang = lang;
- }
- public void setContentLanguage(String lang) {
- this.contentLang = lang;
- }
-
- public int getUseCount() {
- return useCount;
- }
-
private void recordUse() {
useCount++;
}
diff --git a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/Client.java b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/Client.java
index a0635aadf..a27f64b88 100644
--- a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/Client.java
+++ b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/Client.java
@@ -2,6 +2,7 @@ package org.hl7.fhir.dstu3.utils.client.network;
import java.io.IOException;
import java.net.URI;
+import java.util.Collections;
import java.util.Map;
import java.util.concurrent.TimeUnit;
@@ -10,10 +11,8 @@ import org.hl7.fhir.dstu3.model.Resource;
import org.hl7.fhir.dstu3.utils.client.EFhirClientException;
import org.hl7.fhir.utilities.ToolingClientLogger;
-import okhttp3.Headers;
-import okhttp3.MediaType;
-import okhttp3.Request;
-import okhttp3.RequestBody;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPRequest;
public class Client {
@@ -61,21 +60,22 @@ public class Client {
String resourceFormat,
String message,
long timeout) throws IOException {
- Request.Builder request = new Request.Builder()
- .method("OPTIONS", null)
- .url(optionsUri.toURL());
- return executeFhirRequest(request, resourceFormat, new Headers.Builder().build(), message, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(optionsUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.OPTIONS);
+
+ return executeFhirRequest(request, resourceFormat, Collections.emptyList(), message, retryCount, timeout);
}
public ResourceRequest issueGetResourceRequest(URI resourceUri,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
long timeout) throws IOException {
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL());
-
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
@@ -87,21 +87,21 @@ public class Client {
String resourceFormat,
String message,
long timeout) throws IOException {
- return issuePutRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePutRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePutRequest(URI resourceUri,
byte[] payload,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
long timeout) throws IOException {
if (payload == null) throw new EFhirClientException("PUT requests require a non-null payload");
- RequestBody body = RequestBody.create(payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .put(body);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.PUT)
+ .withBody(payload);
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
@@ -110,36 +110,36 @@ public class Client {
String resourceFormat,
String message,
long timeout) throws IOException {
- return issuePostRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePostRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePostRequest(URI resourceUri,
byte[] payload,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
long timeout) throws IOException {
if (payload == null) throw new EFhirClientException("POST requests require a non-null payload");
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .post(body);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload);
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
public boolean issueDeleteRequest(URI resourceUri) throws IOException {
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .delete();
- return executeFhirRequest(request, null, new Headers.Builder().build(), null, retryCount, timeout).isSuccessfulRequest();
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.DELETE);
+ return executeFhirRequest(request, null, Collections.emptyList(), null, retryCount, timeout).isSuccessfulRequest();
}
public Bundle issueGetFeedRequest(URI resourceUri, String resourceFormat) throws IOException {
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL());
-
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
public Bundle issuePostFeedRequest(URI resourceUri,
@@ -149,12 +149,13 @@ public class Client {
String resourceFormat) throws IOException {
String boundary = "----WebKitFormBoundarykbMUo6H8QaUnYtRy";
byte[] payload = ByteUtils.encodeFormSubmission(parameters, resourceName, resource, boundary);
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .post(body);
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload);
+
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
public Bundle postBatchRequest(URI resourceUri,
@@ -163,17 +164,17 @@ public class Client {
String message,
int timeout) throws IOException {
if (payload == null) throw new EFhirClientException("POST requests require a non-null payload");
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .post(body);
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), message, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload);
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), message, retryCount, timeout);
}
- public Bundle executeBundleRequest(Request.Builder request,
+ public Bundle executeBundleRequest(HTTPRequest request,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
int retryCount,
long timeout) throws IOException {
@@ -182,23 +183,23 @@ public class Client {
.withResourceFormat(resourceFormat)
.withRetryCount(retryCount)
.withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS)
.executeAsBatch();
}
- public ResourceRequest executeFhirRequest(Request.Builder request,
- String resourceFormat,
- Headers headers,
- String message,
- int retryCount,
- long timeout) throws IOException {
+ public ResourceRequest executeFhirRequest(HTTPRequest request,
+ String resourceFormat,
+ Iterable headers,
+ String message,
+ int retryCount,
+ long timeout) throws IOException {
return new FhirRequestBuilder(request, base)
.withLogger(logger)
.withResourceFormat(resourceFormat)
.withRetryCount(retryCount)
.withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS)
.execute();
}
diff --git a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/ClientHeaders.java b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/ClientHeaders.java
index c235c7b19..0d530b00a 100644
--- a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/ClientHeaders.java
+++ b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/ClientHeaders.java
@@ -5,8 +5,7 @@ import java.util.List;
import java.util.stream.Collectors;
import org.hl7.fhir.exceptions.FHIRException;
-
-import okhttp3.internal.http2.Header;
+import org.hl7.fhir.utilities.http.HTTPHeader;
/**
* Generic Implementation of Client Headers.
@@ -16,30 +15,30 @@ import okhttp3.internal.http2.Header;
*/
public class ClientHeaders {
- private final ArrayList headers;
+ private final ArrayList headers;
public ClientHeaders() {
this.headers = new ArrayList<>();
}
- public ClientHeaders(ArrayList headers) {
+ public ClientHeaders(ArrayList headers) {
this.headers = headers;
}
- public ArrayList headers() {
+ public ArrayList headers() {
return headers;
}
/**
* Add a header to the list of stored headers for network operations.
*
- * @param header {@link Header} to add to the list.
+ * @param header {@link HTTPHeader} to add to the list.
* @throws FHIRException if the header being added is a duplicate.
*/
- public ClientHeaders addHeader(Header header) throws FHIRException {
+ public ClientHeaders addHeader(HTTPHeader header) throws FHIRException {
if (headers.contains(header)) {
- throw new FHIRException("Attempting to add duplicate header, <" + header.name + ", "
- + header.value + ">.");
+ throw new FHIRException("Attempting to add duplicate header, <" + header.getName() + ", "
+ + header.getValue() + ">.");
}
headers.add(header);
return this;
@@ -48,39 +47,39 @@ public class ClientHeaders {
/**
* Add a header to the list of stored headers for network operations.
*
- * @param headerList {@link List} of {@link Header} to add.
+ * @param headerList {@link List} of {@link HTTPHeader} to add.
* @throws FHIRException if any of the headers being added is a duplicate.
*/
- public ClientHeaders addHeaders(List headerList) throws FHIRException {
+ public ClientHeaders addHeaders(List headerList) throws FHIRException {
headerList.forEach(this::addHeader);
return this;
}
/**
* Removes the passed in header from the list of stored headers.
- * @param header {@link Header} to remove from the list.
+ * @param header {@link HTTPHeader} to remove from the list.
* @throws FHIRException if the header passed in does not exist within the stored list.
*/
- public ClientHeaders removeHeader(Header header) throws FHIRException {
+ public ClientHeaders removeHeader(HTTPHeader header) throws FHIRException {
if (!headers.remove(header)) {
- throw new FHIRException("Attempting to remove header, <" + header.name + ", "
- + header.value + ">, from GenericClientHeaders that is not currently stored.");
+ throw new FHIRException("Attempting to remove header, <" + header.getName() + ", "
+ + header.getValue() + ">, from GenericClientHeaders that is not currently stored.");
}
return this;
}
/**
* Removes the passed in headers from the list of stored headers.
- * @param headerList {@link List} of {@link Header} to remove.
+ * @param headerList {@link List} of {@link HTTPHeader} to remove.
* @throws FHIRException if any of the headers passed in does not exist within the stored list.
*/
- public ClientHeaders removeHeaders(List headerList) throws FHIRException {
+ public ClientHeaders removeHeaders(List headerList) throws FHIRException {
headerList.forEach(this::removeHeader);
return this;
}
/**
- * Clears all stored {@link Header}.
+ * Clears all stored {@link HTTPHeader}.
*/
public ClientHeaders clearHeaders() {
headers.clear();
@@ -90,7 +89,7 @@ public class ClientHeaders {
@Override
public String toString() {
return this.headers.stream()
- .map(header -> "\t" + header.name + ":" + header.value)
+ .map(header -> "\t" + header.getName() + ":" + header.getValue())
.collect(Collectors.joining(",\n", "{\n", "\n}"));
}
}
diff --git a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilder.java b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilder.java
index ad15ae999..ee0fbeba7 100644
--- a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilder.java
+++ b/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilder.java
@@ -2,12 +2,11 @@ package org.hl7.fhir.dstu3.utils.client.network;
import java.io.IOException;
import java.util.ArrayList;
-import java.util.Collections;
import java.util.List;
-import java.util.Map;
import java.util.concurrent.TimeUnit;
-import javax.annotation.Nonnull;
+import lombok.Getter;
+import lombok.Setter;
import org.apache.commons.lang3.StringUtils;
import org.hl7.fhir.dstu3.formats.IParser;
@@ -19,33 +18,20 @@ import org.hl7.fhir.dstu3.model.Resource;
import org.hl7.fhir.dstu3.utils.ResourceUtilities;
import org.hl7.fhir.dstu3.utils.client.EFhirClientException;
import org.hl7.fhir.dstu3.utils.client.ResourceFormat;
-import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.utilities.MimeType;
import org.hl7.fhir.utilities.ToolingClientLogger;
-import org.hl7.fhir.utilities.settings.FhirSettings;
+import org.hl7.fhir.utilities.http.*;
-import okhttp3.Authenticator;
-import okhttp3.Credentials;
-import okhttp3.Headers;
-import okhttp3.OkHttpClient;
-import okhttp3.Request;
-import okhttp3.Response;
public class FhirRequestBuilder {
- protected static final String HTTP_PROXY_USER = "http.proxyUser";
- protected static final String HTTP_PROXY_PASS = "http.proxyPassword";
- protected static final String HEADER_PROXY_AUTH = "Proxy-Authorization";
protected static final String LOCATION_HEADER = "location";
protected static final String CONTENT_LOCATION_HEADER = "content-location";
protected static final String DEFAULT_CHARSET = "UTF-8";
- /**
- * The singleton instance of the HttpClient, used for all requests.
- */
- private static OkHttpClient okHttpClient;
- private final Request.Builder httpRequest;
+
+ private final HTTPRequest httpRequest;
private String resourceFormat = null;
- private Headers headers = null;
+ private Iterable headers = null;
private String message = null;
private int retryCount = 1;
/**
@@ -59,60 +45,41 @@ public class FhirRequestBuilder {
/**
* {@link ToolingClientLogger} for log output.
*/
+ @Getter @Setter
private ToolingClientLogger logger = null;
private String source;
- public FhirRequestBuilder(Request.Builder httpRequest, String source) {
+ public FhirRequestBuilder(HTTPRequest httpRequest, String source) {
this.httpRequest = httpRequest;
this.source = source;
}
/**
- * Adds necessary default headers, formatting headers, and any passed in {@link Headers} to the passed in
- * {@link okhttp3.Request.Builder}
+ * Adds necessary default headers, formatting headers, and any passed in {@link HTTPHeader}s to the passed in
+ * {@link HTTPRequest}
*
- * @param request {@link okhttp3.Request.Builder} to add headers to.
+ * @param request {@link HTTPRequest} to add headers to.
* @param format Expected {@link Resource} format.
- * @param headers Any additional {@link Headers} to add to the request.
+ * @param headers Any additional {@link HTTPHeader}s to add to the request.
*/
- protected static void formatHeaders(Request.Builder request, String format, Headers headers) {
- addDefaultHeaders(request, headers);
- if (format != null) addResourceFormatHeaders(request, format);
- if (headers != null) addHeaders(request, headers);
- }
+ protected static HTTPRequest formatHeaders(HTTPRequest request, String format, Iterable headers) {
+ List allHeaders = new ArrayList<>();
+ request.getHeaders().forEach(allHeaders::add);
- /**
- * Adds necessary headers for all REST requests.
- * User-Agent : hapi-fhir-tooling-client
- *
- * @param request {@link Request.Builder} to add default headers to.
- */
- protected static void addDefaultHeaders(Request.Builder request, Headers headers) {
- if (headers == null || !headers.names().contains("User-Agent")) {
- request.addHeader("User-Agent", "hapi-fhir-tooling-client");
- }
- }
-
- /**
- * Adds necessary headers for the given resource format provided.
- *
- * @param request {@link Request.Builder} to add default headers to.
- */
- protected static void addResourceFormatHeaders(Request.Builder request, String format) {
- request.addHeader("Accept", format);
- request.addHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET);
- }
-
- /**
- * Iterates through the passed in {@link Headers} and adds them to the provided {@link Request.Builder}.
- *
- * @param request {@link Request.Builder} to add headers to.
- * @param headers {@link Headers} to add to request.
- */
- protected static void addHeaders(Request.Builder request, Headers headers) {
- if (headers != null) {
- headers.forEach(header -> request.addHeader(header.getFirst(), header.getSecond()));
+ if (format != null) getResourceFormatHeaders(request, format).forEach(allHeaders::add);
+ if (headers != null) headers.forEach(allHeaders::add);
+ return request.withHeaders(allHeaders);
+ }
+ protected static Iterable getResourceFormatHeaders(HTTPRequest httpRequest, String format) {
+ List headers = new ArrayList<>();
+ headers.add(new HTTPHeader("Accept", format));
+ if (httpRequest.getMethod() == HTTPRequest.HttpMethod.PUT
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.POST
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.PATCH
+ ) {
+ headers.add(new HTTPHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET));
}
+ return headers;
}
/**
@@ -131,69 +98,24 @@ public class FhirRequestBuilder {
}
/**
- * Extracts the 'location' header from the passes in {@link Headers}. If no value for 'location' exists, the
- * value for 'content-location' is returned. If neither header exists, we return null.
+ * Extracts the 'location' header from the passed {@link Iterable}. If no
+ * value for 'location' exists, the value for 'content-location' is returned. If
+ * neither header exists, we return null.
*
- * @param headers {@link Headers} to evaluate
+ * @param headers {@link HTTPHeader} to evaluate
* @return {@link String} header value, or null if no location headers are set.
*/
- protected static String getLocationHeader(Headers headers) {
- Map> headerMap = headers.toMultimap();
- if (headerMap.containsKey(LOCATION_HEADER)) {
- return headerMap.get(LOCATION_HEADER).get(0);
- } else if (headerMap.containsKey(CONTENT_LOCATION_HEADER)) {
- return headerMap.get(CONTENT_LOCATION_HEADER).get(0);
- } else {
- return null;
+ protected static String getLocationHeader(Iterable headers) {
+ String locationHeader = HTTPHeaderUtil.getSingleHeader(headers, LOCATION_HEADER);
+
+ if (locationHeader != null) {
+ return locationHeader;
}
+ return HTTPHeaderUtil.getSingleHeader(headers, CONTENT_LOCATION_HEADER);
}
- /**
- * We only ever want to have one copy of the HttpClient kicking around at any given time. If we need to make changes
- * to any configuration, such as proxy settings, timeout, caches, etc, we can do a per-call configuration through
- * the {@link OkHttpClient#newBuilder()} method. That will return a builder that shares the same connection pool,
- * dispatcher, and configuration with the original client.
- *
- * The {@link OkHttpClient} uses the proxy auth properties set in the current system properties. The reason we don't
- * set the proxy address and authentication explicitly, is due to the fact that this class is often used in conjunction
- * with other http client tools which rely on the system.properties settings to determine proxy settings. It's easier
- * to keep the method consistent across the board. ...for now.
- *
- * @return {@link OkHttpClient} instance
- */
- protected OkHttpClient getHttpClient() {
- if (FhirSettings.isProhibitNetworkAccess()) {
- throw new FHIRException("Network Access is prohibited in this context");
- }
-
- if (okHttpClient == null) {
- okHttpClient = new OkHttpClient();
- }
-
- Authenticator proxyAuthenticator = getAuthenticator();
-
- return okHttpClient.newBuilder()
- .addInterceptor(new RetryInterceptor(retryCount))
- .connectTimeout(timeout, timeoutUnit)
- .writeTimeout(timeout, timeoutUnit)
- .readTimeout(timeout, timeoutUnit)
- .proxyAuthenticator(proxyAuthenticator)
- .build();
- }
-
- @Nonnull
- private static Authenticator getAuthenticator() {
- return (route, response) -> {
- final String httpProxyUser = System.getProperty(HTTP_PROXY_USER);
- final String httpProxyPass = System.getProperty(HTTP_PROXY_PASS);
- if (httpProxyUser != null && httpProxyPass != null) {
- String credential = Credentials.basic(httpProxyUser, httpProxyPass);
- return response.request().newBuilder()
- .header(HEADER_PROXY_AUTH, credential)
- .build();
- }
- return response.request().newBuilder().build();
- };
+ protected ManagedFhirWebAccessor getManagedWebAccessor() {
+ return ManagedWebAccess.fhirAccessor().withRetries(retryCount).withTimeout(timeout, timeoutUnit).withLogger(logger);
}
public FhirRequestBuilder withResourceFormat(String resourceFormat) {
@@ -201,7 +123,7 @@ public class FhirRequestBuilder {
return this;
}
- public FhirRequestBuilder withHeaders(Headers headers) {
+ public FhirRequestBuilder withHeaders(Iterable headers) {
this.headers = headers;
return this;
}
@@ -227,25 +149,16 @@ public class FhirRequestBuilder {
return this;
}
- protected Request buildRequest() {
- return httpRequest.build();
- }
-
public ResourceRequest execute() throws IOException {
- formatHeaders(httpRequest, resourceFormat, headers);
- final Request request = httpRequest.build();
- log(request.method(), request.url().toString(), request.headers(), request.body() != null ? request.body().toString().getBytes() : null);
- Response response = getHttpClient().newCall(request).execute();
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, headers);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);
T resource = unmarshalReference(response, resourceFormat);
- return new ResourceRequest(resource, response.code(), getLocationHeader(response.headers()));
+ return new ResourceRequest(resource, response.getCode(), getLocationHeader(response.getHeaders()));
}
public Bundle executeAsBatch() throws IOException {
- formatHeaders(httpRequest, resourceFormat, null);
- final Request request = httpRequest.build();
- log(request.method(), request.url().toString(), request.headers(), request.body() != null ? request.body().toString().getBytes() : null);
-
- Response response = getHttpClient().newCall(request).execute();
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, null);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);
return unmarshalFeed(response, resourceFormat);
}
@@ -253,14 +166,14 @@ public class FhirRequestBuilder {
* Unmarshalls a resource from the response stream.
*/
@SuppressWarnings("unchecked")
- protected T unmarshalReference(Response response, String format) {
+ protected T unmarshalReference(HTTPResult response, String format) {
T resource = null;
OperationOutcome error = null;
- if (response.body() != null) {
+ if (response.getContent() != null) {
try {
- byte[] body = response.body().bytes();
- log(response.code(), response.headers(), body);
+ byte[] body = response.getContent();
+
resource = (T) getParser(format).parse(body);
if (resource instanceof OperationOutcome && hasError((OperationOutcome) resource)) {
error = (OperationOutcome) resource;
@@ -282,13 +195,13 @@ public class FhirRequestBuilder {
/**
* Unmarshalls Bundle from response stream.
*/
- protected Bundle unmarshalFeed(Response response, String format) {
+ protected Bundle unmarshalFeed(HTTPResult response, String format) {
Bundle feed = null;
OperationOutcome error = null;
try {
- byte[] body = response.body().bytes();
- log(response.code(), response.headers(), body);
- String contentType = response.header("Content-Type");
+ byte[] body = response.getContent();
+
+ String contentType = HTTPHeaderUtil.getSingleHeader(response.getHeaders(), "Content-Type");
if (body != null) {
if (contentType.contains(ResourceFormat.RESOURCE_XML.getHeader()) || contentType.contains(ResourceFormat.RESOURCE_JSON.getHeader()) || contentType.contains("text/xml+fhir")) {
Resource rf = getParser(format).parse(body);
@@ -315,7 +228,7 @@ public class FhirRequestBuilder {
/**
* Returns the appropriate parser based on the format type passed in. Defaults to XML parser if a blank format is
* provided...because reasons.
- *
+ *
* Currently supports only "json" and "xml" formats.
*
* @param format One of "json" or "xml".
@@ -334,52 +247,4 @@ public class FhirRequestBuilder {
throw new EFhirClientException("Invalid format: " + format);
}
}
-
- /**
- * Logs the given {@link Request}, using the current {@link ToolingClientLogger}. If the current
- * {@link FhirRequestBuilder#logger} is null, no action is taken.
- *
- * @param method HTTP request method
- * @param url request URL
- * @param requestHeaders {@link Headers} for request
- * @param requestBody Byte array request
- */
- protected void log(String method, String url, Headers requestHeaders, byte[] requestBody) {
- if (logger != null) {
- List headerList = new ArrayList<>(Collections.emptyList());
- Map> headerMap = requestHeaders.toMultimap();
- headerMap.keySet().forEach(key -> headerMap.get(key).forEach(value -> headerList.add(key + ":" + value)));
-
- logger.logRequest(method, url, headerList, requestBody);
- }
-
- }
-
- /**
- * Logs the given {@link Response}, using the current {@link ToolingClientLogger}. If the current
- * {@link FhirRequestBuilder#logger} is null, no action is taken.
- *
- * @param responseCode HTTP response code
- * @param responseHeaders {@link Headers} from response
- * @param responseBody Byte array response
- */
- protected void log(int responseCode, Headers responseHeaders, byte[] responseBody) {
- if (logger != null) {
- List headerList = new ArrayList<>(Collections.emptyList());
- Map> headerMap = responseHeaders.toMultimap();
- headerMap.keySet().forEach(key -> headerMap.get(key).forEach(value -> headerList.add(key + ":" + value)));
-
- try {
- if (logger != null) {
- logger.logResponse(Integer.toString(responseCode), headerList, responseBody, 0);
- }
- } catch (Exception e) {
- System.out.println("Error parsing response body passed in to logger ->\n" + e.getLocalizedMessage());
- }
- }
-// else { // TODO fix logs
-// System.out.println("Call to log HTTP response with null ToolingClientLogger set... are you forgetting to " +
-// "initialize your logger?");
-// }
- }
}
diff --git a/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/FhirToolingClientTest.java b/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/FhirToolingClientTest.java
new file mode 100644
index 000000000..54f363798
--- /dev/null
+++ b/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/FhirToolingClientTest.java
@@ -0,0 +1,225 @@
+package org.hl7.fhir.dstu3.utils.client;
+
+import org.hl7.fhir.dstu3.model.*;
+import org.hl7.fhir.dstu3.utils.client.network.Client;
+import org.hl7.fhir.dstu3.utils.client.network.ResourceRequest;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+import org.mockito.*;
+
+import java.io.IOException;
+import java.net.URI;
+import java.net.URISyntaxException;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.List;
+
+import static org.junit.jupiter.api.Assertions.assertTrue;
+
+public class FhirToolingClientTest {
+ String TX_ADDR = "http://tx.fhir.org";
+
+ HTTPHeader h1 = new HTTPHeader("header1", "value1");
+ HTTPHeader h2 = new HTTPHeader("header2", "value2");
+ HTTPHeader h3 = new HTTPHeader("header3", "value3");
+
+ HTTPHeader agentHeader = new HTTPHeader("User-Agent", "fhir/test-cases");
+
+ private Client mockClient;
+ private FHIRToolingClient toolingClient;
+
+ @Captor
+ private ArgumentCaptor> headersArgumentCaptor;
+
+
+ @BeforeEach
+ void setUp() throws IOException, URISyntaxException {
+ MockitoAnnotations.openMocks(this);
+ mockClient = Mockito.mock(Client.class);
+ ResourceRequest resourceResourceRequest = new ResourceRequest<>(generateBundle(), 200, "");
+
+ // GET
+ Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ Mockito
+ .when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new Parameters(), 200, "location"));
+ Mockito
+ .when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
+ Mockito
+ .when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
+
+ // PUT
+ Mockito.when(mockClient.issuePutRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ // POST
+ Mockito.when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ Mockito
+ .when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.contains("validate"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new OperationOutcome(), 200, "location"));
+ // BUNDLE REQ
+ Mockito
+ .when(mockClient.executeBundleRequest(Mockito.any(HTTPRequest.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyLong()))
+ .thenReturn(generateBundle());
+ toolingClient = new FHIRToolingClient(TX_ADDR, "fhir/test-cases");
+ toolingClient.setClient(mockClient);
+ }
+
+ private List getHeaders() {
+ return new ArrayList<>(Arrays.asList(h1, h2, h3));
+ }
+
+ private List getHeadersWithAgent() {
+ return new ArrayList<>(Arrays.asList(h1, h2, h3, agentHeader));
+ }
+
+ private Bundle generateBundle() {
+ Patient patient = generatePatient();
+ Observation observation = generateObservation();
+
+ // The observation refers to the patient using the ID, which is already
+ // set to a temporary UUID
+ observation.setSubject(new Reference(patient.getIdElement().getValue()));
+
+ // Create a bundle that will be used as a transaction
+ Bundle bundle = new Bundle();
+
+ // Add the patient as an entry.
+ bundle.addEntry().setFullUrl(patient.getIdElement().getValue()).setResource(patient).getRequest().setUrl("Patient")
+ .setIfNoneExist("identifier=http://acme.org/mrns|12345").setMethod(Bundle.HTTPVerb.POST);
+
+ return bundle;
+ }
+
+ private Patient generatePatient() {
+ // Create a patient object
+ Patient patient = new Patient();
+ patient.addIdentifier().setSystem("http://acme.org/mrns").setValue("12345");
+ patient.addName().setFamily("Jameson").addGiven("J").addGiven("Jonah");
+ patient.setGender(Enumerations.AdministrativeGender.MALE);
+
+ // Give the patient a temporary UUID so that other resources in
+ // the transaction can refer to it
+ patient.setId(IdType.newRandomUuid());
+ return patient;
+ }
+
+ private Observation generateObservation() {
+ // Create an observation object
+ Observation observation = new Observation();
+ observation.getCode().addCoding().setSystem("http://loinc.org").setCode("789-8")
+ .setDisplay("Erythrocytes [#/volume] in Blood by Automated count");
+ observation.setValue(new Quantity().setValue(4.12).setUnit("10 trillion/L").setSystem("http://unitsofmeasure.org")
+ .setCode("10*12/L"));
+ return observation;
+ }
+
+ private void checkHeaders(Iterable argumentCaptorValue) {
+ List capturedHeaders = new ArrayList<>();
+ argumentCaptorValue.forEach(capturedHeaders::add);
+
+ getHeadersWithAgent().forEach(header -> {
+ assertTrue(capturedHeaders.contains(header));
+ });
+ }
+
+ @Test
+ void getTerminologyCapabilities() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getTerminologyCapabilities();
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void getCapabilitiesStatement() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getCapabilitiesStatement();
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void getCapabilitiesStatementQuick() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getCapabilitiesStatementQuick();
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void read() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.read(Patient.class, "id");
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void vread() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.vread(Patient.class, "id", "version");
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void getCanonical() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getCanonical(Patient.class, "canonicalURL");
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void update() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.update(generatePatient());
+ Mockito.verify(mockClient).issuePutRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
+ ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
+ ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void validate() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.validate(Patient.class, generatePatient(), "id");
+ Mockito.verify(mockClient).issuePostRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
+ ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
+ ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+}
diff --git a/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/network/ClientTest.java b/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/network/ClientTest.java
new file mode 100644
index 000000000..18f1bed48
--- /dev/null
+++ b/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/network/ClientTest.java
@@ -0,0 +1,169 @@
+package org.hl7.fhir.dstu3.utils.client.network;
+
+import okhttp3.HttpUrl;
+import okhttp3.mockwebserver.MockResponse;
+import okhttp3.mockwebserver.MockWebServer;
+import okhttp3.mockwebserver.RecordedRequest;
+import org.hl7.fhir.dstu3.formats.JsonParser;
+import org.hl7.fhir.dstu3.model.*;
+import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Test;
+import org.mockito.Mockito;
+
+import java.io.IOException;
+import java.net.URI;
+import java.net.URISyntaxException;
+import java.util.Random;
+import java.util.concurrent.TimeUnit;
+
+public class ClientTest {
+ private static final long TIMEOUT = 5000;
+
+ private MockWebServer server;
+ private HttpUrl serverUrl;
+ private Client client;
+
+ private final Address address = new Address()
+ .setCity("Toronto")
+ .setState("Ontario")
+ .setCountry("Canada");
+ private final HumanName humanName = new HumanName()
+ .addGiven("Mark")
+ .setFamily("Iantorno");
+ private final Patient patient = new Patient()
+ .addName(humanName)
+ .addAddress(address)
+ .setGender(Enumerations.AdministrativeGender.MALE);
+
+ @BeforeEach
+ void setup() {
+ setupMockServer();
+ client = new Client();
+ }
+
+ void setupMockServer() {
+ server = new MockWebServer();
+ serverUrl = server.url("/v1/endpoint");
+ }
+
+ byte[] generateResourceBytes(Resource resource) throws IOException {
+ return new JsonParser().composeBytes(resource);
+ }
+
+ @Test
+ @DisplayName("GET request, happy path.")
+ void test_get_happy_path() throws IOException, URISyntaxException {
+ server.enqueue(
+ new MockResponse()
+ .setBody(new String(generateResourceBytes(patient)))
+ );
+ ResourceRequest resourceRequest = client.issueGetResourceRequest(new URI(serverUrl.toString()),
+ "json", null, null, TIMEOUT);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+ @Test
+ @DisplayName("GET request, test client retries after timeout failure.")
+ void test_get_retries_with_timeout() throws IOException, URISyntaxException {
+ int failedAttempts = new Random().nextInt(5) + 1;
+ System.out.println("Simulating <" + failedAttempts + "> failed connections (timeouts) before success.");
+ for (int i = 0; i < failedAttempts; i++) {
+ server.enqueue(
+ new MockResponse()
+ .setHeadersDelay(TIMEOUT * 10, TimeUnit.MILLISECONDS)
+ .setBody(new String(generateResourceBytes(patient)))
+ );
+ }
+ server.enqueue(new MockResponse().setBody(new String(generateResourceBytes(patient))));
+ client.setRetryCount(failedAttempts + 1);
+
+ ResourceRequest resourceRequest = client.issueGetResourceRequest(new URI(serverUrl.toString()),
+ "json", null, null, TIMEOUT);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+ @Test
+ @DisplayName("GET request, test client retries after bad response.")
+ void test_get_retries_with_unsuccessful_response() throws IOException, URISyntaxException {
+ int failedAttempts = new Random().nextInt(5) + 1;
+ System.out.println("Simulating <" + failedAttempts + "> failed connections (bad response codes) before success.");
+ for (int i = 0; i < failedAttempts; i++) {
+ server.enqueue(
+ new MockResponse()
+ .setResponseCode(400 + i)
+ .setBody(new String(generateResourceBytes(patient)))
+ );
+ }
+ server.enqueue(new MockResponse().setBody(new String(generateResourceBytes(patient))));
+ client.setRetryCount(failedAttempts + 1);
+
+ ResourceRequest resourceRequest = client.issueGetResourceRequest(new URI(serverUrl.toString()),
+ "json", null, null, TIMEOUT);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+ @Test
+ @DisplayName("PUT request, test payload received by server matches sent.")
+ void test_put() throws IOException, URISyntaxException, InterruptedException {
+ byte[] payload = ByteUtils.resourceToByteArray(patient, true, false, false);
+ // Mock server response of 200, with the same resource payload returned that we included in the PUT request
+ server.enqueue(
+ new MockResponse()
+ .setResponseCode(200)
+ .setBody(new String(payload))
+ );
+
+ ResourceRequest request = client.issuePutRequest(new URI(serverUrl.toString()), payload,
+ "xml", null, TIMEOUT);
+ RecordedRequest recordedRequest = server.takeRequest();
+ Assertions.assertArrayEquals(payload, recordedRequest.getBody().readByteArray(),
+ "PUT request payload does not match send data.");
+ }
+
+ @Test
+ @DisplayName("POST request, test payload received by server matches sent.")
+ void test_post() throws IOException, URISyntaxException, InterruptedException {
+ byte[] payload = ByteUtils.resourceToByteArray(patient, true, false, false);
+ // Mock server response of 200, with the same resource payload returned that we included in the PUT request
+ server.enqueue(
+ new MockResponse()
+ .setResponseCode(200)
+ .setBody(new String(payload))
+ );
+
+ ResourceRequest request = client.issuePostRequest(new URI(serverUrl.toString()), payload,
+ "xml", null, TIMEOUT);
+ RecordedRequest recordedRequest = server.takeRequest();
+ Assertions.assertArrayEquals(payload, recordedRequest.getBody().readByteArray(),
+ "POST request payload does not match send data.");
+ }
+
+ @Test
+ @DisplayName("Testing the logger works.")
+ void test_logger() throws IOException, URISyntaxException, InterruptedException {
+ byte[] payload = ByteUtils.resourceToByteArray(patient, true, false, false);
+ server.enqueue(
+ new MockResponse()
+ .setResponseCode(200)
+ .setBody(new String(payload))
+ );
+ ToolingClientLogger mockLogger = Mockito.mock(ToolingClientLogger.class);
+ client.setLogger(mockLogger);
+ client.issuePostRequest(new URI(serverUrl.toString()), payload,
+ "xml", null, TIMEOUT);
+ server.takeRequest();
+ Mockito.verify(mockLogger, Mockito.times(1))
+ .logRequest(Mockito.anyString(), Mockito.anyString(), Mockito.anyList(), Mockito.any());
+ Mockito.verify(mockLogger, Mockito.times(1))
+ .logResponse(Mockito.anyString(), Mockito.anyList(), Mockito.any(), Mockito.anyLong());
+ }
+}
diff --git a/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilderTests.java b/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilderTests.java
index 7cd2b5edd..b5789c50b 100644
--- a/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilderTests.java
+++ b/org.hl7.fhir.dstu3/src/test/java/org/hl7/fhir/dstu3/utils/client/network/FhirRequestBuilderTests.java
@@ -1,107 +1,126 @@
package org.hl7.fhir.dstu3.utils.client.network;
-import java.io.IOException;
+import java.util.Collections;
+import java.util.List;
+import java.util.Map;
-import org.hl7.fhir.dstu3.formats.IParser;
-import org.hl7.fhir.utilities.ToolingClientLogger;
-import org.junit.jupiter.api.BeforeEach;
-import org.junit.jupiter.api.Nested;
-import org.junit.jupiter.api.Test;
+import org.hl7.fhir.dstu3.model.OperationOutcome;
+import org.hl7.fhir.utilities.http.*;
+import org.junit.jupiter.api.*;
import org.junit.jupiter.api.extension.ExtendWith;
-import org.mockito.AdditionalMatchers;
-import org.mockito.ArgumentMatchers;
-import org.mockito.Mock;
-import org.mockito.Mockito;
import org.mockito.junit.jupiter.MockitoExtension;
-import okhttp3.Call;
-import okhttp3.MediaType;
-import okhttp3.OkHttpClient;
-import okhttp3.Protocol;
-import okhttp3.Request;
-import okhttp3.Response;
-import okhttp3.ResponseBody;
-
@ExtendWith(MockitoExtension.class)
public class FhirRequestBuilderTests {
- private static final String DUMMY_URL = "https://some-url.com/";
+ @Test
+ @DisplayName("Test resource format headers are added correctly (GET).")
+ void addResourceFormatHeadersGET() {
+ String testFormat = "yaml";
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.GET);
- Request mockRequest = new Request.Builder()
- .url(DUMMY_URL)
- .build();
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
- final String RESPONSE_BODY_STRING = "{}";
-
- Response response = new Response.Builder()
- .request(mockRequest)
- .protocol(Protocol.HTTP_2)
- .code(200) // status code
- .message("")
- .body(ResponseBody.create(RESPONSE_BODY_STRING,
- MediaType.get("application/json; charset=utf-8")
- ))
- .addHeader("Content-Type", "")
- .build();
-
- final Request.Builder requestBuilder = new Request.Builder()
- .url(DUMMY_URL);
-
- final FhirRequestBuilder fhirRequestBuilder = Mockito.spy(new FhirRequestBuilder(requestBuilder, "http://local/local"));
-
- @Mock
- OkHttpClient client;
-
- @Mock
- Call mockCall;
-
- @Mock
- ToolingClientLogger logger;
-
- @BeforeEach
- public void beforeEach() {
- Mockito.doReturn(client).when(fhirRequestBuilder).getHttpClient();
- fhirRequestBuilder.withLogger(logger);
- }
-
- @Nested
- class RequestLoggingTests {
-
- @BeforeEach
- public void beforeEach() throws IOException {
- Mockito.doReturn(response).when(mockCall).execute();
- Mockito.doReturn(mockCall).when(client).newCall(ArgumentMatchers.any());
-
- Mockito.doReturn(null).when(fhirRequestBuilder).unmarshalReference(ArgumentMatchers.any(), ArgumentMatchers.isNull());
- }
-
- @Test
- public void testExecuteLogging() throws IOException {
- fhirRequestBuilder.execute();
- Mockito.verify(logger).logRequest(ArgumentMatchers.eq("GET"), ArgumentMatchers.eq(DUMMY_URL), ArgumentMatchers.anyList(), ArgumentMatchers.isNull());
- }
-
- @Test
- public void testExecuteBatchLogging() throws IOException {
- fhirRequestBuilder.executeAsBatch();
- Mockito.verify(logger).logRequest(ArgumentMatchers.eq("GET"), ArgumentMatchers.eq(DUMMY_URL), ArgumentMatchers.anyList(), ArgumentMatchers.isNull());
- }
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
+ Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
+ Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
+ "Accept header not populated with expected value " + testFormat + ".");
+ Assertions.assertNull(headersMap.get("Content-Type"), "Content-Type header not null.");
}
@Test
- public void testUnmarshallReferenceLogging() {
- IParser parser = Mockito.mock(IParser.class);
- Mockito.doReturn(parser).when(fhirRequestBuilder).getParser(ArgumentMatchers.eq("json"));
+ @DisplayName("Test resource format headers are added correctly (POST).")
+ void addResourceFormatHeadersPOST() {
+ String testFormat = "yaml";
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.POST);
- fhirRequestBuilder.unmarshalReference(response, "json");
- Mockito.verify(logger).logResponse(ArgumentMatchers.eq("200"), ArgumentMatchers.anyList(), AdditionalMatchers.aryEq(RESPONSE_BODY_STRING.getBytes()), ArgumentMatchers.anyLong());
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
+ Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
+ Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
+ "Accept header not populated with expected value " + testFormat + ".");
+
+ Assertions.assertNotNull(headersMap.get("Content-Type"), "Content-Type header null.");
+ Assertions.assertEquals(testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET, headersMap.get("Content-Type").get(0),
+ "Content-Type header not populated with expected value \"" + testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET + "\".");
}
@Test
- public void testUnmarshallFeedLogging() {
- fhirRequestBuilder.unmarshalFeed(response, "application/json");
- Mockito.verify(logger).logResponse(ArgumentMatchers.eq("200"), ArgumentMatchers.anyList(), AdditionalMatchers.aryEq(RESPONSE_BODY_STRING.getBytes()), ArgumentMatchers.anyLong());
+ @DisplayName("Test that FATAL issue severity triggers error.")
+ void hasErrorTestFatal() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.FATAL));
+ Assertions.assertTrue(FhirRequestBuilder.hasError(outcome), "Error check not triggered for FATAL issue severity.");
}
+ @Test
+ @DisplayName("Test that ERROR issue severity triggers error.")
+ void hasErrorTestError() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.ERROR));
+ Assertions.assertTrue(FhirRequestBuilder.hasError(outcome), "Error check not triggered for ERROR issue severity.");
+ }
+
+ @Test
+ @DisplayName("Test that no FATAL or ERROR issue severity does not trigger error.")
+ void hasErrorTestNoErrors() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ Assertions.assertFalse(FhirRequestBuilder.hasError(outcome), "Error check triggered unexpectedly.");
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns header for 'location'.")
+ void getLocationHeaderWhenOnlyLocationIsSet() {
+ final String expectedLocationHeader = "location_header_value";
+ HTTPResult result = new HTTPResult("source",
+ 200,
+ "message",
+ "contentType",
+ new byte[0],
+ List.of(new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader)));
+
+ Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(result.getHeaders()));
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns header for 'content-location'.")
+ void getLocationHeaderWhenOnlyContentLocationIsSet() {
+ final String expectedContentLocationHeader = "content_location_header_value";
+ Iterable headers = List.of(new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader));
+
+ Assertions.assertEquals(expectedContentLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns 'location' header when both 'location' and 'content-location' are set.")
+ void getLocationHeaderWhenLocationAndContentLocationAreSet() {
+ final String expectedLocationHeader = "location_header_value";
+ final String expectedContentLocationHeader = "content_location_header_value";
+
+ Iterable headers = List.of(
+ new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader),
+ new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader)
+ );
+
+ Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns null when no location available.")
+ void getLocationHeaderWhenNoLocationSet() {
+ Assertions.assertNull(FhirRequestBuilder.getLocationHeader(Collections.emptyList()));
+ }
+
+
}
diff --git a/org.hl7.fhir.r4/pom.xml b/org.hl7.fhir.r4/pom.xml
index ead565cbb..179e00ac6 100644
--- a/org.hl7.fhir.r4/pom.xml
+++ b/org.hl7.fhir.r4/pom.xml
@@ -35,6 +35,12 @@
+
+ org.projectlombok
+ lombok
+ provided
+
+
org.fhir
@@ -78,7 +84,12 @@
okio-jvm
true
-
+
+ com.squareup.okhttp3
+ mockwebserver
+ true
+ test
+
org.hl7.fhir.testcases
fhir-test-cases
diff --git a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/FHIRToolingClient.java b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/FHIRToolingClient.java
index b07046dbe..11451ad42 100644
--- a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/FHIRToolingClient.java
+++ b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/FHIRToolingClient.java
@@ -3,11 +3,10 @@ package org.hl7.fhir.r4.utils.client;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
-import java.util.ArrayList;
-import java.util.Base64;
-import java.util.HashMap;
-import java.util.Map;
+import java.util.*;
+import lombok.Getter;
+import lombok.Setter;
import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.r4.model.Bundle;
import org.hl7.fhir.r4.model.CapabilityStatement;
@@ -29,8 +28,7 @@ import org.hl7.fhir.utilities.FHIRBaseToolingClient;
import org.hl7.fhir.utilities.ToolingClientLogger;
import org.hl7.fhir.utilities.Utilities;
-import okhttp3.Headers;
-import okhttp3.internal.http2.Header;
+import org.hl7.fhir.utilities.http.HTTPHeader;
/**
* Very Simple RESTful client. This is purely for use in the standalone tools
@@ -70,16 +68,22 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
private String base;
private ResourceAddress resourceAddress;
+ @Setter
private ResourceFormat preferredResourceFormat;
private int maxResultSetSize = -1;// _count
private CapabilityStatement capabilities;
+ @Getter
+ @Setter
private Client client = new Client();
- private ArrayList headers = new ArrayList<>();
- private String username;
- private String password;
+ private List headers = new ArrayList<>();
+
+ @Getter @Setter
private String userAgent;
- private String acceptLang;
- private String contentLang;
+ @Setter
+ private String acceptLanguage;
+
+ @Setter
+ private String contentLanguage;
private int useCount;
// Pass endpoint for client - URI
@@ -96,14 +100,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
this.maxResultSetSize = -1;
}
- public Client getClient() {
- return client;
- }
-
- public void setClient(Client client) {
- this.client = client;
- }
-
private void checkCapabilities() {
try {
capabilities = getCapabilitiesStatementQuick();
@@ -115,10 +111,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return preferredResourceFormat.getHeader();
}
- public void setPreferredResourceFormat(ResourceFormat resourceFormat) {
- preferredResourceFormat = resourceFormat;
- }
-
public int getMaximumRecordCount() {
return maxResultSetSize;
}
@@ -131,7 +123,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
TerminologyCapabilities capabilities = null;
try {
capabilities = (TerminologyCapabilities) client.issueGetResourceRequest(resourceAddress.resolveMetadataTxCaps(),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "TerminologyCapabilities", timeoutNormal).getReference();
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false), "TerminologyCapabilities", timeoutNormal).getReference();
} catch (Exception e) {
throw new FHIRException("Error fetching the server's terminology capabilities", e);
}
@@ -142,7 +134,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
CapabilityStatement conformance = null;
try {
conformance = (CapabilityStatement) client.issueGetResourceRequest(resourceAddress.resolveMetadataUri(false),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "CapabilitiesStatement", timeoutNormal).getReference();
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false), "CapabilitiesStatement", timeoutNormal).getReference();
} catch (Exception e) {
throw new FHIRException("Error fetching the server's conformance statement", e);
}
@@ -154,7 +146,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return capabilities;
try {
capabilities = (CapabilityStatement) client.issueGetResourceRequest(resourceAddress.resolveMetadataUri(true),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "CapabilitiesStatement-Quick", timeoutNormal)
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false), "CapabilitiesStatement-Quick", timeoutNormal)
.getReference();
} catch (Exception e) {
throw new FHIRException("Error fetching the server's capability statement: " + e.getMessage(), e);
@@ -167,7 +159,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
ResourceRequest result = null;
try {
result = client.issueGetResourceRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "Read " + resourceClass + "/" + id,
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false), "Read " + resourceClass + "/" + id,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -184,7 +176,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
ResourceRequest result = null;
try {
result = client.issueGetResourceRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "Read " + resourceClass.getName() + "/" + id,
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false), "Read " + resourceClass.getName() + "/" + id,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -202,7 +194,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issueGetResourceRequest(
resourceAddress.resolveGetUriFromResourceClassAndIdAndVersion(resourceClass, id, version),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(),
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false),
"VRead " + resourceClass.getName() + "/" + id + "/?_history/" + version, timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -220,7 +212,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issueGetResourceRequest(
resourceAddress.resolveGetUriFromResourceClassAndCanonical(resourceClass, canonicalURL),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "Read " + resourceClass.getName() + "?url=" + canonicalURL,
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false), "Read " + resourceClass.getName() + "?url=" + canonicalURL,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -244,7 +236,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePutRequest(
resourceAddress.resolveGetUriFromResourceClassAndId(resource.getClass(), resource.getId()),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "Update " + resource.fhirType() + "/" + resource.getId(),
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true), "Update " + resource.fhirType() + "/" + resource.getId(),
timeoutOperation);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -274,7 +266,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issuePutRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "Update " + resource.fhirType() + "/" + id,
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true), "Update " + resource.fhirType() + "/" + id,
timeoutOperation);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -312,10 +304,10 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
URI url = resourceAddress.resolveOperationURLFromClass(resourceClass, name, ps);
if (complex) {
byte[] body = ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true);
- result = client.issuePostRequest(url, body, withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(),
+ result = client.issuePostRequest(url, body, withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true),
"POST " + resourceClass.getName() + "/$" + name, timeoutLong);
} else {
- result = client.issueGetResourceRequest(url, withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(),
+ result = client.issueGetResourceRequest(url, withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false),
"GET " + resourceClass.getName() + "/$" + name, timeoutLong);
}
if (result.isUnsuccessfulRequest()) {
@@ -351,7 +343,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issuePostRequest(resourceAddress.resolveValidateUri(resourceClass, id),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(),
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true),
"POST " + resourceClass.getName() + (id != null ? "/" + id : "") + "/$validate", timeoutLong);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -364,10 +356,14 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
}
/**
- * Helper method to prevent nesting of previously thrown EFhirClientExceptions
+ * Helper method to prevent nesting of previously thrown EFhirClientExceptions. If the e param is an instance of
+ * EFhirClientException, it will be rethrown. Otherwise, a new EFhirClientException will be thrown with e as the
+ * cause.
*
- * @param e
- * @throws EFhirClientException
+ * @param code The EFhirClientException code.
+ * @param message The EFhirClientException message.
+ * @param e The exception.
+ * @throws EFhirClientException representing the exception.
*/
protected void handleException(int code, String message, Exception e) throws EFhirClientException {
if (e instanceof EFhirClientException) {
@@ -378,11 +374,11 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
}
/**
- * Helper method to determine whether desired resource representation is Json or
- * XML.
+ * Helper method to determine whether desired resource representation
+ * is Json or XML.
*
- * @param format
- * @return
+ * @param format The format
+ * @return true if the format is JSON, false otherwise
*/
protected boolean isJson(String format) {
boolean isJson = false;
@@ -408,7 +404,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
org.hl7.fhir.r4.utils.client.network.ResourceRequest result = null;
try {
result = client.issueGetResourceRequest(resourceAddress.resolveOperationUri(CodeSystem.class, "lookup", params),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "CodeSystem/$lookup", timeoutNormal);
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(false), "CodeSystem/$lookup", timeoutNormal);
} catch (IOException e) {
throw new FHIRException(e);
}
@@ -425,7 +421,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(CodeSystem.class, "lookup"),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "CodeSystem/$lookup", timeoutNormal);
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true), "CodeSystem/$lookup", timeoutNormal);
} catch (IOException e) {
throw new FHIRException(e);
}
@@ -442,7 +438,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(ConceptMap.class, "translate"),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "ConceptMap/$translate", timeoutNormal);
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true), "ConceptMap/$translate", timeoutNormal);
} catch (IOException e) {
throw new FHIRException(e);
}
@@ -463,7 +459,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
try {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(ValueSet.class, "expand"),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true), withVer(getPreferredResourceFormat(), "4.0"),
- generateHeaders(), source == null ? "ValueSet/$expand" : "ValueSet/$expand?url=" + source.getUrl(),
+ generateHeaders(true), source == null ? "ValueSet/$expand" : "ValueSet/$expand?url=" + source.getUrl(),
timeoutExpand);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
@@ -494,7 +490,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePostRequest(
resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "Closure?name=" + name, timeoutNormal);
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true), "Closure?name=" + name, timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
(OperationOutcome) result.getPayload());
@@ -515,7 +511,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
result = client.issuePostRequest(
resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), "UpdateClosure?name=" + name, timeoutOperation);
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true), "UpdateClosure?name=" + name, timeoutOperation);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException(result.getHttpStatus(), "Server returned error code " + result.getHttpStatus(),
(OperationOutcome) result.getPayload());
@@ -526,22 +522,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return result == null ? null : (ConceptMap) result.getPayload();
}
- public String getUsername() {
- return username;
- }
-
- public void setUsername(String username) {
- this.username = username;
- }
-
- public String getPassword() {
- return password;
- }
-
- public void setPassword(String password) {
- this.password = password;
- }
-
public ToolingClientLogger getLogger() {
return client.getLogger();
}
@@ -558,50 +538,27 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
client.setRetryCount(retryCount);
}
- public void setClientHeaders(ArrayList headers) {
- this.headers = headers;
+ public void setClientHeaders(Iterable headers) {
+ this.headers = new ArrayList<>();
+ headers.forEach(this.headers::add);
}
- private Headers generateHeaders() {
- Headers.Builder builder = new Headers.Builder();
- // Add basic auth header if it exists
- if (basicAuthHeaderExists()) {
- builder.add(getAuthorizationHeader().toString());
- }
+ private Iterable generateHeaders(boolean hasBody) {
// Add any other headers
- if (this.headers != null) {
- this.headers.forEach(header -> builder.add(header.toString()));
- }
+ List headers = new ArrayList<>(this.headers);
if (!Utilities.noString(userAgent)) {
- builder.add("User-Agent: " + userAgent);
+ headers.add(new HTTPHeader("User-Agent",userAgent));
}
- if (!Utilities.noString(acceptLang)) {
- builder.add("Accept-Language: "+acceptLang);
+ if (!Utilities.noString(acceptLanguage)) {
+ headers.add(new HTTPHeader("Accept-Language", acceptLanguage));
}
- if (!Utilities.noString(contentLang)) {
- builder.add("Content-Language: "+contentLang);
+
+ if (hasBody && !Utilities.noString(contentLanguage)) {
+ headers.add(new HTTPHeader("Content-Language",contentLanguage));
}
-
- return builder.build();
- }
- public boolean basicAuthHeaderExists() {
- return (username != null) && (password != null);
- }
-
- public Header getAuthorizationHeader() {
- String usernamePassword = username + ":" + password;
- String base64usernamePassword = Base64.getEncoder().encodeToString(usernamePassword.getBytes());
- return new Header("Authorization", "Basic " + base64usernamePassword);
- }
-
- public String getUserAgent() {
- return userAgent;
- }
-
- public void setUserAgent(String userAgent) {
- this.userAgent = userAgent;
+ return headers;
}
public String getServerVersion() {
@@ -609,14 +566,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return capabilities == null ? null : capabilities.getSoftware().getVersion();
}
- public void setAcceptLanguage(String lang) {
- this.acceptLang = lang;
- }
-
- public void setContentLanguage(String lang) {
- this.acceptLang = lang;
- }
-
public Bundle search(String type, String criteria) {
recordUse();
return fetchFeed(Utilities.pathURL(base, type+criteria));
@@ -627,7 +576,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
org.hl7.fhir.r4.utils.client.network.ResourceRequest result = null;
try {
result = client.issueGetResourceRequest(resourceAddress.resolveGetResource(resourceClass, id),
- withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(), resourceClass.getName()+"/"+id, timeoutNormal);
+ withVer(getPreferredResourceFormat(), "4.0"), generateHeaders(true), resourceClass.getName()+"/"+id, timeoutNormal);
} catch (IOException e) {
throw new FHIRException(e);
}
diff --git a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/Client.java b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/Client.java
index 37918812b..a17a4f8d7 100644
--- a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/Client.java
+++ b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/Client.java
@@ -2,148 +2,149 @@ package org.hl7.fhir.r4.utils.client.network;
import java.io.IOException;
import java.net.URI;
+import java.util.Collections;
import java.util.Map;
import java.util.concurrent.TimeUnit;
+import lombok.Getter;
+import lombok.Setter;
import org.hl7.fhir.r4.model.Bundle;
import org.hl7.fhir.r4.model.Resource;
import org.hl7.fhir.r4.utils.client.EFhirClientException;
import org.hl7.fhir.utilities.ToolingClientLogger;
-import okhttp3.Headers;
-import okhttp3.MediaType;
-import okhttp3.Request;
-import okhttp3.RequestBody;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPRequest;
public class Client {
-
-
public static final String DEFAULT_CHARSET = "UTF-8";
+
+ @Getter @Setter
private ToolingClientLogger logger;
- private FhirLoggingInterceptor fhirLoggingInterceptor;
+
+ @Getter @Setter
private int retryCount;
+
+ @Getter @Setter
private String base;
-
- public String getBase() {
- return base;
- }
-
- public void setBase(String base) {
- this.base = base;
- }
-
-
- public ToolingClientLogger getLogger() {
- return logger;
- }
-
- public void setLogger(ToolingClientLogger logger) {
- this.logger = logger;
- this.fhirLoggingInterceptor = new FhirLoggingInterceptor(logger);
- }
-
- public int getRetryCount() {
- return retryCount;
- }
-
- public void setRetryCount(int retryCount) {
- this.retryCount = retryCount;
- }
public ResourceRequest issueOptionsRequest(URI optionsUri, String resourceFormat,
String message, long timeout) throws IOException {
- Request.Builder request = new Request.Builder().method("OPTIONS", null).url(optionsUri.toURL());
- return executeFhirRequest(request, resourceFormat, new Headers.Builder().build(), message, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(optionsUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.OPTIONS);
+ return executeFhirRequest(request, resourceFormat, Collections.emptyList(), message, retryCount, timeout);
}
public ResourceRequest issueGetResourceRequest(URI resourceUri, String resourceFormat,
- Headers headers, String message, long timeout) throws IOException {
- Request.Builder request = new Request.Builder().url(resourceUri.toURL());
-
+ Iterable headers, String message, long timeout) throws IOException {
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
- public int tester(int trytry) {
- return 5;
- }
-
public ResourceRequest issuePutRequest(URI resourceUri, byte[] payload, String resourceFormat,
String message, long timeout) throws IOException {
- return issuePutRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePutRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePutRequest(URI resourceUri, byte[] payload, String resourceFormat,
- Headers headers, String message, long timeout) throws IOException {
+ Iterable headers, String message, long timeout) throws IOException {
if (payload == null)
throw new EFhirClientException("PUT requests require a non-null payload");
- RequestBody body = RequestBody.create(payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).put(body);
+
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.PUT)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
public ResourceRequest issuePostRequest(URI resourceUri, byte[] payload,
String resourceFormat, String message, long timeout) throws IOException {
- return issuePostRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePostRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePostRequest(URI resourceUri, byte[] payload,
- String resourceFormat, Headers headers, String message, long timeout) throws IOException {
+ String resourceFormat, Iterable headers, String message, long timeout) throws IOException {
if (payload == null)
throw new EFhirClientException("POST requests require a non-null payload");
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).post(body);
+
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
public boolean issueDeleteRequest(URI resourceUri, int timeout) throws IOException {
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).delete();
- return executeFhirRequest(request, null, new Headers.Builder().build(), null, retryCount, timeout)
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.DELETE);
+ return executeFhirRequest(request, null, Collections.emptyList(), null, retryCount, timeout)
.isSuccessfulRequest();
}
public Bundle issueGetFeedRequest(URI resourceUri, String resourceFormat, int timeout) throws IOException {
- Request.Builder request = new Request.Builder().url(resourceUri.toURL());
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
public Bundle issuePostFeedRequest(URI resourceUri, Map parameters, String resourceName,
Resource resource, String resourceFormat, int timeout) throws IOException {
String boundary = "----WebKitFormBoundarykbMUo6H8QaUnYtRy";
byte[] payload = ByteUtils.encodeFormSubmission(parameters, resourceName, resource, boundary);
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).post(body);
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
+
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
public Bundle postBatchRequest(URI resourceUri, byte[] payload, String resourceFormat, String message, int timeout)
throws IOException {
if (payload == null)
throw new EFhirClientException("POST requests require a non-null payload");
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).post(body);
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), message, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
+
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), message, retryCount, timeout);
}
- public Bundle executeBundleRequest(Request.Builder request, String resourceFormat,
- Headers headers, String message, int retryCount, long timeout) throws IOException {
- return new FhirRequestBuilder(request, base).withLogger(fhirLoggingInterceptor).withResourceFormat(resourceFormat)
+ private static String getContentTypeWithDefaultCharset(String resourceFormat) {
+ return resourceFormat + ";charset=" + DEFAULT_CHARSET;
+ }
+
+ public Bundle executeBundleRequest(HTTPRequest request, String resourceFormat,
+ Iterable headers, String message, int retryCount, long timeout) throws IOException {
+ return new FhirRequestBuilder(request, base).withLogger(logger).withResourceFormat(resourceFormat)
.withRetryCount(retryCount).withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS).executeAsBatch();
}
- public ResourceRequest executeFhirRequest(Request.Builder request, String resourceFormat,
- Headers headers, String message, int retryCount, long timeout) throws IOException {
- return new FhirRequestBuilder(request, base).withLogger(fhirLoggingInterceptor).withResourceFormat(resourceFormat)
+ public ResourceRequest executeFhirRequest(HTTPRequest request, String resourceFormat,
+ Iterable headers, String message, int retryCount, long timeout) throws IOException {
+ return new FhirRequestBuilder(request, base).withLogger(logger).withResourceFormat(resourceFormat)
.withRetryCount(retryCount).withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS).execute();
}
}
diff --git a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/ClientHeaders.java b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/ClientHeaders.java
index 6a1a6ad6a..ef42c73b4 100644
--- a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/ClientHeaders.java
+++ b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/ClientHeaders.java
@@ -6,7 +6,7 @@ import java.util.stream.Collectors;
import org.hl7.fhir.exceptions.FHIRException;
-import okhttp3.internal.http2.Header;
+import org.hl7.fhir.utilities.http.HTTPHeader;
/**
* Generic Implementation of Client Headers.
@@ -16,29 +16,29 @@ import okhttp3.internal.http2.Header;
*/
public class ClientHeaders {
- private final ArrayList headers;
+ private final ArrayList headers;
public ClientHeaders() {
this.headers = new ArrayList<>();
}
- public ClientHeaders(ArrayList headers) {
+ public ClientHeaders(ArrayList headers) {
this.headers = headers;
}
- public ArrayList headers() {
+ public ArrayList headers() {
return headers;
}
/**
* Add a header to the list of stored headers for network operations.
*
- * @param header {@link Header} to add to the list.
+ * @param header {@link HTTPHeader} to add to the list.
* @throws FHIRException if the header being added is a duplicate.
*/
- public ClientHeaders addHeader(Header header) throws FHIRException {
+ public ClientHeaders addHeader(HTTPHeader header) throws FHIRException {
if (headers.contains(header)) {
- throw new FHIRException("Attempting to add duplicate header, <" + header.name + ", " + header.value + ">.");
+ throw new FHIRException("Attempting to add duplicate header, <" + header.getName() + ", " + header.getValue() + ">.");
}
headers.add(header);
return this;
@@ -47,10 +47,10 @@ public class ClientHeaders {
/**
* Add a header to the list of stored headers for network operations.
*
- * @param headerList {@link List} of {@link Header} to add.
+ * @param headerList {@link List} of {@link HTTPHeader} to add.
* @throws FHIRException if any of the headers being added is a duplicate.
*/
- public ClientHeaders addHeaders(List headerList) throws FHIRException {
+ public ClientHeaders addHeaders(List headerList) throws FHIRException {
headerList.forEach(this::addHeader);
return this;
}
@@ -58,13 +58,13 @@ public class ClientHeaders {
/**
* Removes the passed in header from the list of stored headers.
*
- * @param header {@link Header} to remove from the list.
+ * @param header {@link HTTPHeader} to remove from the list.
* @throws FHIRException if the header passed in does not exist within the
* stored list.
*/
- public ClientHeaders removeHeader(Header header) throws FHIRException {
+ public ClientHeaders removeHeader(HTTPHeader header) throws FHIRException {
if (!headers.remove(header)) {
- throw new FHIRException("Attempting to remove header, <" + header.name + ", " + header.value
+ throw new FHIRException("Attempting to remove header, <" + header.getName() + ", " + header.getValue()
+ ">, from GenericClientHeaders that is not currently stored.");
}
return this;
@@ -73,17 +73,17 @@ public class ClientHeaders {
/**
* Removes the passed in headers from the list of stored headers.
*
- * @param headerList {@link List} of {@link Header} to remove.
+ * @param headerList {@link List} of {@link HTTPHeader} to remove.
* @throws FHIRException if any of the headers passed in does not exist within
* the stored list.
*/
- public ClientHeaders removeHeaders(List headerList) throws FHIRException {
+ public ClientHeaders removeHeaders(List headerList) throws FHIRException {
headerList.forEach(this::removeHeader);
return this;
}
/**
- * Clears all stored {@link Header}.
+ * Clears all stored {@link HTTPHeader}.
*/
public ClientHeaders clearHeaders() {
headers.clear();
@@ -92,7 +92,7 @@ public class ClientHeaders {
@Override
public String toString() {
- return this.headers.stream().map(header -> "\t" + header.name + ":" + header.value)
+ return this.headers.stream().map(header -> "\t" + header.getName() + ":" + header.getValue())
.collect(Collectors.joining(",\n", "{\n", "\n}"));
}
}
diff --git a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/FhirLoggingInterceptor.java b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/FhirLoggingInterceptor.java
deleted file mode 100644
index 38d195380..000000000
--- a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/FhirLoggingInterceptor.java
+++ /dev/null
@@ -1,74 +0,0 @@
-package org.hl7.fhir.r4.utils.client.network;
-
-import java.io.IOException;
-import java.util.ArrayList;
-import java.util.List;
-import java.util.Map;
-
-import javax.annotation.Nonnull;
-
-import org.hl7.fhir.utilities.ToolingClientLogger;
-
-import okhttp3.Interceptor;
-import okhttp3.MediaType;
-import okhttp3.Request;
-import okhttp3.Response;
-import okhttp3.ResponseBody;
-import okio.Buffer;
-
-public class FhirLoggingInterceptor implements Interceptor {
-
- private ToolingClientLogger logger;
-
- public FhirLoggingInterceptor(ToolingClientLogger logger) {
- this.logger = logger;
- }
-
- public FhirLoggingInterceptor setLogger(ToolingClientLogger logger) {
- this.logger = logger;
- return this;
- }
-
- @Override
- public Response intercept(@Nonnull Interceptor.Chain chain) throws IOException {
- // Log Request
- Request request = chain.request();
- List hdrs = new ArrayList<>();
- for (String s : request.headers().toString().split("\\n")) {
- hdrs.add(s.trim());
- }
- byte[] cnt = null;
- if (request.body() != null) {
- Buffer buf = new Buffer();
- request.body().writeTo(buf);
- cnt = buf.readByteArray();
- }
- if (logger != null) {
- logger.logRequest(request.method(), request.url().toString(), hdrs, cnt);
- }
-
- // Log Response
- Response response = null;
- response = chain.proceed(chain.request());
-
- MediaType contentType = null;
- byte[] bodyBytes = null;
- if (response.body() != null) {
- contentType = response.body().contentType();
- bodyBytes = response.body().bytes();
- }
-
- // Get Headers as List
- List headerList = new ArrayList<>();
- Map> headerMap = response.headers().toMultimap();
- headerMap.keySet().forEach(key -> headerMap.get(key).forEach(value -> headerList.add(key + ":" + value)));
-
- if (logger != null) {
- logger.logResponse(Integer.toString(response.code()), headerList, bodyBytes, 0);
- }
-
- // Reading byte[] clears body. Need to recreate.
- ResponseBody body = ResponseBody.create(bodyBytes, contentType);
- return response.newBuilder().body(body).build();
- }
-}
diff --git a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/FhirRequestBuilder.java b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/FhirRequestBuilder.java
index a9b0d78ad..579e3d1b3 100644
--- a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/FhirRequestBuilder.java
+++ b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/FhirRequestBuilder.java
@@ -1,16 +1,13 @@
package org.hl7.fhir.r4.utils.client.network;
-import static org.hl7.fhir.r4.utils.OperationOutcomeUtilities.outcomeFromTextError;
-
import java.io.IOException;
+import java.util.ArrayList;
import java.util.List;
-import java.util.Map;
import java.util.concurrent.TimeUnit;
-import javax.annotation.Nonnull;
-
+import lombok.Getter;
+import lombok.Setter;
import org.apache.commons.lang3.StringUtils;
-import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.r4.formats.IParser;
import org.hl7.fhir.r4.formats.JsonParser;
import org.hl7.fhir.r4.formats.XmlParser;
@@ -22,31 +19,19 @@ import org.hl7.fhir.r4.utils.ResourceUtilities;
import org.hl7.fhir.r4.utils.client.EFhirClientException;
import org.hl7.fhir.r4.utils.client.ResourceFormat;
import org.hl7.fhir.utilities.MimeType;
-import org.hl7.fhir.utilities.settings.FhirSettings;
+import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.hl7.fhir.utilities.http.*;
import org.hl7.fhir.utilities.xhtml.XhtmlUtils;
-import okhttp3.Authenticator;
-import okhttp3.Credentials;
-import okhttp3.Headers;
-import okhttp3.OkHttpClient;
-import okhttp3.Request;
-import okhttp3.Response;
-
public class FhirRequestBuilder {
- protected static final String HTTP_PROXY_USER = "http.proxyUser";
- protected static final String HTTP_PROXY_PASS = "http.proxyPassword";
- protected static final String HEADER_PROXY_AUTH = "Proxy-Authorization";
protected static final String LOCATION_HEADER = "location";
protected static final String CONTENT_LOCATION_HEADER = "content-location";
protected static final String DEFAULT_CHARSET = "UTF-8";
- /**
- * The singleton instance of the HttpClient, used for all requests.
- */
- private static OkHttpClient okHttpClient;
- private final Request.Builder httpRequest;
+
+ private final HTTPRequest httpRequest;
private String resourceFormat = null;
- private Headers headers = null;
+ private Iterable headers = null;
private String message = null;
private int retryCount = 1;
/**
@@ -60,63 +45,46 @@ public class FhirRequestBuilder {
private TimeUnit timeoutUnit = TimeUnit.MILLISECONDS;
/**
- * {@link FhirLoggingInterceptor} for log output.
+ * {@link ToolingClientLogger} for log output.
*/
- private FhirLoggingInterceptor logger = null;
+ @Getter
+ @Setter
+ private ToolingClientLogger logger = null;
+
private String source;
- public FhirRequestBuilder(Request.Builder httpRequest, String source) {
- this.httpRequest = httpRequest;
+ public FhirRequestBuilder(HTTPRequest httpRequest, String source) {
this.source = source;
+ this.httpRequest = httpRequest;
}
/**
* Adds necessary default headers, formatting headers, and any passed in
- * {@link Headers} to the passed in {@link okhttp3.Request.Builder}
+ * {@link HTTPHeader}s to the passed in {@link HTTPRequest}
*
- * @param request {@link okhttp3.Request.Builder} to add headers to.
+ * @param request {@link HTTPRequest} to add headers to.
* @param format Expected {@link Resource} format.
- * @param headers Any additional {@link Headers} to add to the request.
+ * @param headers Any additional {@link HTTPHeader}s to add to the request.
*/
- protected static void formatHeaders(Request.Builder request, String format, Headers headers) {
- addDefaultHeaders(request, headers);
- if (format != null)
- addResourceFormatHeaders(request, format);
- if (headers != null)
- addHeaders(request, headers);
+ protected static HTTPRequest formatHeaders(HTTPRequest request, String format, Iterable headers) {
+ List allHeaders = new ArrayList<>();
+ request.getHeaders().forEach(allHeaders::add);
+
+ if (format != null) getResourceFormatHeaders(request, format).forEach(allHeaders::add);
+ if (headers != null) headers.forEach(allHeaders::add);
+ return request.withHeaders(allHeaders);
}
- /**
- * Adds necessary headers for all REST requests.
- * User-Agent : hapi-fhir-tooling-client
- *
- * @param request {@link Request.Builder} to add default headers to.
- */
- protected static void addDefaultHeaders(Request.Builder request, Headers headers) {
- if (headers == null || !headers.names().contains("User-Agent")) {
- request.addHeader("User-Agent", "hapi-fhir-tooling-client");
+ protected static Iterable getResourceFormatHeaders(HTTPRequest httpRequest, String format) {
+ List headers = new ArrayList<>();
+ headers.add(new HTTPHeader("Accept", format));
+ if (httpRequest.getMethod() == HTTPRequest.HttpMethod.PUT
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.POST
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.PATCH
+ ) {
+ headers.add(new HTTPHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET));
}
- }
-
- /**
- * Adds necessary headers for the given resource format provided.
- *
- * @param request {@link Request.Builder} to add default headers to.
- */
- protected static void addResourceFormatHeaders(Request.Builder request, String format) {
- request.addHeader("Accept", format);
- request.addHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET);
- }
-
- /**
- * Iterates through the passed in {@link Headers} and adds them to the provided
- * {@link Request.Builder}.
- *
- * @param request {@link Request.Builder} to add headers to.
- * @param headers {@link Headers} to add to request.
- */
- protected static void addHeaders(Request.Builder request, Headers headers) {
- headers.forEach(header -> request.addHeader(header.getFirst(), header.getSecond()));
+ return headers;
}
/**
@@ -136,73 +104,24 @@ public class FhirRequestBuilder {
}
/**
- * Extracts the 'location' header from the passes in {@link Headers}. If no
+ * Extracts the 'location' header from the passed {@link Iterable}. If no
* value for 'location' exists, the value for 'content-location' is returned. If
* neither header exists, we return null.
*
- * @param headers {@link Headers} to evaluate
+ * @param headers {@link HTTPHeader} to evaluate
* @return {@link String} header value, or null if no location headers are set.
*/
- protected static String getLocationHeader(Headers headers) {
- Map> headerMap = headers.toMultimap();
- if (headerMap.containsKey(LOCATION_HEADER)) {
- return headerMap.get(LOCATION_HEADER).get(0);
- } else if (headerMap.containsKey(CONTENT_LOCATION_HEADER)) {
- return headerMap.get(CONTENT_LOCATION_HEADER).get(0);
- } else {
- return null;
+ protected static String getLocationHeader(Iterable headers) {
+ String locationHeader = HTTPHeaderUtil.getSingleHeader(headers, LOCATION_HEADER);
+
+ if (locationHeader != null) {
+ return locationHeader;
}
+ return HTTPHeaderUtil.getSingleHeader(headers, CONTENT_LOCATION_HEADER);
}
- /**
- * We only ever want to have one copy of the HttpClient kicking around at any
- * given time. If we need to make changes to any configuration, such as proxy
- * settings, timeout, caches, etc, we can do a per-call configuration through
- * the {@link OkHttpClient#newBuilder()} method. That will return a builder that
- * shares the same connection pool, dispatcher, and configuration with the
- * original client.
- *
- * The {@link OkHttpClient} uses the proxy auth properties set in the current
- * system properties. The reason we don't set the proxy address and
- * authentication explicitly, is due to the fact that this class is often used
- * in conjunction with other http client tools which rely on the
- * system.properties settings to determine proxy settings. It's easier to keep
- * the method consistent across the board. ...for now.
- *
- * @return {@link OkHttpClient} instance
- */
- protected OkHttpClient getHttpClient() {
- if (FhirSettings.isProhibitNetworkAccess()) {
- throw new FHIRException("Network Access is prohibited in this context");
- }
-
- if (okHttpClient == null) {
- okHttpClient = new OkHttpClient();
- }
-
- Authenticator proxyAuthenticator = getAuthenticator();
-
- OkHttpClient.Builder builder = okHttpClient.newBuilder();
- if (logger != null)
- builder.addInterceptor(logger);
- builder.addInterceptor(new RetryInterceptor(retryCount));
-
- return builder.connectTimeout(timeout, timeoutUnit).addInterceptor(new RetryInterceptor(retryCount))
- .connectTimeout(timeout, timeoutUnit).writeTimeout(timeout, timeoutUnit).readTimeout(timeout, timeoutUnit)
- .proxyAuthenticator(proxyAuthenticator).build();
- }
-
- @Nonnull
- private static Authenticator getAuthenticator() {
- return (route, response) -> {
- final String httpProxyUser = System.getProperty(HTTP_PROXY_USER);
- final String httpProxyPass = System.getProperty(HTTP_PROXY_PASS);
- if (httpProxyUser != null && httpProxyPass != null) {
- String credential = Credentials.basic(httpProxyUser, httpProxyPass);
- return response.request().newBuilder().header(HEADER_PROXY_AUTH, credential).build();
- }
- return response.request().newBuilder().build();
- };
+ protected ManagedFhirWebAccessor getManagedWebAccessor() {
+ return ManagedWebAccess.fhirAccessor().withRetries(retryCount).withTimeout(timeout, timeoutUnit).withLogger(logger);
}
public FhirRequestBuilder withResourceFormat(String resourceFormat) {
@@ -210,7 +129,7 @@ public class FhirRequestBuilder {
return this;
}
- public FhirRequestBuilder withHeaders(Headers headers) {
+ public FhirRequestBuilder withHeaders(Iterable headers) {
this.headers = headers;
return this;
}
@@ -225,7 +144,7 @@ public class FhirRequestBuilder {
return this;
}
- public FhirRequestBuilder withLogger(FhirLoggingInterceptor logger) {
+ public FhirRequestBuilder withLogger(ToolingClientLogger logger) {
this.logger = logger;
return this;
}
@@ -236,33 +155,31 @@ public class FhirRequestBuilder {
return this;
}
- protected Request buildRequest() {
- return httpRequest.build();
- }
public ResourceRequest execute() throws IOException {
- formatHeaders(httpRequest, resourceFormat, headers);
- Response response = getHttpClient().newCall(httpRequest.build()).execute();
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, headers);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);
+
T resource = unmarshalReference(response, resourceFormat, null);
- return new ResourceRequest(resource, response.code(), getLocationHeader(response.headers()));
+ return new ResourceRequest(resource, response.getCode(), getLocationHeader(response.getHeaders()));
}
public Bundle executeAsBatch() throws IOException {
- formatHeaders(httpRequest, resourceFormat, null);
- Response response = getHttpClient().newCall(httpRequest.build()).execute();
- return unmarshalFeed(response, resourceFormat);
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, null);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);
+ return unmarshalFeed(response, resourceFormat);
}
/**
* Unmarshalls a resource from the response stream.
*/
@SuppressWarnings("unchecked")
- protected T unmarshalReference(Response response, String format, String resourceType) {
- int code = response.code();
+ protected T unmarshalReference(HTTPResult response, String format, String resourceType) {
+ int code = response.getCode();
boolean ok = code >= 200 && code < 300;
- if (response.body() == null) {
+ if (response.getContent() == null) {
if (!ok) {
- throw new EFhirClientException(response.message());
+ throw new EFhirClientException(response.getMessage());
} else {
return null;
}
@@ -271,9 +188,9 @@ public class FhirRequestBuilder {
Resource resource = null;
try {
- body = response.body().string();
- String ct = response.header("Content-Type");
- if (ct == null) {
+ body = response.getContentAsString();
+ String contentType = HTTPHeaderUtil.getSingleHeader(response.getHeaders(), "Content-Type");
+ if (contentType == null) {
if (ok) {
resource = getParser(format).parse(body);
} else {
@@ -282,10 +199,10 @@ public class FhirRequestBuilder {
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
}
} else {
- if (ct.contains(";")) {
- ct = ct.substring(0, ct.indexOf(";"));
+ if (contentType.contains(";")) {
+ contentType = contentType.substring(0, contentType.indexOf(";"));
}
- switch (ct) {
+ switch (contentType) {
case "application/json":
case "application/fhir+json":
if (!format.contains("json")) {
@@ -299,16 +216,16 @@ public class FhirRequestBuilder {
if (!format.contains("xml")) {
System.out.println("Got xml response expecting "+format+" from "+source+" with status "+code);
}
- resource = getParser(ResourceFormat.RESOURCE_XML.getHeader()).parse(response.body().bytes());
+ resource = getParser(ResourceFormat.RESOURCE_XML.getHeader()).parse(response.getContent());
break;
case "text/plain":
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
break;
case "text/html" :
- resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.body().string(), source));
+ resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.getContentAsString(), source));
break;
default: // not sure what else to do?
- System.out.println("Got content-type '"+ct+"' from "+source);
+ System.out.println("Got content-type '"+contentType+"' from "+source);
System.out.println(body);
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
}
@@ -343,14 +260,14 @@ public class FhirRequestBuilder {
/**
* Unmarshalls Bundle from response stream.
*/
- protected Bundle unmarshalFeed(Response response, String format) {
+ protected Bundle unmarshalFeed(HTTPResult response, String format) {
return unmarshalReference(response, format, "Bundle");
}
/**
* Returns the appropriate parser based on the format type passed in. Defaults
* to XML parser if a blank format is provided...because reasons.
- *
+ *
* Currently supports only "json" and "xml" formats.
*
* @param format One of "json" or "xml".
diff --git a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/RetryInterceptor.java b/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/RetryInterceptor.java
deleted file mode 100644
index b90085b02..000000000
--- a/org.hl7.fhir.r4/src/main/java/org/hl7/fhir/r4/utils/client/network/RetryInterceptor.java
+++ /dev/null
@@ -1,67 +0,0 @@
-package org.hl7.fhir.r4.utils.client.network;
-
-import java.io.IOException;
-
-import okhttp3.Interceptor;
-import okhttp3.Request;
-import okhttp3.Response;
-
-/**
- * An {@link Interceptor} for {@link okhttp3.OkHttpClient} that controls the
- * number of times we retry a to execute a given request, before reporting a
- * failure. This includes unsuccessful return codes and timeouts.
- */
-public class RetryInterceptor implements Interceptor {
-
- // Delay between retying failed requests, in millis
- private final long RETRY_TIME = 2000;
-
- // Maximum number of times to retry the request before failing
- private final int maxRetry;
-
- // Internal counter for tracking the number of times we've tried this request
- private int retryCounter = 0;
-
- public RetryInterceptor(int maxRetry) {
- this.maxRetry = maxRetry;
- }
-
- @Override
- public Response intercept(Interceptor.Chain chain) throws IOException {
- Request request = chain.request();
- Response response = null;
-
- do {
- try {
- // If we are retrying a failed request that failed due to a bad response from
- // the server, we must close it first
- if (response != null) {
-// System.out.println("Previous " + chain.request().method() + " attempt returned HTTP<" + (response.code())
-// + "> from url -> " + chain.request().url() + ".");
- response.close();
- }
- // System.out.println(chain.request().method() + " attempt <" + (retryCounter +
- // 1) + "> to url -> " + chain.request().url());
- response = chain.proceed(request);
- } catch (IOException e) {
- try {
- // Include a small break in between requests.
- Thread.sleep(RETRY_TIME);
- } catch (InterruptedException e1) {
- System.out.println(chain.request().method() + " to url -> " + chain.request().url() + " interrupted on try <"
- + retryCounter + ">");
- }
- } finally {
- retryCounter++;
- }
- } while ((response == null || !response.isSuccessful()) && (retryCounter <= maxRetry + 1));
-
- /*
- * if something has gone wrong, and we are unable to complete the request, we
- * still need to initialize the return response so we don't get a null pointer
- * exception.
- */
- return response != null ? response : chain.proceed(request);
- }
-
-}
\ No newline at end of file
diff --git a/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/FhirToolingClientTest.java b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/FhirToolingClientTest.java
new file mode 100644
index 000000000..43de724c1
--- /dev/null
+++ b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/FhirToolingClientTest.java
@@ -0,0 +1,224 @@
+package org.hl7.fhir.r4.utils.client;
+
+import org.hl7.fhir.r4.model.*;
+import org.hl7.fhir.r4.utils.client.network.Client;
+import org.hl7.fhir.r4.utils.client.network.ResourceRequest;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+import org.mockito.*;
+
+import java.io.IOException;
+import java.net.URI;
+import java.net.URISyntaxException;
+import java.util.ArrayList;
+import java.util.Arrays;
+import java.util.List;
+
+import static org.junit.jupiter.api.Assertions.assertTrue;
+
+public class FhirToolingClientTest {
+ String TX_ADDR = "http://tx.fhir.org";
+
+ HTTPHeader h1 = new HTTPHeader("header1", "value1");
+ HTTPHeader h2 = new HTTPHeader("header2", "value2");
+ HTTPHeader h3 = new HTTPHeader("header3", "value3");
+
+ HTTPHeader agentHeader = new HTTPHeader("User-Agent", "fhir/test-cases");
+
+ private Client mockClient;
+ private FHIRToolingClient toolingClient;
+
+ @Captor
+ private ArgumentCaptor> headersArgumentCaptor;
+
+ @BeforeEach
+ void setUp() throws IOException, URISyntaxException {
+ MockitoAnnotations.openMocks(this);
+ mockClient = Mockito.mock(Client.class);
+ ResourceRequest resourceResourceRequest = new ResourceRequest<>(generateBundle(), 200, "");
+
+ // GET
+ Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ Mockito
+ .when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new TerminologyCapabilities(), 200, "location"));
+ Mockito
+ .when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
+ Mockito
+ .when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
+
+ // PUT
+ Mockito.when(mockClient.issuePutRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ // POST
+ Mockito.when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ Mockito
+ .when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.contains("validate"), Mockito.anyLong()))
+ .thenReturn(new ResourceRequest<>(new OperationOutcome(), 200, "location"));
+ // BUNDLE REQ
+ Mockito
+ .when(mockClient.executeBundleRequest(Mockito.any(HTTPRequest.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyLong()))
+ .thenReturn(generateBundle());
+ toolingClient = new FHIRToolingClient(TX_ADDR, "fhir/test-cases");
+ toolingClient.setClient(mockClient);
+ }
+
+ private List getHeaders() {
+ return new ArrayList<>(Arrays.asList(h1, h2, h3));
+ }
+
+ private List getHeadersWithAgent() {
+ return new ArrayList<>(Arrays.asList(h1, h2, h3, agentHeader));
+ }
+
+ private Bundle generateBundle() {
+ Patient patient = generatePatient();
+ Observation observation = generateObservation();
+
+ // The observation refers to the patient using the ID, which is already
+ // set to a temporary UUID
+ observation.setSubject(new Reference(patient.getIdElement().getValue()));
+
+ // Create a bundle that will be used as a transaction
+ Bundle bundle = new Bundle();
+
+ // Add the patient as an entry.
+ bundle.addEntry().setFullUrl(patient.getIdElement().getValue()).setResource(patient).getRequest().setUrl("Patient")
+ .setIfNoneExist("identifier=http://acme.org/mrns|12345").setMethod(Bundle.HTTPVerb.POST);
+
+ return bundle;
+ }
+
+ private Patient generatePatient() {
+ // Create a patient object
+ Patient patient = new Patient();
+ patient.addIdentifier().setSystem("http://acme.org/mrns").setValue("12345");
+ patient.addName().setFamily("Jameson").addGiven("J").addGiven("Jonah");
+ patient.setGender(Enumerations.AdministrativeGender.MALE);
+
+ // Give the patient a temporary UUID so that other resources in
+ // the transaction can refer to it
+ patient.setId(IdType.newRandomUuid());
+ return patient;
+ }
+
+ private Observation generateObservation() {
+ // Create an observation object
+ Observation observation = new Observation();
+ observation.getCode().addCoding().setSystem("http://loinc.org").setCode("789-8")
+ .setDisplay("Erythrocytes [#/volume] in Blood by Automated count");
+ observation.setValue(new Quantity().setValue(4.12).setUnit("10 trillion/L").setSystem("http://unitsofmeasure.org")
+ .setCode("10*12/L"));
+ return observation;
+ }
+
+ private void checkHeaders(Iterable argumentCaptorValue) {
+ List capturedHeaders = new ArrayList<>();
+ argumentCaptorValue.forEach(capturedHeaders::add);
+
+ getHeadersWithAgent().forEach(header -> {
+ assertTrue(capturedHeaders.contains(header));
+ });
+ }
+
+ @Test
+ void getTerminologyCapabilities() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getTerminologyCapabilities();
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void getCapabilitiesStatement() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getCapabilitiesStatement();
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void getCapabilitiesStatementQuick() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getCapabilitiesStatementQuick();
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void read() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.read(Patient.class, "id");
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void vread() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.vread(Patient.class, "id", "version");
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void getCanonical() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.getCanonical(Patient.class, "canonicalURL");
+ Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
+ headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void update() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.update(generatePatient());
+ Mockito.verify(mockClient).issuePutRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
+ ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
+ ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+
+ @Test
+ void validate() throws IOException {
+ toolingClient.setClientHeaders(getHeaders());
+ toolingClient.validate(Patient.class, generatePatient(), "id");
+ Mockito.verify(mockClient).issuePostRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
+ ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
+ ArgumentMatchers.anyLong());
+
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
+ checkHeaders(argumentCaptorValue);
+ }
+}
diff --git a/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/ClientHeadersTest.java b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/ClientHeadersTest.java
new file mode 100644
index 000000000..93ed42f40
--- /dev/null
+++ b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/ClientHeadersTest.java
@@ -0,0 +1,104 @@
+package org.hl7.fhir.r4.utils.client.network;
+
+import org.hl7.fhir.exceptions.FHIRException;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Test;
+
+import java.util.Arrays;
+import java.util.List;
+
+public class ClientHeadersTest {
+ ClientHeaders clientHeaders;
+
+ HTTPHeader h1 = new HTTPHeader("header1", "value1");
+ HTTPHeader h2 = new HTTPHeader("header2", "value2");
+ HTTPHeader h3 = new HTTPHeader("header3", "value3");
+
+ @BeforeEach
+ void setUp() {
+ clientHeaders = new ClientHeaders();
+ }
+
+ @Test
+ @DisplayName("Happy path add headers individually.")
+ void addHeader() {
+ clientHeaders.addHeader(h1);
+ Assertions.assertEquals(1, clientHeaders.headers().size());
+ clientHeaders.addHeader(h2);
+ Assertions.assertEquals(2, clientHeaders.headers().size());
+ }
+
+ @Test
+ @DisplayName("Test duplicate header added individually throws FHIRException.")
+ void addHeaderDuplicateAdd() {
+ clientHeaders.addHeader(h1);
+ Assertions.assertThrows(FHIRException.class, () -> clientHeaders.addHeader(h1));
+ }
+
+ @Test
+ @DisplayName("Happy path add headers as list.")
+ void addHeaders() {
+ List headersList = Arrays.asList(h1, h2, h3);
+ clientHeaders.addHeaders(headersList);
+ Assertions.assertEquals(3, clientHeaders.headers().size());
+ Assertions.assertEquals(headersList, clientHeaders.headers());
+ }
+
+ @Test
+ @DisplayName("Happy path add headers as list.")
+ void addHeadersDuplicateAdd() {
+ List headersList = Arrays.asList(h1, h2, h1);
+ Assertions.assertThrows(FHIRException.class, () -> clientHeaders.addHeaders(headersList));
+ }
+
+ @Test
+ @DisplayName("Happy path remove existing header.")
+ void removeHeader() {
+ clientHeaders.addHeader(h1);
+ clientHeaders.addHeader(h2);
+ clientHeaders.addHeader(h3);
+ clientHeaders.removeHeader(h2);
+ Assertions.assertEquals(2, clientHeaders.headers().size());
+ clientHeaders.removeHeader(new HTTPHeader("header3", "value3"));
+ Assertions.assertEquals(1, clientHeaders.headers().size());
+ }
+
+ @Test
+ @DisplayName("Remove header not contained in list.")
+ void removeHeaderUnknown() {
+ clientHeaders.addHeader(h1);
+ clientHeaders.addHeader(h2);
+ Assertions.assertThrows(FHIRException.class, () -> clientHeaders.removeHeader(h3));
+ }
+
+ @Test
+ @DisplayName("Happy path remove list of existing headers.")
+ void removeHeaders() {
+ List headersToAddList = Arrays.asList(h1, h2, h3);
+ List headersToRemoveList = Arrays.asList(h2, h3);
+ clientHeaders.addHeaders(headersToAddList);
+ clientHeaders.removeHeaders(headersToRemoveList);
+ Assertions.assertEquals(1, clientHeaders.headers().size());
+ }
+
+ @Test
+ @DisplayName("Remove list containing unknown header.")
+ void removeHeadersUnknown() {
+ List headersToAddList = Arrays.asList(h1, h3);
+ List headersToRemoveList = Arrays.asList(h2, h3);
+ clientHeaders.addHeaders(headersToAddList);
+ Assertions.assertThrows(FHIRException.class, () -> clientHeaders.removeHeaders(headersToRemoveList));
+ }
+
+ @Test
+ void clearHeaders() {
+ List headersToAddList = Arrays.asList(h1, h3);
+ clientHeaders.addHeaders(headersToAddList);
+ Assertions.assertEquals(2, clientHeaders.headers().size());
+ clientHeaders.clearHeaders();
+ Assertions.assertEquals(0, clientHeaders.headers().size());
+ }
+}
diff --git a/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/ClientTest.java b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/ClientTest.java
new file mode 100644
index 000000000..19cc74d1d
--- /dev/null
+++ b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/ClientTest.java
@@ -0,0 +1,140 @@
+package org.hl7.fhir.r4.utils.client.network;
+
+import okhttp3.HttpUrl;
+import okhttp3.mockwebserver.MockWebServer;
+import okhttp3.mockwebserver.MockResponse;
+import okhttp3.mockwebserver.RecordedRequest;
+import org.hl7.fhir.r4.context.HTMLClientLogger;
+import org.hl7.fhir.r4.formats.JsonParser;
+import org.hl7.fhir.r4.model.*;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Test;
+import org.mockito.Mockito;
+
+import java.io.IOException;
+import java.net.URI;
+import java.net.URISyntaxException;
+import java.util.Random;
+import java.util.concurrent.TimeUnit;
+
+public class ClientTest {
+ private static final long TIMEOUT = 5000;
+
+ private MockWebServer server;
+ private HttpUrl serverUrl;
+ private Client client;
+
+ private final Address address = new Address().setCity("Toronto").setState("Ontario").setCountry("Canada");
+ private final HumanName humanName = new HumanName().addGiven("Mark").setFamily("Iantorno");
+ private final Patient patient = new Patient().addName(humanName).addAddress(address)
+ .setGender(Enumerations.AdministrativeGender.MALE);
+
+ @BeforeEach
+ void setup() {
+ setupMockServer();
+ client = new Client();
+ }
+
+ void setupMockServer() {
+ server = new MockWebServer();
+ serverUrl = server.url("/v1/endpoint");
+ }
+
+ byte[] generateResourceBytes(Resource resource) throws IOException {
+ return new JsonParser().composeBytes(resource);
+ }
+
+ @Test
+ @DisplayName("GET request, happy path.")
+ void test_get_happy_path() throws IOException, URISyntaxException {
+ server.enqueue(new MockResponse().setBody(new String(generateResourceBytes(patient))));
+ ResourceRequest resourceRequest = client.issueGetResourceRequest(new URI(serverUrl.toString()), "json",
+ null, null, TIMEOUT);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+ @Test
+ @DisplayName("GET request, test client retries after timeout failure.")
+ void test_get_retries_with_timeout() throws IOException, URISyntaxException {
+ int failedAttempts = new Random().nextInt(5) + 1;
+ System.out.println("Simulating <" + failedAttempts + "> failed connections (timeouts) before success.");
+ for (int i = 0; i < failedAttempts; i++) {
+ server.enqueue(new MockResponse().setHeadersDelay(TIMEOUT * 10, TimeUnit.MILLISECONDS)
+ .setBody(new String(generateResourceBytes(patient))));
+ }
+ server.enqueue(new MockResponse().setBody(new String(generateResourceBytes(patient))));
+ client.setRetryCount(failedAttempts + 1);
+
+ ResourceRequest resourceRequest = client.issueGetResourceRequest(new URI(serverUrl.toString()), "json",
+ null, null, TIMEOUT);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+ @Test
+ @DisplayName("GET request, test client retries after bad response.")
+ void test_get_retries_with_unsuccessful_response() throws IOException, URISyntaxException {
+ int failedAttempts = new Random().nextInt(5) + 1;
+ System.out.println("Simulating <" + failedAttempts + "> failed connections (bad response codes) before success.");
+ for (int i = 0; i < failedAttempts; i++) {
+ server.enqueue(new MockResponse().setResponseCode(400 + i).setBody(new String(generateResourceBytes(patient))));
+ }
+ server.enqueue(new MockResponse().setBody(new String(generateResourceBytes(patient))));
+ client.setRetryCount(failedAttempts + 1);
+
+ ResourceRequest resourceRequest = client.issueGetResourceRequest(new URI(serverUrl.toString()), "json",
+ null, null, TIMEOUT);
+ Assertions.assertTrue(resourceRequest.isSuccessfulRequest());
+ Assertions.assertTrue(patient.equalsDeep(resourceRequest.getPayload()),
+ "GET request returned resource does not match expected.");
+ }
+
+ @Test
+ @DisplayName("PUT request, test payload received by server matches sent.")
+ void test_put() throws IOException, URISyntaxException, InterruptedException {
+ byte[] payload = ByteUtils.resourceToByteArray(patient, true, false, false);
+ // Mock server response of 200, with the same resource payload returned that we
+ // included in the PUT request
+ server.enqueue(new MockResponse().setResponseCode(200).setBody(new String(payload)));
+
+ ResourceRequest request = client.issuePutRequest(new URI(serverUrl.toString()), payload, "xml", null,
+ TIMEOUT);
+ RecordedRequest recordedRequest = server.takeRequest();
+ Assertions.assertArrayEquals(payload, recordedRequest.getBody().readByteArray(),
+ "PUT request payload does not match send data.");
+ }
+
+ @Test
+ @DisplayName("POST request, test payload received by server matches sent.")
+ void test_post() throws IOException, URISyntaxException, InterruptedException {
+ byte[] payload = ByteUtils.resourceToByteArray(patient, true, false, false);
+ // Mock server response of 200, with the same resource payload returned that we
+ // included in the PUT request
+ server.enqueue(new MockResponse().setResponseCode(200).setBody(new String(payload)));
+
+ ResourceRequest request = client.issuePostRequest(new URI(serverUrl.toString()), payload, "xml", null,
+ TIMEOUT);
+ RecordedRequest recordedRequest = server.takeRequest();
+ Assertions.assertArrayEquals(payload, recordedRequest.getBody().readByteArray(),
+ "POST request payload does not match send data.");
+ }
+
+ @Test
+ @DisplayName("Testing the logger works.")
+ void test_logger() throws IOException, URISyntaxException, InterruptedException {
+ byte[] payload = ByteUtils.resourceToByteArray(patient, true, false, false);
+ server.enqueue(new MockResponse().setResponseCode(200).setBody(new String(payload)));
+ HTMLClientLogger mockLogger = Mockito.mock(HTMLClientLogger.class);
+ client.setLogger(mockLogger);
+ client.issuePostRequest(new URI(serverUrl.toString()), payload, "xml", null, TIMEOUT);
+ server.takeRequest();
+ Mockito.verify(mockLogger, Mockito.times(1)).logRequest(Mockito.anyString(), Mockito.anyString(), Mockito.anyList(),
+ Mockito.any());
+ Mockito.verify(mockLogger, Mockito.times(1)).logResponse(Mockito.anyString(), Mockito.anyList(), Mockito.any(), Mockito.anyLong());
+ }
+}
diff --git a/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/FhirRequestBuilderTest.java b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/FhirRequestBuilderTest.java
new file mode 100644
index 000000000..5d278d574
--- /dev/null
+++ b/org.hl7.fhir.r4/src/test/java/org/hl7/fhir/r4/utils/client/network/FhirRequestBuilderTest.java
@@ -0,0 +1,128 @@
+package org.hl7.fhir.r4.utils.client.network;
+
+import okhttp3.Request;
+import org.hl7.fhir.r4.model.OperationOutcome;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPHeaderUtil;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Test;
+
+import java.util.Collections;
+import java.util.List;
+import java.util.Map;
+
+public class FhirRequestBuilderTest {
+ @Test
+ @DisplayName("Test resource format headers are added correctly.")
+ void addResourceFormatHeadersGET() {
+ String testFormat = "yaml";
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.GET);
+
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
+ Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
+ Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
+ "Accept header not populated with expected value " + testFormat + ".");
+
+ Assertions.assertNull(headersMap.get("Content-Type"), "Content-Type header null.");
+ }
+
+ @Test
+ @DisplayName("Test resource format headers are added correctly (POST).")
+ void addResourceFormatHeadersPOST() {
+ String testFormat = "yaml";
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.POST);
+
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
+ Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
+ Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
+ "Accept header not populated with expected value " + testFormat + ".");
+
+ Assertions.assertNotNull(headersMap.get("Content-Type"), "Content-Type header null.");
+ Assertions.assertEquals(testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET, headersMap.get("Content-Type").get(0),
+ "Content-Type header not populated with expected value \"" + testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET + "\".");
+ }
+
+ @Test
+ @DisplayName("Test that FATAL issue severity triggers error.")
+ void hasErrorTestFatal() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.FATAL));
+ Assertions.assertTrue(FhirRequestBuilder.hasError(outcome), "Error check not triggered for FATAL issue severity.");
+ }
+
+ @Test
+ @DisplayName("Test that ERROR issue severity triggers error.")
+ void hasErrorTestError() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.ERROR));
+ Assertions.assertTrue(FhirRequestBuilder.hasError(outcome), "Error check not triggered for ERROR issue severity.");
+ }
+
+ @Test
+ @DisplayName("Test that no FATAL or ERROR issue severity does not trigger error.")
+ void hasErrorTestNoErrors() {
+ OperationOutcome outcome = new OperationOutcome();
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.INFORMATION));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.NULL));
+ outcome.addIssue(
+ new OperationOutcome.OperationOutcomeIssueComponent().setSeverity(OperationOutcome.IssueSeverity.WARNING));
+ Assertions.assertFalse(FhirRequestBuilder.hasError(outcome), "Error check triggered unexpectedly.");
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns header for 'location'.")
+ void getLocationHeaderWhenOnlyLocationIsSet() {
+ final String expectedLocationHeader = "location_header_value";
+ Iterable headers = List.of(new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader));
+ Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns header for 'content-location'.")
+ void getLocationHeaderWhenOnlyContentLocationIsSet() {
+ final String expectedContentLocationHeader = "content_location_header_value";
+ Iterable headers = List.of(new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader));
+ Assertions.assertEquals(expectedContentLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns 'location' header when both 'location' and 'content-location' are set.")
+ void getLocationHeaderWhenLocationAndContentLocationAreSet() {
+ final String expectedLocationHeader = "location_header_value";
+ final String expectedContentLocationHeader = "content_location_header_value";
+ Iterable headers = List.of(
+ new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader),
+ new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader)
+ );
+ Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
+ }
+
+ @Test
+ @DisplayName("Test that getLocationHeader returns null when no location available.")
+ void getLocationHeaderWhenNoLocationSet() {
+
+ Assertions.assertNull(FhirRequestBuilder.getLocationHeader(Collections.emptyList()));
+ }
+}
diff --git a/org.hl7.fhir.r4b/pom.xml b/org.hl7.fhir.r4b/pom.xml
index 919630c72..a25a275d8 100644
--- a/org.hl7.fhir.r4b/pom.xml
+++ b/org.hl7.fhir.r4b/pom.xml
@@ -34,6 +34,12 @@
+
+ org.projectlombok
+ lombok
+ provided
+
+
org.apache.commons
diff --git a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/terminologies/TerminologyCacheManager.java b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/terminologies/TerminologyCacheManager.java
index 68b94ed83..819d36fc2 100644
--- a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/terminologies/TerminologyCacheManager.java
+++ b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/terminologies/TerminologyCacheManager.java
@@ -1,22 +1,14 @@
package org.hl7.fhir.r4b.terminologies;
-import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
-import java.io.DataOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
-import java.io.InputStreamReader;
import java.io.OutputStream;
-import java.net.HttpURLConnection;
-import java.net.MalformedURLException;
-import java.net.URL;
-import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
-import java.util.Base64;
import java.util.Date;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
@@ -156,9 +148,9 @@ public class TerminologyCacheManager {
String url = "https://tx.fhir.org/post/tx-cache/" + ghOrg + "/" + ghRepo + "/" + ghBranch + ".zip";
System.out.println("Sending tx-cache to " + url + " (" + Utilities.describeSize(bs.toByteArray().length) + ")");
- HTTPResult res = ManagedWebAccess.builder()
+ HTTPResult res = ManagedWebAccess.accessor()
.withBasicAuth(token.substring(0, token.indexOf(':')), token.substring(token.indexOf(':') + 1))
- .withAccept("application/zip").put(url, bs.toByteArray(), null);
+ .put(url, bs.toByteArray(), null, "application/zip");
if (res.getCode() >= 300) {
System.out.println("sending cache failed: " + res.getCode());
} else {
diff --git a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClient.java b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClient.java
index 65f74dc19..558763932 100644
--- a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClient.java
+++ b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClient.java
@@ -1,7 +1,8 @@
package org.hl7.fhir.r4b.utils.client;
-import okhttp3.Headers;
-import okhttp3.internal.http2.Header;
+import lombok.Getter;
+import lombok.Setter;
+
import org.hl7.fhir.exceptions.FHIRException;
/*
@@ -37,14 +38,13 @@ import org.hl7.fhir.r4b.model.*;
import org.hl7.fhir.r4b.model.Parameters.ParametersParameterComponent;
import org.hl7.fhir.r4b.utils.client.network.ByteUtils;
import org.hl7.fhir.r4b.utils.client.network.Client;
-import org.hl7.fhir.r4b.utils.client.network.ClientHeaders;
import org.hl7.fhir.r4b.utils.client.network.ResourceRequest;
import org.hl7.fhir.utilities.FHIRBaseToolingClient;
import org.hl7.fhir.utilities.ToolingClientLogger;
import org.hl7.fhir.utilities.Utilities;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import java.io.IOException;
-import java.net.MalformedURLException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.*;
@@ -90,12 +90,21 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
private ResourceFormat preferredResourceFormat;
private int maxResultSetSize = -1;// _count
private CapabilityStatement capabilities;
+ @Getter
+ @Setter
private Client client = new Client();
- private ArrayList headers = new ArrayList<>();
- private String username;
- private String password;
+ private List headers = new ArrayList<>();
+
+ @Setter
+ @Getter
private String userAgent;
+ @Setter
+ private String acceptLanguage;
+
+ @Setter
+ private String contentLanguage;
+
// Pass endpoint for client - URI
public FHIRToolingClient(String baseServiceUrl, String userAgent) throws URISyntaxException {
preferredResourceFormat = ResourceFormat.RESOURCE_XML;
@@ -111,14 +120,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
checkCapabilities();
}
- public Client getClient() {
- return client;
- }
-
- public void setClient(Client client) {
- this.client = client;
- }
-
private void checkCapabilities() {
try {
capabilities = getCapabilitiesStatementQuick();
@@ -146,7 +147,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
TerminologyCapabilities capabilities = null;
try {
capabilities = (TerminologyCapabilities) client.issueGetResourceRequest(resourceAddress.resolveMetadataTxCaps(),
- getPreferredResourceFormat(), generateHeaders(), "TerminologyCapabilities", timeoutNormal).getReference();
+ getPreferredResourceFormat(), generateHeaders(false), "TerminologyCapabilities", timeoutNormal).getReference();
} catch (Exception e) {
throw new FHIRException("Error fetching the server's terminology capabilities", e);
}
@@ -157,7 +158,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
CapabilityStatement conformance = null;
try {
conformance = (CapabilityStatement) client.issueGetResourceRequest(resourceAddress.resolveMetadataUri(false),
- getPreferredResourceFormat(), generateHeaders(), "CapabilitiesStatement", timeoutNormal).getReference();
+ getPreferredResourceFormat(), generateHeaders(false), "CapabilitiesStatement", timeoutNormal).getReference();
} catch (Exception e) {
throw new FHIRException("Error fetching the server's conformance statement", e);
}
@@ -169,7 +170,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
return capabilities;
try {
capabilities = (CapabilityStatement) client.issueGetResourceRequest(resourceAddress.resolveMetadataUri(true),
- getPreferredResourceFormat(), generateHeaders(), "CapabilitiesStatement-Quick", timeoutNormal)
+ getPreferredResourceFormat(), generateHeaders(false), "CapabilitiesStatement-Quick", timeoutNormal)
.getReference();
} catch (Exception e) {
throw new FHIRException("Error fetching the server's capability statement: " + e.getMessage(), e);
@@ -181,7 +182,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
ResourceRequest result = null;
try {
result = client.issueGetResourceRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
- getPreferredResourceFormat(), generateHeaders(), "Read " + resourceClass.getName() + "/" + id,
+ getPreferredResourceFormat(), generateHeaders(false), "Read " + resourceClass.getName() + "/" + id,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
@@ -198,7 +199,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
try {
result = client.issueGetResourceRequest(
resourceAddress.resolveGetUriFromResourceClassAndIdAndVersion(resourceClass, id, version),
- getPreferredResourceFormat(), generateHeaders(),
+ getPreferredResourceFormat(), generateHeaders(false),
"VRead " + resourceClass.getName() + "/" + id + "/?_history/" + version, timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
@@ -215,7 +216,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
try {
result = client.issueGetResourceRequest(
resourceAddress.resolveGetUriFromResourceClassAndCanonical(resourceClass, canonicalURL),
- getPreferredResourceFormat(), generateHeaders(), "Read " + resourceClass.getName() + "?url=" + canonicalURL,
+ getPreferredResourceFormat(), generateHeaders(false), "Read " + resourceClass.getName() + "?url=" + canonicalURL,
timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
@@ -238,7 +239,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
result = client.issuePutRequest(
resourceAddress.resolveGetUriFromResourceClassAndId(resource.getClass(), resource.getId()),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
- getPreferredResourceFormat(), generateHeaders(), "Update " + resource.fhirType() + "/" + resource.getId(),
+ getPreferredResourceFormat(), generateHeaders(true), "Update " + resource.fhirType() + "/" + resource.getId(),
timeoutOperation);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
@@ -267,7 +268,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
try {
result = client.issuePutRequest(resourceAddress.resolveGetUriFromResourceClassAndId(resourceClass, id),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
- getPreferredResourceFormat(), generateHeaders(), "Update " + resource.fhirType() + "/" + id,
+ getPreferredResourceFormat(), generateHeaders(true), "Update " + resource.fhirType() + "/" + id,
timeoutOperation);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
@@ -305,10 +306,10 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
URI url = resourceAddress.resolveOperationURLFromClass(resourceClass, name, ps);
if (complex) {
byte[] body = ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true);
- result = client.issuePostRequest(url, body, getPreferredResourceFormat(), generateHeaders(),
+ result = client.issuePostRequest(url, body, getPreferredResourceFormat(), generateHeaders(true),
"POST " + resourceClass.getName() + "/$" + name, timeoutLong);
} else {
- result = client.issueGetResourceRequest(url, getPreferredResourceFormat(), generateHeaders(),
+ result = client.issueGetResourceRequest(url, getPreferredResourceFormat(), generateHeaders(false),
"GET " + resourceClass.getName() + "/$" + name, timeoutLong);
}
if (result.isUnsuccessfulRequest()) {
@@ -334,7 +335,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
try {
transactionResult = client.postBatchRequest(resourceAddress.getBaseServiceUri(),
ByteUtils.resourceToByteArray(batch, false, isJson(getPreferredResourceFormat()), false),
- getPreferredResourceFormat(), generateHeaders(), "transaction",
+ getPreferredResourceFormat(), generateHeaders(true), "transaction",
timeoutOperation + (timeoutEntry * batch.getEntry().size()));
} catch (Exception e) {
handleException("An error occurred trying to process this transaction request", e);
@@ -348,7 +349,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
try {
result = client.issuePostRequest(resourceAddress.resolveValidateUri(resourceClass, id),
ByteUtils.resourceToByteArray(resource, false, isJson(getPreferredResourceFormat()), false),
- getPreferredResourceFormat(), generateHeaders(),
+ getPreferredResourceFormat(), generateHeaders(true),
"POST " + resourceClass.getName() + (id != null ? "/" + id : "") + "/$validate", timeoutLong);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
@@ -361,10 +362,14 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
}
/**
- * Helper method to prevent nesting of previously thrown EFhirClientExceptions
+ * Helper method to prevent nesting of previously thrown EFhirClientExceptions. If the e param is an instance of
+ * EFhirClientException, it will be rethrown. Otherwise, a new EFhirClientException will be thrown with e as the
+ * cause.
*
- * @param e
- * @throws EFhirClientException
+
+ * @param message The EFhirClientException message.
+ * @param e The exception
+ * @throws EFhirClientException EFhirClientException representing the exception.
*/
protected void handleException(String message, Exception e) throws EFhirClientException {
if (e instanceof EFhirClientException) {
@@ -378,8 +383,8 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
* Helper method to determine whether desired resource representation is Json or
* XML.
*
- * @param format
- * @return
+ * @param format the format
+ * @return true if the format is JSON, false otherwise
*/
protected boolean isJson(String format) {
boolean isJson = false;
@@ -406,7 +411,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
try {
result = client.issuePostRequest(resourceAddress.resolveOperationUri(ValueSet.class, "expand"),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true), getPreferredResourceFormat(),
- generateHeaders(), "ValueSet/$expand?url=" + source.getUrl(), timeoutExpand);
+ generateHeaders(true), "ValueSet/$expand?url=" + source.getUrl(), timeoutExpand);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
(OperationOutcome) result.getPayload());
@@ -421,7 +426,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
org.hl7.fhir.r4b.utils.client.network.ResourceRequest result = null;
try {
result = client.issueGetResourceRequest(resourceAddress.resolveOperationUri(CodeSystem.class, "lookup", params),
- getPreferredResourceFormat(), generateHeaders(), "CodeSystem/$lookup", timeoutNormal);
+ getPreferredResourceFormat(), generateHeaders(false), "CodeSystem/$lookup", timeoutNormal);
} catch (IOException e) {
e.printStackTrace();
}
@@ -443,7 +448,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
result = client.issuePostRequest(resourceAddress.resolveOperationUri(ValueSet.class, "expand", params),
ByteUtils.resourceToByteArray(p, false, isJson(getPreferredResourceFormat()), true), getPreferredResourceFormat(),
- generateHeaders(), "ValueSet/$expand?url=" + source.getUrl(), timeoutExpand);
+ generateHeaders(true), "ValueSet/$expand?url=" + source.getUrl(), timeoutExpand);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
(OperationOutcome) result.getPayload());
@@ -466,7 +471,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
result = client.issuePostRequest(
resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true),
- getPreferredResourceFormat(), generateHeaders(), "Closure?name=" + name, timeoutNormal);
+ getPreferredResourceFormat(), generateHeaders(true), "Closure?name=" + name, timeoutNormal);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
(OperationOutcome) result.getPayload());
@@ -486,7 +491,7 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
result = client.issuePostRequest(
resourceAddress.resolveOperationUri(null, "closure", new HashMap()),
ByteUtils.resourceToByteArray(params, false, isJson(getPreferredResourceFormat()), true),
- getPreferredResourceFormat(), generateHeaders(), "UpdateClosure?name=" + name, timeoutOperation);
+ getPreferredResourceFormat(), generateHeaders(true), "UpdateClosure?name=" + name, timeoutOperation);
if (result.isUnsuccessfulRequest()) {
throw new EFhirClientException("Server returned error code " + result.getHttpStatus(),
(OperationOutcome) result.getPayload());
@@ -497,22 +502,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
return result == null ? null : (ConceptMap) result.getPayload();
}
- public String getUsername() {
- return username;
- }
-
- public void setUsername(String username) {
- this.username = username;
- }
-
- public String getPassword() {
- return password;
- }
-
- public void setPassword(String password) {
- this.password = password;
- }
-
public long getTimeout() {
return client.getTimeout();
}
@@ -537,42 +526,27 @@ public class FHIRToolingClient extends FHIRBaseToolingClient{
client.setRetryCount(retryCount);
}
- public void setClientHeaders(ArrayList headers) {
- this.headers = headers;
+ public void setClientHeaders(Iterable headers) {
+ this.headers = new ArrayList<>();
+ headers.forEach(this.headers::add);
}
- private Headers generateHeaders() {
- Headers.Builder builder = new Headers.Builder();
- // Add basic auth header if it exists
- if (basicAuthHeaderExists()) {
- builder.add(getAuthorizationHeader().toString());
- }
+ private Iterable generateHeaders(boolean hasBody) {
// Add any other headers
- if (this.headers != null) {
- this.headers.forEach(header -> builder.add(header.toString()));
- }
+ List headers = new ArrayList<>(this.headers);
if (!Utilities.noString(userAgent)) {
- builder.add("User-Agent: " + userAgent);
+ headers.add(new HTTPHeader("User-Agent",userAgent));
}
- return builder.build();
- }
- public boolean basicAuthHeaderExists() {
- return (username != null) && (password != null);
- }
+ if (!Utilities.noString(acceptLanguage)) {
+ headers.add(new HTTPHeader("Accept-Language", acceptLanguage));
+ }
- public Header getAuthorizationHeader() {
- String usernamePassword = username + ":" + password;
- String base64usernamePassword = Base64.getEncoder().encodeToString(usernamePassword.getBytes());
- return new Header("Authorization", "Basic " + base64usernamePassword);
- }
+ if (hasBody && !Utilities.noString(contentLanguage)) {
+ headers.add(new HTTPHeader("Content-Language",contentLanguage));
+ }
- public String getUserAgent() {
- return userAgent;
- }
-
- public void setUserAgent(String userAgent) {
- this.userAgent = userAgent;
+ return headers;
}
public String getServerVersion() {
diff --git a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/Client.java b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/Client.java
index 6cf5fa10f..d5f145f07 100644
--- a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/Client.java
+++ b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/Client.java
@@ -1,16 +1,17 @@
package org.hl7.fhir.r4b.utils.client.network;
-import okhttp3.Headers;
-import okhttp3.MediaType;
-import okhttp3.Request;
-import okhttp3.RequestBody;
+import lombok.Getter;
+import lombok.Setter;
import org.hl7.fhir.r4b.model.Bundle;
import org.hl7.fhir.r4b.model.Resource;
import org.hl7.fhir.r4b.utils.client.EFhirClientException;
import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import java.io.IOException;
import java.net.URI;
+import java.util.Collections;
import java.util.Map;
import java.util.concurrent.TimeUnit;
@@ -18,144 +19,132 @@ public class Client {
public static final String DEFAULT_CHARSET = "UTF-8";
private static final long DEFAULT_TIMEOUT = 5000;
+ @Getter @Setter
private ToolingClientLogger logger;
- private FhirLoggingInterceptor fhirLoggingInterceptor;
+
+ @Setter
+ @Getter
private int retryCount;
+
+ @Setter
+ @Getter
private long timeout = DEFAULT_TIMEOUT;
- private byte[] payload;
+
+ @Setter @Getter
private String base;
-
- public String getBase() {
- return base;
- }
-
- public void setBase(String base) {
- this.base = base;
- }
-
-
- public ToolingClientLogger getLogger() {
- return logger;
- }
-
- public void setLogger(ToolingClientLogger logger) {
- this.logger = logger;
- this.fhirLoggingInterceptor = new FhirLoggingInterceptor(logger);
- }
-
- public int getRetryCount() {
- return retryCount;
- }
-
- public void setRetryCount(int retryCount) {
- this.retryCount = retryCount;
- }
-
- public long getTimeout() {
- return timeout;
- }
-
- public void setTimeout(long timeout) {
- this.timeout = timeout;
- }
public ResourceRequest issueOptionsRequest(URI optionsUri, String resourceFormat,
String message, long timeout) throws IOException {
- this.payload = null;
- Request.Builder request = new Request.Builder().method("OPTIONS", null).url(optionsUri.toURL());
-
- return executeFhirRequest(request, resourceFormat, new Headers.Builder().build(), message, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(optionsUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.OPTIONS);
+ return executeFhirRequest(request, resourceFormat, Collections.emptyList(), message, retryCount, timeout);
}
public ResourceRequest issueGetResourceRequest(URI resourceUri, String resourceFormat,
- Headers headers, String message, long timeout) throws IOException {
- this.payload = null;
- Request.Builder request = new Request.Builder().url(resourceUri.toURL());
-
+ Iterable headers, String message, long timeout) throws IOException {
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
- public int tester(int trytry) {
- return 5;
- }
-
public ResourceRequest issuePutRequest(URI resourceUri, byte[] payload, String resourceFormat,
String message, long timeout) throws IOException {
- return issuePutRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePutRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePutRequest(URI resourceUri, byte[] payload, String resourceFormat,
- Headers headers, String message, long timeout) throws IOException {
+ Iterable headers, String message, long timeout) throws IOException {
if (payload == null)
throw new EFhirClientException("PUT requests require a non-null payload");
- this.payload = payload;
- RequestBody body = RequestBody.create(payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).put(body);
+
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.PUT)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
public ResourceRequest issuePostRequest(URI resourceUri, byte[] payload,
String resourceFormat, String message, long timeout) throws IOException {
- return issuePostRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePostRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePostRequest(URI resourceUri, byte[] payload,
- String resourceFormat, Headers headers, String message, long timeout) throws IOException {
+ String resourceFormat, Iterable headers, String message, long timeout) throws IOException {
if (payload == null)
throw new EFhirClientException("POST requests require a non-null payload");
- this.payload = payload;
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).post(body);
+
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
public boolean issueDeleteRequest(URI resourceUri) throws IOException {
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).delete();
- return executeFhirRequest(request, null, new Headers.Builder().build(), null, retryCount, timeout)
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.DELETE);
+ return executeFhirRequest(request, null, Collections.emptyList(), null, retryCount, timeout)
.isSuccessfulRequest();
}
public Bundle issueGetFeedRequest(URI resourceUri, String resourceFormat) throws IOException {
- Request.Builder request = new Request.Builder().url(resourceUri.toURL());
-
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
public Bundle issuePostFeedRequest(URI resourceUri, Map parameters, String resourceName,
Resource resource, String resourceFormat) throws IOException {
String boundary = "----WebKitFormBoundarykbMUo6H8QaUnYtRy";
byte[] payload = ByteUtils.encodeFormSubmission(parameters, resourceName, resource, boundary);
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).post(body);
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
- public Bundle postBatchRequest(URI resourceUri, byte[] payload, String resourceFormat, Headers headers,
+ public Bundle postBatchRequest(URI resourceUri, byte[] payload, String resourceFormat, Iterable headers,
String message, int timeout) throws IOException {
if (payload == null)
throw new EFhirClientException("POST requests require a non-null payload");
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder().url(resourceUri.toURL()).post(body);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeBundleRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
- public Bundle executeBundleRequest(Request.Builder request, String resourceFormat,
- Headers headers, String message, int retryCount, long timeout) throws IOException {
- return new FhirRequestBuilder(request, base).withLogger(fhirLoggingInterceptor).withResourceFormat(resourceFormat)
+ private static String getContentTypeWithDefaultCharset(String resourceFormat) {
+ return resourceFormat + ";charset=" + DEFAULT_CHARSET;
+ }
+
+ public Bundle executeBundleRequest(HTTPRequest request, String resourceFormat,
+ Iterable headers, String message, int retryCount, long timeout) throws IOException {
+ return new FhirRequestBuilder(request, base).withLogger(logger).withResourceFormat(resourceFormat)
.withRetryCount(retryCount).withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS).executeAsBatch();
}
- public ResourceRequest executeFhirRequest(Request.Builder request, String resourceFormat,
- Headers headers, String message, int retryCount, long timeout) throws IOException {
- return new FhirRequestBuilder(request, base).withLogger(fhirLoggingInterceptor).withResourceFormat(resourceFormat)
+ public ResourceRequest executeFhirRequest(HTTPRequest request, String resourceFormat,
+ Iterable headers, String message, int retryCount, long timeout) throws IOException {
+ return new FhirRequestBuilder(request, base).withLogger(logger).withResourceFormat(resourceFormat)
.withRetryCount(retryCount).withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS).execute();
}
}
diff --git a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/ClientHeaders.java b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/ClientHeaders.java
index 6b6999d8d..b837bd094 100644
--- a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/ClientHeaders.java
+++ b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/ClientHeaders.java
@@ -1,7 +1,8 @@
package org.hl7.fhir.r4b.utils.client.network;
-import okhttp3.internal.http2.Header;
+
import org.hl7.fhir.exceptions.FHIRException;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import java.util.ArrayList;
import java.util.List;
@@ -9,35 +10,35 @@ import java.util.stream.Collectors;
/**
* Generic Implementation of Client Headers.
- *
+ *
* Stores a list of headers for HTTP calls to the TX server. Users can implement
* their own instance if they desire specific, custom behavior.
*/
public class ClientHeaders {
- private final ArrayList headers;
+ private final ArrayList headers;
public ClientHeaders() {
this.headers = new ArrayList<>();
}
- public ClientHeaders(ArrayList headers) {
+ public ClientHeaders(ArrayList headers) {
this.headers = headers;
}
- public ArrayList headers() {
+ public ArrayList headers() {
return headers;
}
/**
* Add a header to the list of stored headers for network operations.
*
- * @param header {@link Header} to add to the list.
+ * @param header {@link HTTPHeader} to add to the list.
* @throws FHIRException if the header being added is a duplicate.
*/
- public ClientHeaders addHeader(Header header) throws FHIRException {
+ public ClientHeaders addHeader(HTTPHeader header) throws FHIRException {
if (headers.contains(header)) {
- throw new FHIRException("Attempting to add duplicate header, <" + header.name + ", " + header.value + ">.");
+ throw new FHIRException("Attempting to add duplicate header, <" + header.getName() + ", " + header.getValue() + ">.");
}
headers.add(header);
return this;
@@ -46,10 +47,10 @@ public class ClientHeaders {
/**
* Add a header to the list of stored headers for network operations.
*
- * @param headerList {@link List} of {@link Header} to add.
+ * @param headerList {@link List} of {@link HTTPHeader} to add.
* @throws FHIRException if any of the headers being added is a duplicate.
*/
- public ClientHeaders addHeaders(List headerList) throws FHIRException {
+ public ClientHeaders addHeaders(List headerList) throws FHIRException {
headerList.forEach(this::addHeader);
return this;
}
@@ -57,13 +58,13 @@ public class ClientHeaders {
/**
* Removes the passed in header from the list of stored headers.
*
- * @param header {@link Header} to remove from the list.
+ * @param header {@link HTTPHeader} to remove from the list.
* @throws FHIRException if the header passed in does not exist within the
* stored list.
*/
- public ClientHeaders removeHeader(Header header) throws FHIRException {
+ public ClientHeaders removeHeader(HTTPHeader header) throws FHIRException {
if (!headers.remove(header)) {
- throw new FHIRException("Attempting to remove header, <" + header.name + ", " + header.value
+ throw new FHIRException("Attempting to remove header, <" + header.getName() + ", " + header.getValue()
+ ">, from GenericClientHeaders that is not currently stored.");
}
return this;
@@ -72,17 +73,17 @@ public class ClientHeaders {
/**
* Removes the passed in headers from the list of stored headers.
*
- * @param headerList {@link List} of {@link Header} to remove.
+ * @param headerList {@link List} of {@link HTTPHeader} to remove.
* @throws FHIRException if any of the headers passed in does not exist within
* the stored list.
*/
- public ClientHeaders removeHeaders(List headerList) throws FHIRException {
+ public ClientHeaders removeHeaders(List headerList) throws FHIRException {
headerList.forEach(this::removeHeader);
return this;
}
/**
- * Clears all stored {@link Header}.
+ * Clears all stored {@link HTTPHeader}.
*/
public ClientHeaders clearHeaders() {
headers.clear();
@@ -91,7 +92,7 @@ public class ClientHeaders {
@Override
public String toString() {
- return this.headers.stream().map(header -> "\t" + header.name + ":" + header.value)
+ return this.headers.stream().map(header -> "\t" + header.getName() + ":" + header.getValue())
.collect(Collectors.joining(",\n", "{\n", "\n}"));
}
}
diff --git a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/FhirLoggingInterceptor.java b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/FhirLoggingInterceptor.java
deleted file mode 100644
index 7a9a6a282..000000000
--- a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/FhirLoggingInterceptor.java
+++ /dev/null
@@ -1,69 +0,0 @@
-package org.hl7.fhir.r4b.utils.client.network;
-
-import okhttp3.*;
-import okio.Buffer;
-
-import org.hl7.fhir.utilities.ToolingClientLogger;
-
-import javax.annotation.Nonnull;
-import java.io.IOException;
-import java.util.ArrayList;
-import java.util.List;
-import java.util.Map;
-
-public class FhirLoggingInterceptor implements Interceptor {
-
- private ToolingClientLogger logger;
-
- public FhirLoggingInterceptor(ToolingClientLogger logger) {
- this.logger = logger;
- }
-
- public FhirLoggingInterceptor setLogger(ToolingClientLogger logger) {
- this.logger = logger;
- return this;
- }
-
- @Override
- public Response intercept(@Nonnull Interceptor.Chain chain) throws IOException {
- // Log Request
- Request request = chain.request();
- List hdrs = new ArrayList<>();
- for (String s : request.headers().toString().split("\\n")) {
- hdrs.add(s.trim());
- }
- byte[] cnt = null;
- if (request.body() != null) {
- Buffer buf = new Buffer();
- request.body().writeTo(buf);
- cnt = buf.readByteArray();
- }
- if (logger != null) {
- logger.logRequest(request.method(), request.url().toString(), hdrs, cnt);
- }
-
- // Log Response
- Response response = null;
- response = chain.proceed(chain.request());
-
- MediaType contentType = null;
- byte[] bodyBytes = null;
- if (response.body() != null) {
- contentType = response.body().contentType();
- bodyBytes = response.body().bytes();
- }
-
- // Get Headers as List
- List headerList = new ArrayList<>();
- Map> headerMap = response.headers().toMultimap();
- headerMap.keySet().forEach(key -> headerMap.get(key).forEach(value -> headerList.add(key + ":" + value)));
-
- if (logger != null) {
- logger.logResponse(Integer.toString(response.code()), headerList, bodyBytes, 0);
- }
-
- // Reading byte[] clears body. Need to recreate.
- ResponseBody body = ResponseBody.create(bodyBytes, contentType);
- return response.newBuilder().body(body).build();
- }
-}
diff --git a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilder.java b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilder.java
index fa7a66300..d8f253683 100644
--- a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilder.java
+++ b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilder.java
@@ -1,11 +1,10 @@
package org.hl7.fhir.r4b.utils.client.network;
import java.io.IOException;
+import java.util.ArrayList;
import java.util.List;
-import java.util.Map;
import java.util.concurrent.TimeUnit;
-import javax.annotation.Nonnull;
import org.apache.commons.lang3.StringUtils;
import org.hl7.fhir.r4b.formats.IParser;
@@ -18,15 +17,10 @@ import org.hl7.fhir.r4b.utils.OperationOutcomeUtilities;
import org.hl7.fhir.r4b.utils.ResourceUtilities;
import org.hl7.fhir.r4b.utils.client.EFhirClientException;
import org.hl7.fhir.r4b.utils.client.ResourceFormat;
+import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.hl7.fhir.utilities.http.*;
import org.hl7.fhir.utilities.xhtml.XhtmlUtils;
-import okhttp3.Authenticator;
-import okhttp3.Credentials;
-import okhttp3.Headers;
-import okhttp3.OkHttpClient;
-import okhttp3.Request;
-import okhttp3.Response;
-
public class FhirRequestBuilder {
protected static final String HTTP_PROXY_USER = "http.proxyUser";
@@ -35,13 +29,10 @@ public class FhirRequestBuilder {
protected static final String LOCATION_HEADER = "location";
protected static final String CONTENT_LOCATION_HEADER = "content-location";
protected static final String DEFAULT_CHARSET = "UTF-8";
- /**
- * The singleton instance of the HttpClient, used for all requests.
- */
- private static OkHttpClient okHttpClient;
- private final Request.Builder httpRequest;
+
+ private final HTTPRequest httpRequest;
private String resourceFormat = null;
- private Headers headers = null;
+ private Iterable headers = null;
private String message = null;
private int retryCount = 1;
/**
@@ -55,63 +46,43 @@ public class FhirRequestBuilder {
private TimeUnit timeoutUnit = TimeUnit.MILLISECONDS;
/**
- * {@link FhirLoggingInterceptor} for log output.
+ * {@link ToolingClientLogger} for log output.
*/
- private FhirLoggingInterceptor logger = null;
+ private ToolingClientLogger logger = null;
private String source;
- public FhirRequestBuilder(Request.Builder httpRequest, String source) {
- this.httpRequest = httpRequest;
+ public FhirRequestBuilder(HTTPRequest httpRequest, String source) {
this.source = source;
+ this.httpRequest = httpRequest;
}
/**
* Adds necessary default headers, formatting headers, and any passed in
- * {@link Headers} to the passed in {@link okhttp3.Request.Builder}
+ * {@link HTTPHeader} to the passed in {@link okhttp3.Request.Builder}
*
* @param request {@link okhttp3.Request.Builder} to add headers to.
* @param format Expected {@link Resource} format.
- * @param headers Any additional {@link Headers} to add to the request.
+ * @param headers Any additional {@link HTTPHeader} to add to the request.
*/
- protected static void formatHeaders(Request.Builder request, String format, Headers headers) {
- addDefaultHeaders(request, headers);
- if (format != null)
- addResourceFormatHeaders(request, format);
- if (headers != null)
- addHeaders(request, headers);
+ protected static HTTPRequest formatHeaders(HTTPRequest request, String format, Iterable headers) {
+ List allHeaders = new ArrayList<>();
+ request.getHeaders().forEach(allHeaders::add);
+
+ if (format != null) getResourceFormatHeaders(request, format).forEach(allHeaders::add);
+ if (headers != null) headers.forEach(allHeaders::add);
+ return request.withHeaders(allHeaders);
}
- /**
- * Adds necessary headers for all REST requests.
- * User-Agent : hapi-fhir-tooling-client
- *
- * @param request {@link Request.Builder} to add default headers to.
- */
- protected static void addDefaultHeaders(Request.Builder request, Headers headers) {
- if (headers == null || !headers.names().contains("User-Agent")) {
- request.addHeader("User-Agent", "hapi-fhir-tooling-client");
+ protected static Iterable getResourceFormatHeaders(HTTPRequest httpRequest, String format) {
+ List headers = new ArrayList<>();
+ headers.add(new HTTPHeader("Accept", format));
+ if (httpRequest.getMethod() == HTTPRequest.HttpMethod.PUT
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.POST
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.PATCH
+ ) {
+ headers.add( new HTTPHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET));
}
- }
-
- /**
- * Adds necessary headers for the given resource format provided.
- *
- * @param request {@link Request.Builder} to add default headers to.
- */
- protected static void addResourceFormatHeaders(Request.Builder request, String format) {
- request.addHeader("Accept", format);
- request.addHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET);
- }
-
- /**
- * Iterates through the passed in {@link Headers} and adds them to the provided
- * {@link Request.Builder}.
- *
- * @param request {@link Request.Builder} to add headers to.
- * @param headers {@link Headers} to add to request.
- */
- protected static void addHeaders(Request.Builder request, Headers headers) {
- headers.forEach(header -> request.addHeader(header.getFirst(), header.getSecond()));
+ return headers;
}
/**
@@ -131,68 +102,24 @@ public class FhirRequestBuilder {
}
/**
- * Extracts the 'location' header from the passes in {@link Headers}. If no
+ * Extracts the 'location' header from the passed in {@link HTTPHeader}s. If no
* value for 'location' exists, the value for 'content-location' is returned. If
* neither header exists, we return null.
*
- * @param headers {@link Headers} to evaluate
+ * @param headers {@link HTTPHeader}s to evaluate
* @return {@link String} header value, or null if no location headers are set.
*/
- protected static String getLocationHeader(Headers headers) {
- Map> headerMap = headers.toMultimap();
- if (headerMap.containsKey(LOCATION_HEADER)) {
- return headerMap.get(LOCATION_HEADER).get(0);
- } else if (headerMap.containsKey(CONTENT_LOCATION_HEADER)) {
- return headerMap.get(CONTENT_LOCATION_HEADER).get(0);
- } else {
- return null;
+ protected static String getLocationHeader(Iterable headers) {
+ String locationHeader = HTTPHeaderUtil.getSingleHeader(headers, LOCATION_HEADER);
+
+ if (locationHeader != null) {
+ return locationHeader;
}
+ return HTTPHeaderUtil.getSingleHeader(headers, CONTENT_LOCATION_HEADER);
}
- /**
- * We only ever want to have one copy of the HttpClient kicking around at any
- * given time. If we need to make changes to any configuration, such as proxy
- * settings, timeout, caches, etc, we can do a per-call configuration through
- * the {@link OkHttpClient#newBuilder()} method. That will return a builder that
- * shares the same connection pool, dispatcher, and configuration with the
- * original client.
- *
- * The {@link OkHttpClient} uses the proxy auth properties set in the current
- * system properties. The reason we don't set the proxy address and
- * authentication explicitly, is due to the fact that this class is often used
- * in conjunction with other http client tools which rely on the
- * system.properties settings to determine proxy settings. It's easier to keep
- * the method consistent across the board. ...for now.
- *
- * @return {@link OkHttpClient} instance
- */
- protected OkHttpClient getHttpClient() {
- if (okHttpClient == null) {
- okHttpClient = new OkHttpClient();
- }
-
- Authenticator proxyAuthenticator = getAuthenticator();
-
- OkHttpClient.Builder builder = okHttpClient.newBuilder();
- if (logger != null)
- builder.addInterceptor(logger);
- builder.addInterceptor(new RetryInterceptor(retryCount));
-
- return builder.connectTimeout(timeout, timeoutUnit).writeTimeout(timeout, timeoutUnit)
- .readTimeout(timeout, timeoutUnit).proxyAuthenticator(proxyAuthenticator).build();
- }
-
- @Nonnull
- private static Authenticator getAuthenticator() {
- return (route, response) -> {
- final String httpProxyUser = System.getProperty(HTTP_PROXY_USER);
- final String httpProxyPass = System.getProperty(HTTP_PROXY_PASS);
- if (httpProxyUser != null && httpProxyPass != null) {
- String credential = Credentials.basic(httpProxyUser, httpProxyPass);
- return response.request().newBuilder().header(HEADER_PROXY_AUTH, credential).build();
- }
- return response.request().newBuilder().build();
- };
+ protected ManagedFhirWebAccessor getManagedWebAccessor() {
+ return ManagedWebAccess.fhirAccessor().withRetries(retryCount).withTimeout(timeout, timeoutUnit).withLogger(logger);
}
public FhirRequestBuilder withResourceFormat(String resourceFormat) {
@@ -200,7 +127,7 @@ public class FhirRequestBuilder {
return this;
}
- public FhirRequestBuilder withHeaders(Headers headers) {
+ public FhirRequestBuilder withHeaders(Iterable headers) {
this.headers = headers;
return this;
}
@@ -215,7 +142,7 @@ public class FhirRequestBuilder {
return this;
}
- public FhirRequestBuilder withLogger(FhirLoggingInterceptor logger) {
+ public FhirRequestBuilder withLogger(ToolingClientLogger logger) {
this.logger = logger;
return this;
}
@@ -226,33 +153,28 @@ public class FhirRequestBuilder {
return this;
}
- protected Request buildRequest() {
- return httpRequest.build();
- }
-
public ResourceRequest execute() throws IOException {
- formatHeaders(httpRequest, resourceFormat, headers);
- Response response = getHttpClient().newCall(httpRequest.build()).execute();
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, headers);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);
T resource = unmarshalReference(response, resourceFormat, null);
- return new ResourceRequest(resource, response.code(), getLocationHeader(response.headers()));
+ return new ResourceRequest(resource, response.getCode(), getLocationHeader(response.getHeaders()));
}
public Bundle executeAsBatch() throws IOException {
- formatHeaders(httpRequest, resourceFormat, null);
- Response response = getHttpClient().newCall(httpRequest.build()).execute();
- return unmarshalFeed(response, resourceFormat);
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, null);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);return unmarshalFeed(response, resourceFormat);
}
/**
* Unmarshalls a resource from the response stream.
*/
@SuppressWarnings("unchecked")
- protected T unmarshalReference(Response response, String format, String resourceType) {
- int code = response.code();
+ protected T unmarshalReference(HTTPResult response, String format, String resourceType) {
+ int code = response.getCode();
boolean ok = code >= 200 && code < 300;
- if (response.body() == null) {
+ if (response.getContent() == null) {
if (!ok) {
- throw new EFhirClientException(response.message());
+ throw new EFhirClientException(response.getMessage());
} else {
return null;
}
@@ -261,9 +183,9 @@ public class FhirRequestBuilder {
Resource resource = null;
try {
- body = response.body().string();
- String ct = response.header("Content-Type");
- if (ct == null) {
+ body = response.getContentAsString();
+ String contentType = HTTPHeaderUtil.getSingleHeader(response.getHeaders(), "Content-Type");
+ if (contentType == null) {
if (ok) {
resource = getParser(format).parse(body);
} else {
@@ -272,10 +194,10 @@ public class FhirRequestBuilder {
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
}
} else {
- if (ct.contains(";")) {
- ct = ct.substring(0, ct.indexOf(";"));
+ if (contentType.contains(";")) {
+ contentType = contentType.substring(0, contentType.indexOf(";"));
}
- switch (ct) {
+ switch (contentType) {
case "application/json":
case "application/fhir+json":
if (!format.contains("json")) {
@@ -295,10 +217,10 @@ public class FhirRequestBuilder {
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
break;
case "text/html" :
- resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.body().string(), source));
+ resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.getContentAsString(), source));
break;
default: // not sure what else to do?
- System.out.println("Got content-type '"+ct+"' from "+source);
+ System.out.println("Got content-type '"+contentType+"' from "+source);
System.out.println(body);
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
}
@@ -333,7 +255,7 @@ public class FhirRequestBuilder {
/**
* Unmarshalls Bundle from response stream.
*/
- protected Bundle unmarshalFeed(Response response, String format) {
+ protected Bundle unmarshalFeed(HTTPResult response, String format) {
return unmarshalReference(response, format, "Bundle");
}
diff --git a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/RetryInterceptor.java b/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/RetryInterceptor.java
deleted file mode 100644
index 931f66ddd..000000000
--- a/org.hl7.fhir.r4b/src/main/java/org/hl7/fhir/r4b/utils/client/network/RetryInterceptor.java
+++ /dev/null
@@ -1,67 +0,0 @@
-package org.hl7.fhir.r4b.utils.client.network;
-
-import okhttp3.Interceptor;
-import okhttp3.Request;
-import okhttp3.Response;
-
-import java.io.IOException;
-
-/**
- * An {@link Interceptor} for {@link okhttp3.OkHttpClient} that controls the
- * number of times we retry a to execute a given request, before reporting a
- * failure. This includes unsuccessful return codes and timeouts.
- */
-public class RetryInterceptor implements Interceptor {
-
- // Delay between retying failed requests, in millis
- private final long RETRY_TIME = 2000;
-
- // Maximum number of times to retry the request before failing
- private final int maxRetry;
-
- // Internal counter for tracking the number of times we've tried this request
- private int retryCounter = 0;
-
- public RetryInterceptor(int maxRetry) {
- this.maxRetry = maxRetry;
- }
-
- @Override
- public Response intercept(Interceptor.Chain chain) throws IOException {
- Request request = chain.request();
- Response response = null;
-
- do {
- try {
- // If we are retrying a failed request that failed due to a bad response from
- // the server, we must close it first
- if (response != null) {
-// System.out.println("Previous " + chain.request().method() + " attempt returned HTTP<" + (response.code())
-// + "> from url -> " + chain.request().url() + ".");
- response.close();
- }
- // System.out.println(chain.request().method() + " attempt <" + (retryCounter +
- // 1) + "> to url -> " + chain.request().url());
- response = chain.proceed(request);
- } catch (IOException e) {
- try {
- // Include a small break in between requests.
- Thread.sleep(RETRY_TIME);
- } catch (InterruptedException e1) {
- System.out.println(chain.request().method() + " to url -> " + chain.request().url() + " interrupted on try <"
- + retryCounter + ">");
- }
- } finally {
- retryCounter++;
- }
- } while ((response == null || !response.isSuccessful()) && (retryCounter <= maxRetry + 1));
-
- /*
- * if something has gone wrong, and we are unable to complete the request, we
- * still need to initialize the return response so we don't get a null pointer
- * exception.
- */
- return response != null ? response : chain.proceed(request);
- }
-
-}
\ No newline at end of file
diff --git a/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClientTest.java b/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClientTest.java
index eecbc0234..83099c4c6 100644
--- a/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClientTest.java
+++ b/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/FHIRToolingClientTest.java
@@ -1,79 +1,89 @@
package org.hl7.fhir.r4b.utils.client;
-import okhttp3.Headers;
-import okhttp3.Request;
-import okhttp3.internal.http2.Header;
import org.hl7.fhir.r4b.model.*;
import org.hl7.fhir.r4b.utils.client.network.Client;
import org.hl7.fhir.r4b.utils.client.network.ResourceRequest;
-import org.junit.jupiter.api.Assertions;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
-import org.mockito.ArgumentCaptor;
-import org.mockito.ArgumentMatchers;
-import org.mockito.Mockito;
+import org.mockito.*;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Arrays;
+import java.util.List;
+
+import static org.junit.jupiter.api.Assertions.assertTrue;
class FHIRToolingClientTest {
String TX_ADDR = "http://tx.fhir.org";
- Header h1 = new Header("header1", "value1");
- Header h2 = new Header("header2", "value2");
- Header h3 = new Header("header3", "value3");
+ HTTPHeader h1 = new HTTPHeader("header1", "value1");
+ HTTPHeader h2 = new HTTPHeader("header2", "value2");
+ HTTPHeader h3 = new HTTPHeader("header3", "value3");
+
+ HTTPHeader agentHeader = new HTTPHeader("User-Agent", "fhir/test-cases");
private Client mockClient;
private FHIRToolingClient toolingClient;
+ @Captor
+ private ArgumentCaptor> headersArgumentCaptor;
+
+
@BeforeEach
void setUp() throws IOException, URISyntaxException {
+ MockitoAnnotations.openMocks(this);
mockClient = Mockito.mock(Client.class);
ResourceRequest resourceResourceRequest = new ResourceRequest<>(generateBundle(), 200, "");
// GET
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
Mockito
.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new TerminologyCapabilities(), 200, "location"));
Mockito
.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
Mockito
.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
// PUT
Mockito.when(mockClient.issuePutRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
// POST
Mockito.when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong())).thenReturn(resourceResourceRequest);
Mockito
.when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.contains("validate"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.contains("validate"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new OperationOutcome(), 200, "location"));
// BUNDLE REQ
Mockito
- .when(mockClient.executeBundleRequest(Mockito.any(Request.Builder.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyInt(), Mockito.anyLong()))
+ .when(mockClient.executeBundleRequest(Mockito.any(HTTPRequest.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyLong()))
.thenReturn(generateBundle());
toolingClient = new FHIRToolingClient(TX_ADDR, "fhir/test-cases");
toolingClient.setClient(mockClient);
}
- private ArrayList getHeaders() {
+ private List getHeaders() {
return new ArrayList<>(Arrays.asList(h1, h2, h3));
}
+ private List getHeadersWithAgent() {
+ return new ArrayList<>(Arrays.asList(h1, h2, h3, agentHeader));
+ }
+
private Bundle generateBundle() {
Patient patient = generatePatient();
Observation observation = generateObservation();
@@ -115,108 +125,102 @@ class FHIRToolingClientTest {
return observation;
}
- private void checkHeaders(Headers argumentCaptorValue) {
- getHeaders().forEach(header -> {
-// System.out.println("Checking header <" + header.component1().utf8() + ", " + header.component2().utf8() + ">");
- Assertions.assertEquals(argumentCaptorValue.get(header.component1().utf8()), header.component2().utf8());
+ private void checkHeaders(Iterable argumentCaptorValue) {
+ List capturedHeaders = new ArrayList<>();
+ argumentCaptorValue.forEach(capturedHeaders::add);
+
+ getHeadersWithAgent().forEach(header -> {
+ assertTrue(capturedHeaders.contains(header));
});
}
@Test
void getTerminologyCapabilities() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getTerminologyCapabilities();
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getCapabilitiesStatement() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getCapabilitiesStatement();
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getCapabilitiesStatementQuick() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getCapabilitiesStatementQuick();
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void read() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.read(Patient.class, "id");
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void vread() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.vread(Patient.class, "id", "version");
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getCanonical() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getCanonical(Patient.class, "canonicalURL");
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void update() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.update(generatePatient());
Mockito.verify(mockClient).issuePutRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void validate() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.validate(Patient.class, generatePatient(), "id");
Mockito.verify(mockClient).issuePostRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
}
\ No newline at end of file
diff --git a/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/ClientHeadersTest.java b/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/ClientHeadersTest.java
index 75c7129d9..72051706c 100644
--- a/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/ClientHeadersTest.java
+++ b/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/ClientHeadersTest.java
@@ -1,7 +1,7 @@
package org.hl7.fhir.r4b.utils.client.network;
-import okhttp3.internal.http2.Header;
import org.hl7.fhir.exceptions.FHIRException;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
@@ -14,9 +14,9 @@ class ClientHeadersTest {
ClientHeaders clientHeaders;
- Header h1 = new Header("header1", "value1");
- Header h2 = new Header("header2", "value2");
- Header h3 = new Header("header3", "value3");
+ HTTPHeader h1 = new HTTPHeader("header1", "value1");
+ HTTPHeader h2 = new HTTPHeader("header2", "value2");
+ HTTPHeader h3 = new HTTPHeader("header3", "value3");
@BeforeEach
void setUp() {
@@ -42,7 +42,7 @@ class ClientHeadersTest {
@Test
@DisplayName("Happy path add headers as list.")
void addHeaders() {
- List headersList = Arrays.asList(h1, h2, h3);
+ List headersList = Arrays.asList(h1, h2, h3);
clientHeaders.addHeaders(headersList);
Assertions.assertEquals(3, clientHeaders.headers().size());
Assertions.assertEquals(headersList, clientHeaders.headers());
@@ -51,7 +51,7 @@ class ClientHeadersTest {
@Test
@DisplayName("Happy path add headers as list.")
void addHeadersDuplicateAdd() {
- List headersList = Arrays.asList(h1, h2, h1);
+ List headersList = Arrays.asList(h1, h2, h1);
Assertions.assertThrows(FHIRException.class, () -> clientHeaders.addHeaders(headersList));
}
@@ -63,7 +63,7 @@ class ClientHeadersTest {
clientHeaders.addHeader(h3);
clientHeaders.removeHeader(h2);
Assertions.assertEquals(2, clientHeaders.headers().size());
- clientHeaders.removeHeader(new Header("header3", "value3"));
+ clientHeaders.removeHeader(new HTTPHeader("header3", "value3"));
Assertions.assertEquals(1, clientHeaders.headers().size());
}
@@ -78,8 +78,8 @@ class ClientHeadersTest {
@Test
@DisplayName("Happy path remove list of existing headers.")
void removeHeaders() {
- List headersToAddList = Arrays.asList(h1, h2, h3);
- List headersToRemoveList = Arrays.asList(h2, h3);
+ List headersToAddList = Arrays.asList(h1, h2, h3);
+ List headersToRemoveList = Arrays.asList(h2, h3);
clientHeaders.addHeaders(headersToAddList);
clientHeaders.removeHeaders(headersToRemoveList);
Assertions.assertEquals(1, clientHeaders.headers().size());
@@ -88,15 +88,15 @@ class ClientHeadersTest {
@Test
@DisplayName("Remove list containing unknown header.")
void removeHeadersUnknown() {
- List headersToAddList = Arrays.asList(h1, h3);
- List headersToRemoveList = Arrays.asList(h2, h3);
+ List headersToAddList = Arrays.asList(h1, h3);
+ List headersToRemoveList = Arrays.asList(h2, h3);
clientHeaders.addHeaders(headersToAddList);
Assertions.assertThrows(FHIRException.class, () -> clientHeaders.removeHeaders(headersToRemoveList));
}
@Test
void clearHeaders() {
- List headersToAddList = Arrays.asList(h1, h3);
+ List headersToAddList = Arrays.asList(h1, h3);
clientHeaders.addHeaders(headersToAddList);
Assertions.assertEquals(2, clientHeaders.headers().size());
clientHeaders.clearHeaders();
diff --git a/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilderTest.java b/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilderTest.java
index 8cd65e5d3..9024625fa 100644
--- a/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilderTest.java
+++ b/org.hl7.fhir.r4b/src/test/java/org/hl7/fhir/r4b/utils/client/network/FhirRequestBuilderTest.java
@@ -1,71 +1,51 @@
package org.hl7.fhir.r4b.utils.client.network;
-import okhttp3.Headers;
-import okhttp3.OkHttpClient;
-import okhttp3.Request;
+
import org.hl7.fhir.r4b.model.OperationOutcome;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPHeaderUtil;
+import org.hl7.fhir.utilities.http.HTTPRequest;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
-import java.util.HashMap;
-import java.util.HashMap;
-import java.util.List;
-import java.util.Map;
+import java.util.*;
class FhirRequestBuilderTest {
- @Test
- @DisplayName("Test default headers are added correctly.")
- void addDefaultHeaders() {
- Request.Builder request = new Request.Builder().url("http://www.google.com");
- FhirRequestBuilder.addDefaultHeaders(request, null);
-
- Map> headersMap = request.build().headers().toMultimap();
- Assertions.assertNotNull(headersMap.get("User-Agent"), "User-Agent header null.");
- Assertions.assertEquals("hapi-fhir-tooling-client", headersMap.get("User-Agent").get(0),
- "User-Agent header not populated with expected value \"hapi-fhir-tooling-client\".");
-
- }
@Test
@DisplayName("Test resource format headers are added correctly.")
- void addResourceFormatHeaders() {
+ void addResourceFormatHeadersGET() {
String testFormat = "yaml";
- Request.Builder request = new Request.Builder().url("http://www.google.com");
- FhirRequestBuilder.addResourceFormatHeaders(request, testFormat);
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.GET);
- Map> headersMap = request.build().headers().toMultimap();
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
"Accept header not populated with expected value " + testFormat + ".");
- Assertions.assertNotNull(headersMap.get("Content-Type"), "Content-Type header null.");
- Assertions.assertEquals(testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET,
- headersMap.get("Content-Type").get(0), "Content-Type header not populated with expected value \"" + testFormat
- + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET + "\".");
+ Assertions.assertNull(headersMap.get("Content-Type"), "Content-Type header null.");
}
@Test
- @DisplayName("Test a list of provided headers are added correctly.")
- void addHeaders() {
- String headerName1 = "headerName1";
- String headerValue1 = "headerValue1";
- String headerName2 = "headerName2";
- String headerValue2 = "headerValue2";
+ @DisplayName("Test resource format headers are added correctly (POST).")
+ void addResourceFormatHeadersPOST() {
+ String testFormat = "yaml";
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.POST);
- Headers headers = new Headers.Builder().add(headerName1, headerValue1).add(headerName2, headerValue2).build();
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
- Request.Builder request = new Request.Builder().url("http://www.google.com");
- FhirRequestBuilder.addHeaders(request, headers);
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
+ Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
+ Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
+ "Accept header not populated with expected value " + testFormat + ".");
- Map> headersMap = request.build().headers().toMultimap();
- Assertions.assertNotNull(headersMap.get(headerName1), headerName1 + " header null.");
- Assertions.assertEquals(headerValue1, headersMap.get(headerName1).get(0),
- headerName1 + " header not populated with expected value " + headerValue1 + ".");
- Assertions.assertNotNull(headersMap.get(headerName2), headerName2 + " header null.");
- Assertions.assertEquals(headerValue2, headersMap.get(headerName2).get(0),
- headerName2 + " header not populated with expected value " + headerValue2 + ".");
+ Assertions.assertNotNull(headersMap.get("Content-Type"), "Content-Type header null.");
+ Assertions.assertEquals(testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET, headersMap.get("Content-Type").get(0),
+ "Content-Type header not populated with expected value \"" + testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET + "\".");
}
@Test
@@ -115,7 +95,7 @@ class FhirRequestBuilderTest {
@DisplayName("Test that getLocationHeader returns header for 'location'.")
void getLocationHeaderWhenOnlyLocationIsSet() {
final String expectedLocationHeader = "location_header_value";
- Headers headers = new Headers.Builder().add(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader).build();
+ Iterable headers = List.of(new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader));
Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
}
@@ -123,8 +103,7 @@ class FhirRequestBuilderTest {
@DisplayName("Test that getLocationHeader returns header for 'content-location'.")
void getLocationHeaderWhenOnlyContentLocationIsSet() {
final String expectedContentLocationHeader = "content_location_header_value";
- Headers headers = new Headers.Builder()
- .add(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader).build();
+ Iterable headers = List.of(new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader));
Assertions.assertEquals(expectedContentLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
}
@@ -133,15 +112,17 @@ class FhirRequestBuilderTest {
void getLocationHeaderWhenLocationAndContentLocationAreSet() {
final String expectedLocationHeader = "location_header_value";
final String expectedContentLocationHeader = "content_location_header_value";
- Headers headers = new Headers.Builder().add(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader)
- .add(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader).build();
+ Iterable headers = List.of(
+ new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader),
+ new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader)
+ );
Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
}
@Test
@DisplayName("Test that getLocationHeader returns null when no location available.")
void getLocationHeaderWhenNoLocationSet() {
- Headers headers = new Headers.Builder().build();
- Assertions.assertNull(FhirRequestBuilder.getLocationHeader(headers));
+
+ Assertions.assertNull(FhirRequestBuilder.getLocationHeader(Collections.emptyList()));
}
}
\ No newline at end of file
diff --git a/org.hl7.fhir.r5/pom.xml b/org.hl7.fhir.r5/pom.xml
index b441c2f4c..e17a32aa9 100644
--- a/org.hl7.fhir.r5/pom.xml
+++ b/org.hl7.fhir.r5/pom.xml
@@ -126,6 +126,12 @@
+
+ org.assertj
+ assertj-core
+ test
+
+
com.fasterxml.jackson.core
jackson-databind
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/fhirpath/FHIRPathEngine.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/fhirpath/FHIRPathEngine.java
index 28cda588a..c89561a64 100644
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/fhirpath/FHIRPathEngine.java
+++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/fhirpath/FHIRPathEngine.java
@@ -604,7 +604,7 @@ public class FHIRPathEngine {
warnings.addAll(typeWarnings);
return res;
}
-
+
public TypeDetails checkOnTypes(Object appContext, String resourceType, TypeDetails types, ExpressionNode expr, List warnings) throws FHIRLexerException, PathEngineException, DefinitionException {
typeWarnings.clear();
TypeDetails res = executeType(new ExecutionTypeContext(appContext, resourceType, types, types), types, expr, null, true, false, expr);
@@ -612,6 +612,13 @@ public class FHIRPathEngine {
return res;
}
+ public TypeDetails checkOnTypes(Object appContext, String resourceType, TypeDetails types, ExpressionNode expr, List warnings, boolean canBeNone) throws FHIRLexerException, PathEngineException, DefinitionException {
+ typeWarnings.clear();
+ TypeDetails res = executeType(new ExecutionTypeContext(appContext, resourceType, types, types), types, expr, null, true, canBeNone, expr);
+ warnings.addAll(typeWarnings);
+ return res;
+ }
+
/**
* check that paths referred to in the ExpressionNode are valid
*
@@ -6585,7 +6592,7 @@ public class FHIRPathEngine {
/** given an element definition in a profile, what element contains the differentiating fixed
- * for the element, given the differentiating expresssion. The expression is only allowed to
+ * for the element, given the differentiating expression. The expression is only allowed to
* use a subset of FHIRPath
*
* @param profile
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/TerminologyCacheManager.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/TerminologyCacheManager.java
index 3e63ac0a8..daa378a5b 100644
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/TerminologyCacheManager.java
+++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/TerminologyCacheManager.java
@@ -1,22 +1,14 @@
package org.hl7.fhir.r5.terminologies;
-import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
-import java.io.DataOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
-import java.io.InputStreamReader;
import java.io.OutputStream;
-import java.net.HttpURLConnection;
-import java.net.MalformedURLException;
-import java.net.URL;
-import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
-import java.util.Base64;
import java.util.Date;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
@@ -156,9 +148,9 @@ public class TerminologyCacheManager {
// post it to
String url = "https://tx.fhir.org/post/tx-cache/"+ghOrg+"/"+ghRepo+"/"+ghBranch+".zip";
System.out.println("Sending tx-cache to "+url+" ("+Utilities.describeSize(bs.toByteArray().length)+")");
- HTTPResult res = ManagedWebAccess.builder()
+ HTTPResult res = ManagedWebAccess.accessor()
.withBasicAuth(token.substring(0, token.indexOf(':')), token.substring(token.indexOf(':') + 1))
- .withAccept("application/zip").put(url, bs.toByteArray(), null);
+ .put(url, bs.toByteArray(), null, "application/zip");
if (res.getCode() >= 300) {
System.out.println("sending cache failed: "+res.getCode());
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/client/TerminologyClientR5.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/client/TerminologyClientR5.java
index 063708bb8..b0f2f7b46 100644
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/client/TerminologyClientR5.java
+++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/terminologies/client/TerminologyClientR5.java
@@ -2,7 +2,6 @@ package org.hl7.fhir.r5.terminologies.client;
import java.net.URISyntaxException;
import java.util.EnumSet;
-import java.util.HashMap;
import java.util.Map;
/*
@@ -41,13 +40,10 @@ import org.hl7.fhir.r5.model.CanonicalResource;
import org.hl7.fhir.r5.model.CapabilityStatement;
import org.hl7.fhir.r5.model.CodeSystem;
import org.hl7.fhir.r5.model.Parameters;
-import org.hl7.fhir.r5.model.Parameters.ParametersParameterComponent;
import org.hl7.fhir.r5.model.Resource;
import org.hl7.fhir.r5.model.TerminologyCapabilities;
import org.hl7.fhir.r5.model.ValueSet;
-import org.hl7.fhir.r5.terminologies.client.ITerminologyClient;
import org.hl7.fhir.r5.terminologies.client.TerminologyClientManager.ITerminologyClientFactory;
-import org.hl7.fhir.r5.terminologies.client.TerminologyClientR5.TerminologyClientR5Factory;
import org.hl7.fhir.r5.utils.client.FHIRToolingClient;
import org.hl7.fhir.r5.utils.client.network.ClientHeaders;
import org.hl7.fhir.utilities.FhirPublication;
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/FHIRToolingClient.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/FHIRToolingClient.java
index a1a112b96..cfc5e3980 100644
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/FHIRToolingClient.java
+++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/FHIRToolingClient.java
@@ -1,12 +1,11 @@
package org.hl7.fhir.r5.utils.client;
-import okhttp3.Headers;
-import okhttp3.internal.http2.Header;
+import lombok.Getter;
+import lombok.Setter;
import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.r5.model.Bundle;
import org.hl7.fhir.r5.model.OperationOutcome;
import org.hl7.fhir.r5.model.Resource;
-import org.hl7.fhir.r5.utils.client.EFhirClientException;
/*
Copyright (c) 2011+, HL7, Inc.
@@ -45,7 +44,7 @@ import org.hl7.fhir.r5.utils.client.network.ResourceRequest;
import org.hl7.fhir.utilities.FHIRBaseToolingClient;
import org.hl7.fhir.utilities.ToolingClientLogger;
import org.hl7.fhir.utilities.Utilities;
-import org.hl7.fhir.utilities.npm.FilesystemPackageCacheManager;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
@@ -93,18 +92,24 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
private String base;
private ResourceAddress resourceAddress;
+ @Setter
private ResourceFormat preferredResourceFormat;
private int maxResultSetSize = -1;//_count
private CapabilityStatement capabilities;
+ @Getter
+ @Setter
private Client client = new Client();
- private ArrayList headers = new ArrayList<>();
- private String username;
- private String password;
+ private List headers = new ArrayList<>();
+
+ @Setter
+ @Getter
private String userAgent;
+ @Setter
+ private String acceptLanguage;
- private String acceptLang;
- private String contentLang;
+ @Setter
+ private String contentLanguage;
private int useCount;
@@ -124,22 +129,10 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
this.maxResultSetSize = -1;
}
- public Client getClient() {
- return client;
- }
-
- public void setClient(Client client) {
- this.client = client;
- }
-
public String getPreferredResourceFormat() {
return preferredResourceFormat.getHeader();
}
- public void setPreferredResourceFormat(ResourceFormat resourceFormat) {
- preferredResourceFormat = resourceFormat;
- }
-
public int getMaximumRecordCount() {
return maxResultSetSize;
}
@@ -410,10 +403,14 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
}
/**
- * Helper method to prevent nesting of previously thrown EFhirClientExceptions
+ * Helper method to prevent nesting of previously thrown EFhirClientExceptions. If the e param is an instance of
+ * EFhirClientException, it will be rethrown. Otherwise, a new EFhirClientException will be thrown with e as the
+ * cause.
*
- * @param e
- * @throws EFhirClientException
+ * @param code The EFhirClientException code.
+ * @param message The EFhirClientException message.
+ * @param e The exception.
+ * @throws EFhirClientException representing the exception.
*/
protected void handleException(int code, String message, Exception e) throws EFhirClientException {
if (e instanceof EFhirClientException) {
@@ -427,8 +424,8 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
* Helper method to determine whether desired resource representation
* is Json or XML.
*
- * @param format
- * @return
+ * @param format The format
+ * @return true if the format is JSON, false otherwise
*/
protected boolean isJson(String format) {
boolean isJson = false;
@@ -573,22 +570,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return result == null ? null : (ConceptMap) result.getPayload();
}
- public String getUsername() {
- return username;
- }
-
- public void setUsername(String username) {
- this.username = username;
- }
-
- public String getPassword() {
- return password;
- }
-
- public void setPassword(String password) {
- this.password = password;
- }
-
public long getTimeout() {
return client.getTimeout();
}
@@ -613,51 +594,27 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
client.setRetryCount(retryCount);
}
- public void setClientHeaders(ArrayList headers) {
- this.headers = headers;
+ public void setClientHeaders(Iterable headers) {
+ this.headers = new ArrayList<>();
+ headers.forEach(this.headers::add);
}
- private Headers generateHeaders(boolean hasBody) {
- Headers.Builder builder = new Headers.Builder();
- // Add basic auth header if it exists
- if (basicAuthHeaderExists()) {
- builder.add(getAuthorizationHeader().toString());
- }
+ private Iterable generateHeaders(boolean hasBody) {
// Add any other headers
- if(this.headers != null) {
- this.headers.forEach(header -> builder.add(header.toString()));
- }
+ List headers = new ArrayList<>(this.headers);
if (!Utilities.noString(userAgent)) {
- builder.add("User-Agent: "+userAgent);
+ headers.add(new HTTPHeader("User-Agent",userAgent));
}
- if (!Utilities.noString(acceptLang)) {
- builder.add("Accept-Language: "+acceptLang);
+ if (!Utilities.noString(acceptLanguage)) {
+ headers.add(new HTTPHeader("Accept-Language", acceptLanguage));
}
- if (hasBody && !Utilities.noString(contentLang)) {
- builder.add("Content-Language: "+contentLang);
+ if (hasBody && !Utilities.noString(contentLanguage)) {
+ headers.add(new HTTPHeader("Content-Language", contentLanguage));
}
- return builder.build();
- }
-
- public boolean basicAuthHeaderExists() {
- return (username != null) && (password != null);
- }
-
- public Header getAuthorizationHeader() {
- String usernamePassword = username + ":" + password;
- String base64usernamePassword = Base64.getEncoder().encodeToString(usernamePassword.getBytes());
- return new Header("Authorization", "Basic " + base64usernamePassword);
- }
-
- public String getUserAgent() {
- return userAgent;
- }
-
- public void setUserAgent(String userAgent) {
- this.userAgent = userAgent;
+ return headers;
}
public String getServerVersion() {
@@ -671,14 +628,6 @@ public class FHIRToolingClient extends FHIRBaseToolingClient {
return capabilities == null ? null : capabilities.getSoftware().getVersion();
}
- public void setAcceptLanguage(String lang) {
- this.acceptLang = lang;
- }
-
- public void setContentLanguage(String lang) {
- this.contentLang = lang;
- }
-
public Bundle search(String type, String criteria) {
recordUse();
return fetchFeed(Utilities.pathURL(base, type+criteria));
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/Client.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/Client.java
index 1895c2ccf..7eea93a94 100644
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/Client.java
+++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/Client.java
@@ -1,16 +1,18 @@
package org.hl7.fhir.r5.utils.client.network;
-import okhttp3.Headers;
-import okhttp3.MediaType;
-import okhttp3.Request;
-import okhttp3.RequestBody;
+import lombok.Getter;
+import lombok.Setter;
+
import org.hl7.fhir.r5.model.Bundle;
import org.hl7.fhir.r5.model.Resource;
import org.hl7.fhir.r5.utils.client.EFhirClientException;
import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import java.io.IOException;
import java.net.URI;
+import java.util.Collections;
import java.util.Map;
import java.util.concurrent.TimeUnit;
@@ -18,93 +20,60 @@ public class Client {
public static final String DEFAULT_CHARSET = "UTF-8";
private static final long DEFAULT_TIMEOUT = 5000;
+ @Getter @Setter
private ToolingClientLogger logger;
- private FhirLoggingInterceptor fhirLoggingInterceptor;
+
+ @Setter @Getter
private int retryCount;
+ @Setter @Getter
private long timeout = DEFAULT_TIMEOUT;
- private byte[] payload;
+
+ @Setter @Getter
private String base;
-
- public String getBase() {
- return base;
- }
-
- public void setBase(String base) {
- this.base = base;
- }
-
- public ToolingClientLogger getLogger() {
- return logger;
- }
-
- public void setLogger(ToolingClientLogger logger) {
- this.logger = logger;
- this.fhirLoggingInterceptor = new FhirLoggingInterceptor(logger);
- }
-
- public int getRetryCount() {
- return retryCount;
- }
-
- public void setRetryCount(int retryCount) {
- this.retryCount = retryCount;
- }
-
- public long getTimeout() {
- return timeout;
- }
-
- public void setTimeout(long timeout) {
- this.timeout = timeout;
- }
public ResourceRequest issueOptionsRequest(URI optionsUri,
String resourceFormat,
String message,
long timeout) throws IOException {
- this.payload = null;
- Request.Builder request = new Request.Builder()
- .method("OPTIONS", null)
- .url(optionsUri.toURL());
-
- return executeFhirRequest(request, resourceFormat, new Headers.Builder().build(), message, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(optionsUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.OPTIONS);
+ return executeFhirRequest(request, resourceFormat, Collections.emptyList(), message, retryCount, timeout);
}
public ResourceRequest issueGetResourceRequest(URI resourceUri,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
long timeout) throws IOException {
- this.payload = null;
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL());
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
- public int tester(int trytry) {
- return 5;
- }
public ResourceRequest issuePutRequest(URI resourceUri,
byte[] payload,
String resourceFormat,
String message,
long timeout) throws IOException {
- return issuePutRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePutRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePutRequest(URI resourceUri,
byte[] payload,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
long timeout) throws IOException {
if (payload == null) throw new EFhirClientException(0, "PUT requests require a non-null payload");
- this.payload = payload;
- RequestBody body = RequestBody.create(payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .put(body);
+
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.PUT)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
@@ -114,37 +83,38 @@ public class Client {
String resourceFormat,
String message,
long timeout) throws IOException {
- return issuePostRequest(resourceUri, payload, resourceFormat, new Headers.Builder().build(), message, timeout);
+ return issuePostRequest(resourceUri, payload, resourceFormat, Collections.emptyList(), message, timeout);
}
public ResourceRequest issuePostRequest(URI resourceUri,
byte[] payload,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
long timeout) throws IOException {
if (payload == null) throw new EFhirClientException(0, "POST requests require a non-null payload");
- this.payload = payload;
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .post(body);
+
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeFhirRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
public boolean issueDeleteRequest(URI resourceUri) throws IOException {
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .delete();
- return executeFhirRequest(request, null, new Headers.Builder().build(), null, retryCount, timeout).isSuccessfulRequest();
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.DELETE);
+ return executeFhirRequest(request, null, Collections.emptyList(), null, retryCount, timeout).isSuccessfulRequest();
}
public Bundle issueGetFeedRequest(URI resourceUri, String resourceFormat) throws IOException {
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL());
-
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.GET);
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
public Bundle issuePostFeedRequest(URI resourceUri,
@@ -154,57 +124,63 @@ public class Client {
String resourceFormat) throws IOException {
String boundary = "----WebKitFormBoundarykbMUo6H8QaUnYtRy";
byte[] payload = ByteUtils.encodeFormSubmission(parameters, resourceName, resource, boundary);
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .post(body);
- return executeBundleRequest(request, resourceFormat, new Headers.Builder().build(), null, retryCount, timeout);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
+ return executeBundleRequest(request, resourceFormat, Collections.emptyList(), null, retryCount, timeout);
}
public Bundle postBatchRequest(URI resourceUri,
byte[] payload,
String resourceFormat,
- Headers headers,
+ Iterable headers,
String message,
int timeout) throws IOException {
if (payload == null) throw new EFhirClientException(0, "POST requests require a non-null payload");
- RequestBody body = RequestBody.create(MediaType.parse(resourceFormat + ";charset=" + DEFAULT_CHARSET), payload);
- Request.Builder request = new Request.Builder()
- .url(resourceUri.toURL())
- .post(body);
+ HTTPRequest request = new HTTPRequest()
+ .withUrl(resourceUri.toURL())
+ .withMethod(HTTPRequest.HttpMethod.POST)
+ .withBody(payload)
+ .withContentType(getContentTypeWithDefaultCharset(resourceFormat));
return executeBundleRequest(request, resourceFormat, headers, message, retryCount, timeout);
}
- public Bundle executeBundleRequest(Request.Builder request,
- String resourceFormat,
- Headers headers,
- String message,
- int retryCount,
- long timeout) throws IOException {
+ private static String getContentTypeWithDefaultCharset(String resourceFormat) {
+ return resourceFormat + ";charset=" + DEFAULT_CHARSET;
+ }
+
+ public Bundle executeBundleRequest(HTTPRequest request,
+ String resourceFormat,
+ Iterable headers,
+ String message,
+ int retryCount,
+ long timeout) throws IOException {
return new FhirRequestBuilder(request, base)
- .withLogger(fhirLoggingInterceptor)
+ .withLogger(logger)
.withResourceFormat(resourceFormat)
.withRetryCount(retryCount)
.withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS)
.executeAsBatch();
}
- public ResourceRequest executeFhirRequest(Request.Builder request,
- String resourceFormat,
- Headers headers,
- String message,
- int retryCount,
- long timeout) throws IOException {
+ public ResourceRequest executeFhirRequest(HTTPRequest request,
+ String resourceFormat,
+ Iterable headers,
+ String message,
+ int retryCount,
+ long timeout) throws IOException {
return new FhirRequestBuilder(request, base)
- .withLogger(fhirLoggingInterceptor)
+ .withLogger(logger)
.withResourceFormat(resourceFormat)
.withRetryCount(retryCount)
.withMessage(message)
- .withHeaders(headers == null ? new Headers.Builder().build() : headers)
+ .withHeaders(headers == null ? Collections.emptyList() : headers)
.withTimeout(timeout, TimeUnit.MILLISECONDS)
.execute();
}
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/ClientHeaders.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/ClientHeaders.java
index efe62ea30..14e20fa39 100644
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/ClientHeaders.java
+++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/ClientHeaders.java
@@ -1,7 +1,8 @@
package org.hl7.fhir.r5.utils.client.network;
-import okhttp3.internal.http2.Header;
+
import org.hl7.fhir.exceptions.FHIRException;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import java.util.ArrayList;
import java.util.List;
@@ -15,30 +16,31 @@ import java.util.stream.Collectors;
*/
public class ClientHeaders {
- private final ArrayList headers;
+ private final List headers;
public ClientHeaders() {
this.headers = new ArrayList<>();
}
- public ClientHeaders(ArrayList headers) {
- this.headers = headers;
+ public ClientHeaders(List headers) {
+
+ this.headers = new ArrayList<>(headers);
}
- public ArrayList headers() {
+ public List headers() {
return headers;
}
/**
* Add a header to the list of stored headers for network operations.
*
- * @param header {@link Header} to add to the list.
+ * @param header {@link HTTPHeader} to add to the list.
* @throws FHIRException if the header being added is a duplicate.
*/
- public ClientHeaders addHeader(Header header) throws FHIRException {
+ public ClientHeaders addHeader(HTTPHeader header) throws FHIRException {
if (headers.contains(header)) {
- throw new FHIRException("Attempting to add duplicate header, <" + header.name + ", "
- + header.value + ">.");
+ throw new FHIRException("Attempting to add duplicate header, <" + header.getName() + ", "
+ + header.getValue() + ">.");
}
headers.add(header);
return this;
@@ -47,39 +49,39 @@ public class ClientHeaders {
/**
* Add a header to the list of stored headers for network operations.
*
- * @param headerList {@link List} of {@link Header} to add.
+ * @param headerList {@link List} of {@link HTTPHeader} to add.
* @throws FHIRException if any of the headers being added is a duplicate.
*/
- public ClientHeaders addHeaders(List headerList) throws FHIRException {
+ public ClientHeaders addHeaders(List headerList) throws FHIRException {
headerList.forEach(this::addHeader);
return this;
}
/**
* Removes the passed in header from the list of stored headers.
- * @param header {@link Header} to remove from the list.
+ * @param header {@link HTTPHeader} to remove from the list.
* @throws FHIRException if the header passed in does not exist within the stored list.
*/
- public ClientHeaders removeHeader(Header header) throws FHIRException {
+ public ClientHeaders removeHeader(HTTPHeader header) throws FHIRException {
if (!headers.remove(header)) {
- throw new FHIRException("Attempting to remove header, <" + header.name + ", "
- + header.value + ">, from GenericClientHeaders that is not currently stored.");
+ throw new FHIRException("Attempting to remove header, <" + header.getName() + ", "
+ + header.getValue() + ">, from GenericClientHeaders that is not currently stored.");
}
return this;
}
/**
* Removes the passed in headers from the list of stored headers.
- * @param headerList {@link List} of {@link Header} to remove.
+ * @param headerList {@link List} of {@link HTTPHeader} to remove.
* @throws FHIRException if any of the headers passed in does not exist within the stored list.
*/
- public ClientHeaders removeHeaders(List headerList) throws FHIRException {
+ public ClientHeaders removeHeaders(List headerList) throws FHIRException {
headerList.forEach(this::removeHeader);
return this;
}
/**
- * Clears all stored {@link Header}.
+ * Clears all stored {@link HTTPHeader}.
*/
public ClientHeaders clearHeaders() {
headers.clear();
@@ -89,7 +91,7 @@ public class ClientHeaders {
@Override
public String toString() {
return this.headers.stream()
- .map(header -> "\t" + header.name + ":" + header.value)
+ .map(header -> "\t" + header.getName() + ":" + header.getValue())
.collect(Collectors.joining(",\n", "{\n", "\n}"));
}
}
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/FhirLoggingInterceptor.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/FhirLoggingInterceptor.java
deleted file mode 100644
index 1314e6c8a..000000000
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/FhirLoggingInterceptor.java
+++ /dev/null
@@ -1,70 +0,0 @@
-package org.hl7.fhir.r5.utils.client.network;
-
-import okhttp3.*;
-import okio.Buffer;
-
-import org.hl7.fhir.utilities.ToolingClientLogger;
-
-import javax.annotation.Nonnull;
-import java.io.IOException;
-import java.util.ArrayList;
-import java.util.List;
-import java.util.Map;
-
-public class FhirLoggingInterceptor implements Interceptor {
-
- private ToolingClientLogger logger;
-
- public FhirLoggingInterceptor(ToolingClientLogger logger) {
- this.logger = logger;
- }
-
- public FhirLoggingInterceptor setLogger(ToolingClientLogger logger) {
- this.logger = logger;
- return this;
- }
-
- @Override
- public Response intercept(@Nonnull Interceptor.Chain chain) throws IOException {
- // Log Request
- Request request = chain.request();
- List hdrs = new ArrayList<>();
- for (String s : request.headers().toString().split("\\n")) {
- hdrs.add(s.trim());
- }
- byte[] cnt = null;
- if (request.body() != null) {
- Buffer buf = new Buffer();
- request.body().writeTo(buf);
- cnt = buf.readByteArray();
- }
- if (logger != null) {
- logger.logRequest(request.method(), request.url().toString(), hdrs, cnt);
- }
-
- // Log Response
- Response response = null;
- response = chain.proceed(chain.request());
-
- MediaType contentType = null;
- byte[] bodyBytes = null;
- if (response.body() != null) {
- contentType = response.body().contentType();
- bodyBytes = response.body().bytes();
- }
-
- // Get Headers as List
- List headerList = new ArrayList<>();
- Map> headerMap = response.headers().toMultimap();
- headerMap.keySet().forEach(key -> headerMap.get(key).forEach(value -> headerList.add(key + ":" + value)));
-
- if (logger != null) {
- long responseTimeInMillis = response.receivedResponseAtMillis() - response.sentRequestAtMillis();
- logger.logResponse(Integer.toString(response.code()), headerList, bodyBytes, responseTimeInMillis);
- }
-
- // Reading byte[] clears body. Need to recreate.
- ResponseBody body = ResponseBody.create(bodyBytes, contentType);
- return response.newBuilder().body(body).build();
- }
-}
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilder.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilder.java
index 49ee7d881..750bce23f 100644
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilder.java
+++ b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilder.java
@@ -1,14 +1,12 @@
package org.hl7.fhir.r5.utils.client.network;
import java.io.IOException;
+import java.util.ArrayList;
import java.util.List;
-import java.util.Map;
import java.util.concurrent.TimeUnit;
-import javax.annotation.Nonnull;
import org.apache.commons.lang3.StringUtils;
-import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.r5.formats.IParser;
import org.hl7.fhir.r5.formats.JsonParser;
import org.hl7.fhir.r5.formats.XmlParser;
@@ -20,32 +18,20 @@ import org.hl7.fhir.r5.utils.ResourceUtilities;
import org.hl7.fhir.r5.utils.client.EFhirClientException;
import org.hl7.fhir.r5.utils.client.ResourceFormat;
import org.hl7.fhir.utilities.MimeType;
-import org.hl7.fhir.utilities.Utilities;
-import org.hl7.fhir.utilities.settings.FhirSettings;
+import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.hl7.fhir.utilities.http.*;
import org.hl7.fhir.utilities.xhtml.XhtmlUtils;
-import okhttp3.Authenticator;
-import okhttp3.Credentials;
-import okhttp3.Headers;
-import okhttp3.OkHttpClient;
-import okhttp3.Request;
-import okhttp3.Response;
-
public class FhirRequestBuilder {
- protected static final String HTTP_PROXY_USER = "http.proxyUser";
- protected static final String HTTP_PROXY_PASS = "http.proxyPassword";
- protected static final String HEADER_PROXY_AUTH = "Proxy-Authorization";
+ protected static final String DEFAULT_CHARSET = "UTF-8";
+
protected static final String LOCATION_HEADER = "location";
protected static final String CONTENT_LOCATION_HEADER = "content-location";
- protected static final String DEFAULT_CHARSET = "UTF-8";
- /**
- * The singleton instance of the HttpClient, used for all requests.
- */
- private static OkHttpClient okHttpClient;
- private final Request.Builder httpRequest;
+
+ private final HTTPRequest httpRequest;
private String resourceFormat = null;
- private Headers headers = null;
+ private Iterable headers = null;
private String message = null;
private int retryCount = 1;
/**
@@ -58,69 +44,57 @@ public class FhirRequestBuilder {
private TimeUnit timeoutUnit = TimeUnit.MILLISECONDS;
/**
- * {@link FhirLoggingInterceptor} for log output.
+ * {@link ToolingClientLogger} for log output.
*/
- private FhirLoggingInterceptor logger = null;
+ private ToolingClientLogger logger = null;
+
private String source;
- public FhirRequestBuilder(Request.Builder httpRequest, String source) {
- this.httpRequest = httpRequest;
+ public FhirRequestBuilder(HTTPRequest httpRequest, String source) {
this.source = source;
+ this.httpRequest = httpRequest;
}
/**
- * Adds necessary default headers, formatting headers, and any passed in {@link Headers} to the passed in
- * {@link okhttp3.Request.Builder}
+ * Adds necessary default headers, formatting headers, and any passed in {@link HTTPHeader}s to the passed in
+ * {@link HTTPRequest}
*
- * @param request {@link okhttp3.Request.Builder} to add headers to.
+ * @param request {@link HTTPRequest} to add headers to.
* @param format Expected {@link Resource} format.
- * @param headers Any additional {@link Headers} to add to the request.
+ * @param headers Any additional {@link HTTPHeader}s to add to the request.
*/
- protected static void formatHeaders(Request.Builder request, String format, Headers headers) {
- addDefaultHeaders(request, headers);
- if (format != null) addResourceFormatHeaders(request, format);
- if (headers != null) addHeaders(request, headers);
- }
+ protected static HTTPRequest formatHeaders(HTTPRequest request, String format, Iterable headers) {
+ List allHeaders = new ArrayList<>();
+ request.getHeaders().forEach(allHeaders::add);
- /**
- * Adds necessary headers for all REST requests.
- * User-Agent : hapi-fhir-tooling-client
- *
- * @param request {@link Request.Builder} to add default headers to.
- */
- protected static void addDefaultHeaders(Request.Builder request, Headers headers) {
- if (headers == null || !headers.names().contains("User-Agent")) {
- request.addHeader("User-Agent", "hapi-fhir-tooling-client");
- }
+ if (format != null) getResourceFormatHeaders(request, format).forEach(allHeaders::add);
+ if (headers != null) headers.forEach(allHeaders::add);
+ return request.withHeaders(allHeaders);
}
/**
* Adds necessary headers for the given resource format provided.
*
- * @param request {@link Request.Builder} to add default headers to.
+ * @param httpRequest {@link HTTPRequest} to add default headers to.
+ * @param format Expected {@link Resource} format.
*/
- protected static void addResourceFormatHeaders(Request.Builder request, String format) {
- request.addHeader("Accept", format);
- if (Utilities.existsInList(request.getMethod$okhttp(), "POST", "PUT")) {
- request.addHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET);
+ protected static Iterable getResourceFormatHeaders(HTTPRequest httpRequest, String format) {
+ List headers = new ArrayList<>();
+ headers.add(new HTTPHeader("Accept", format));
+ if (httpRequest.getMethod() == HTTPRequest.HttpMethod.PUT
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.POST
+ || httpRequest.getMethod() == HTTPRequest.HttpMethod.PATCH
+ ) {
+ headers.add( new HTTPHeader("Content-Type", format + ";charset=" + DEFAULT_CHARSET));
}
+ return headers;
}
/**
- * Iterates through the passed in {@link Headers} and adds them to the provided {@link Request.Builder}.
- *
- * @param request {@link Request.Builder} to add headers to.
- * @param headers {@link Headers} to add to request.
- */
- protected static void addHeaders(Request.Builder request, Headers headers) {
- headers.forEach(header -> request.addHeader(header.getFirst(), header.getSecond()));
- }
-
- /**
- * Returns true if any of the {@link org.hl7.fhir.r5.model.OperationOutcome.OperationOutcomeIssueComponent} within the
- * provided {@link OperationOutcome} have an {@link org.hl7.fhir.r5.model.OperationOutcome.IssueSeverity} of
- * {@link org.hl7.fhir.r5.model.OperationOutcome.IssueSeverity#ERROR} or
- * {@link org.hl7.fhir.r5.model.OperationOutcome.IssueSeverity#FATAL}
+ * Returns true if any of the {@link OperationOutcome.OperationOutcomeIssueComponent} within the
+ * provided {@link OperationOutcome} have an {@link OperationOutcome.IssueSeverity} of
+ * {@link OperationOutcome.IssueSeverity#ERROR} or
+ * {@link OperationOutcome.IssueSeverity#FATAL}
*
* @param oo {@link OperationOutcome} to evaluate.
* @return {@link Boolean#TRUE} if an error exists.
@@ -131,71 +105,8 @@ public class FhirRequestBuilder {
|| issue.getSeverity() == OperationOutcome.IssueSeverity.FATAL));
}
- /**
- * Extracts the 'location' header from the passes in {@link Headers}. If no value for 'location' exists, the
- * value for 'content-location' is returned. If neither header exists, we return null.
- *
- * @param headers {@link Headers} to evaluate
- * @return {@link String} header value, or null if no location headers are set.
- */
- protected static String getLocationHeader(Headers headers) {
- Map> headerMap = headers.toMultimap();
- if (headerMap.containsKey(LOCATION_HEADER)) {
- return headerMap.get(LOCATION_HEADER).get(0);
- } else if (headerMap.containsKey(CONTENT_LOCATION_HEADER)) {
- return headerMap.get(CONTENT_LOCATION_HEADER).get(0);
- } else {
- return null;
- }
- }
-
- /**
- * We only ever want to have one copy of the HttpClient kicking around at any given time. If we need to make changes
- * to any configuration, such as proxy settings, timeout, caches, etc, we can do a per-call configuration through
- * the {@link OkHttpClient#newBuilder()} method. That will return a builder that shares the same connection pool,
- * dispatcher, and configuration with the original client.
- *
- * The {@link OkHttpClient} uses the proxy auth properties set in the current system properties. The reason we don't
- * set the proxy address and authentication explicitly, is due to the fact that this class is often used in conjunction
- * with other http client tools which rely on the system.properties settings to determine proxy settings. It's easier
- * to keep the method consistent across the board. ...for now.
- *
- * @return {@link OkHttpClient} instance
- */
- protected OkHttpClient getHttpClient() {
- if (FhirSettings.isProhibitNetworkAccess()) {
- throw new FHIRException("Network Access is prohibited in this context");
- }
-
- if (okHttpClient == null) {
- okHttpClient = new OkHttpClient();
- }
-
- Authenticator proxyAuthenticator = getAuthenticator();
-
- OkHttpClient.Builder builder = okHttpClient.newBuilder();
- if (logger != null) builder.addInterceptor(logger);
- builder.addInterceptor(new RetryInterceptor(retryCount));
- return builder.connectTimeout(timeout, timeoutUnit)
- .writeTimeout(timeout, timeoutUnit)
- .readTimeout(timeout, timeoutUnit)
- .proxyAuthenticator(proxyAuthenticator)
- .build();
- }
-
- @Nonnull
- private static Authenticator getAuthenticator() {
- return (route, response) -> {
- final String httpProxyUser = System.getProperty(HTTP_PROXY_USER);
- final String httpProxyPass = System.getProperty(HTTP_PROXY_PASS);
- if (httpProxyUser != null && httpProxyPass != null) {
- String credential = Credentials.basic(httpProxyUser, httpProxyPass);
- return response.request().newBuilder()
- .header(HEADER_PROXY_AUTH, credential)
- .build();
- }
- return response.request().newBuilder().build();
- };
+ protected ManagedFhirWebAccessor getManagedWebAccessor() {
+ return ManagedWebAccess.fhirAccessor().withRetries(retryCount).withTimeout(timeout, timeoutUnit).withLogger(logger);
}
public FhirRequestBuilder withResourceFormat(String resourceFormat) {
@@ -203,7 +114,7 @@ public class FhirRequestBuilder {
return this;
}
- public FhirRequestBuilder withHeaders(Headers headers) {
+ public FhirRequestBuilder withHeaders(Iterable headers) {
this.headers = headers;
return this;
}
@@ -218,7 +129,7 @@ public class FhirRequestBuilder {
return this;
}
- public FhirRequestBuilder withLogger(FhirLoggingInterceptor logger) {
+ public FhirRequestBuilder withLogger(ToolingClientLogger logger) {
this.logger = logger;
return this;
}
@@ -229,20 +140,16 @@ public class FhirRequestBuilder {
return this;
}
- protected Request buildRequest() {
- return httpRequest.build();
- }
-
public ResourceRequest execute() throws IOException {
- formatHeaders(httpRequest, resourceFormat, headers);
- Response response = getHttpClient().newCall(httpRequest.build()).execute();
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, headers);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);//getHttpClient().newCall(httpRequest.build()).execute();
T resource = unmarshalReference(response, resourceFormat, null);
- return new ResourceRequest(resource, response.code(), getLocationHeader(response.headers()));
+ return new ResourceRequest(resource, response.getCode(), getLocationHeader(response.getHeaders()));
}
public Bundle executeAsBatch() throws IOException {
- formatHeaders(httpRequest, resourceFormat, null);
- Response response = getHttpClient().newCall(httpRequest.build()).execute();
+ HTTPRequest requestWithHeaders = formatHeaders(httpRequest, resourceFormat, null);
+ HTTPResult response = getManagedWebAccessor().httpCall(requestWithHeaders);
return unmarshalFeed(response, resourceFormat);
}
@@ -250,12 +157,12 @@ public class FhirRequestBuilder {
* Unmarshalls a resource from the response stream.
*/
@SuppressWarnings("unchecked")
- protected T unmarshalReference(Response response, String format, String resourceType) {
- int code = response.code();
+ protected T unmarshalReference(HTTPResult response, String format, String resourceType) {
+ int code = response.getCode();
boolean ok = code >= 200 && code < 300;
- if (response.body() == null) {
+ if (response.getContent() == null) {
if (!ok) {
- throw new EFhirClientException(code, response.message());
+ throw new EFhirClientException(code, response.getMessage());
} else {
return null;
}
@@ -264,9 +171,9 @@ public class FhirRequestBuilder {
Resource resource = null;
try {
- body = response.body().string();
- String ct = response.header("Content-Type");
- if (ct == null) {
+ body = response.getContentAsString();
+ String contentType = HTTPHeaderUtil.getSingleHeader(response.getHeaders(), "Content-Type");
+ if (contentType == null) {
if (ok) {
resource = getParser(format).parse(body);
} else {
@@ -275,10 +182,10 @@ public class FhirRequestBuilder {
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
}
} else {
- if (ct.contains(";")) {
- ct = ct.substring(0, ct.indexOf(";"));
+ if (contentType.contains(";")) {
+ contentType = contentType.substring(0, contentType.indexOf(";"));
}
- switch (ct) {
+ switch (contentType) {
case "application/json":
case "application/fhir+json":
if (!format.contains("json")) {
@@ -298,10 +205,10 @@ public class FhirRequestBuilder {
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
break;
case "text/html" :
- resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.body().string(), source));
+ resource = OperationOutcomeUtilities.outcomeFromTextError(XhtmlUtils.convertHtmlToText(response.getContentAsString(), source));
break;
default: // not sure what else to do?
- System.out.println("Got content-type '"+ct+"' from "+source);
+ System.out.println("Got content-type '"+contentType+"' from "+source);
System.out.println(body);
resource = OperationOutcomeUtilities.outcomeFromTextError(body);
}
@@ -336,14 +243,14 @@ public class FhirRequestBuilder {
/**
* Unmarshalls Bundle from response stream.
*/
- protected Bundle unmarshalFeed(Response response, String format) {
+ protected Bundle unmarshalFeed(HTTPResult response, String format) {
return unmarshalReference(response, format, "Bundle");
}
/**
* Returns the appropriate parser based on the format type passed in. Defaults to XML parser if a blank format is
* provided...because reasons.
- *
+ *
* Currently supports only "json" and "xml" formats.
*
* @param format One of "json" or "xml".
@@ -362,4 +269,19 @@ public class FhirRequestBuilder {
throw new EFhirClientException(0, "Invalid format: " + format);
}
}
+
+ /**
+ * Extracts the 'location' header from the passed headers. If no value for 'location' exists, the
+ * value for 'content-location' is returned. If neither header exists, we return null.
+ *
+ * @param headers Headers to search for 'location' or 'content-location'.
+ */
+ protected static String getLocationHeader(Iterable headers) {
+ String locationHeader = HTTPHeaderUtil.getSingleHeader(headers, LOCATION_HEADER);
+
+ if (locationHeader != null) {
+ return locationHeader;
+ }
+ return HTTPHeaderUtil.getSingleHeader(headers, CONTENT_LOCATION_HEADER);
+ }
}
diff --git a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/RetryInterceptor.java b/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/RetryInterceptor.java
deleted file mode 100644
index d6f4e2de1..000000000
--- a/org.hl7.fhir.r5/src/main/java/org/hl7/fhir/r5/utils/client/network/RetryInterceptor.java
+++ /dev/null
@@ -1,63 +0,0 @@
-package org.hl7.fhir.r5.utils.client.network;
-
-import okhttp3.Interceptor;
-import okhttp3.Request;
-import okhttp3.Response;
-import javax.annotation.Nonnull;
-
-import java.io.IOException;
-
-/**
- * An {@link Interceptor} for {@link okhttp3.OkHttpClient} that controls the number of times we retry a to execute a
- * given request, before reporting a failure. This includes unsuccessful return codes and timeouts.
- */
-public class RetryInterceptor implements Interceptor {
-
- // Delay between retying failed requests, in millis
- private final long RETRY_TIME = 2000;
-
- // Maximum number of times to retry the request before failing
- private final int maxRetry;
-
- // Internal counter for tracking the number of times we've tried this request
- private int retryCounter = 0;
-
- public RetryInterceptor(int maxRetry) {
- this.maxRetry = maxRetry;
- }
-
- @Override
- public Response intercept(Interceptor.Chain chain) throws IOException {
- Request request = chain.request();
- Response response = null;
-
- do {
- try {
- // If we are retrying a failed request that failed due to a bad response from the server, we must close it first
- if (response != null) {
-// System.out.println("Previous " + chain.request().method() + " attempt returned HTTP<" + (response.code())
-// + "> from url -> " + chain.request().url() + ".");
- response.close();
- }
- // System.out.println(chain.request().method() + " attempt <" + (retryCounter + 1) + "> to url -> " + chain.request().url());
- response = chain.proceed(request);
- } catch (IOException e) {
- try {
- // Include a small break in between requests.
- Thread.sleep(RETRY_TIME);
- } catch (InterruptedException e1) {
- System.out.println(chain.request().method() + " to url -> " + chain.request().url() + " interrupted on try <" + retryCounter + ">");
- }
- } finally {
- retryCounter++;
- }
- } while ((response == null || !response.isSuccessful()) && (retryCounter <= maxRetry + 1));
-
- /*
- * if something has gone wrong, and we are unable to complete the request, we still need to initialize the return
- * response so we don't get a null pointer exception.
- */
- return response != null ? response : chain.proceed(request);
- }
-
-}
\ No newline at end of file
diff --git a/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/FHIRToolingClientTest.java b/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/FHIRToolingClientTest.java
index 6987d1fc0..caa44aa5e 100644
--- a/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/FHIRToolingClientTest.java
+++ b/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/FHIRToolingClientTest.java
@@ -5,6 +5,7 @@ import java.net.URI;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Arrays;
+import java.util.List;
import org.hl7.fhir.exceptions.FHIRException;
import org.hl7.fhir.exceptions.FHIRFormatError;
@@ -22,17 +23,14 @@ import org.hl7.fhir.r5.model.TerminologyCapabilities;
import org.hl7.fhir.r5.utils.client.network.Client;
import org.hl7.fhir.r5.utils.client.network.ResourceRequest;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import org.hl7.fhir.utilities.settings.FhirSettings;
-import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
-import org.mockito.ArgumentCaptor;
-import org.mockito.ArgumentMatchers;
-import org.mockito.Mockito;
+import org.mockito.*;
import okhttp3.Headers;
-import okhttp3.Request;
-import okhttp3.internal.http2.Header;
import static org.junit.jupiter.api.Assertions.*;
import static org.mockito.Mockito.times;
@@ -41,46 +39,56 @@ class FHIRToolingClientTest {
String TX_ADDR = FhirSettings.getTxFhirDevelopment();
- Header h1 = new Header("header1", "value1");
- Header h2 = new Header("header2", "value2");
- Header h3 = new Header("header3", "value3");
+ HTTPHeader h1 = new HTTPHeader("header1", "value1");
+ HTTPHeader h2 = new HTTPHeader("header2", "value2");
+ HTTPHeader h3 = new HTTPHeader("header3", "value3");
+
+ HTTPHeader agentHeader = new HTTPHeader("User-Agent", "fhir/test-cases");
private Client mockClient;
private FHIRToolingClient toolingClient;
+ @Captor
+ private ArgumentCaptor> headersArgumentCaptor;
+
@BeforeEach
void setUp() throws IOException, URISyntaxException {
+ MockitoAnnotations.openMocks(this);
mockClient = Mockito.mock(Client.class);
ResourceRequest resourceResourceRequest = new ResourceRequest<>(generateBundle(), 200, "");
//GET
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyLong()))
+ Mockito.any(Iterable.class), Mockito.anyString(), Mockito.anyLong()))
.thenReturn(resourceResourceRequest);
//PUT
Mockito.when(mockClient.issuePutRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong()))
.thenReturn(resourceResourceRequest);
//POST
Mockito.when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyLong()))
.thenReturn(resourceResourceRequest);
Mockito.when(mockClient.issuePostRequest(Mockito.any(URI.class), Mockito.any(byte[].class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.contains("validate"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.contains("validate"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new OperationOutcome(), 200, "location"));
//BUNDLE REQ
- Mockito.when(mockClient.executeBundleRequest(Mockito.any(Request.Builder.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.anyString(), Mockito.anyInt(), Mockito.anyLong()))
+ Mockito.when(mockClient.executeBundleRequest(Mockito.any(HTTPRequest.class), Mockito.anyString(),
+ ArgumentMatchers.any(), Mockito.anyString(), Mockito.anyInt(), Mockito.anyLong()))
.thenReturn(generateBundle());
toolingClient = new FHIRToolingClient(TX_ADDR, "fhir/test-cases");
toolingClient.setClient(mockClient);
}
- private ArrayList getHeaders() {
+ private List getHeaders() {
return new ArrayList<>(Arrays.asList(h1, h2, h3));
}
+ private List getHeadersWithAgent() {
+ return new ArrayList<>(Arrays.asList(h1, h2, h3, agentHeader));
+ }
+
private Bundle generateBundle() {
Patient patient = generatePatient();
Observation observation = generateObservation();
@@ -140,36 +148,37 @@ class FHIRToolingClientTest {
return observation;
}
- private void checkHeaders(Headers argumentCaptorValue) {
- getHeaders().forEach(header -> {
-// System.out.println("Checking header <" + header.component1().utf8() + ", " + header.component2().utf8() + ">");
- Assertions.assertEquals(argumentCaptorValue.get(header.component1().utf8()), header.component2().utf8());
+ private void checkHeaders(Iterable argumentCaptorValue) {
+ List capturedHeaders = new ArrayList<>();
+ argumentCaptorValue.forEach(capturedHeaders::add);
+
+ getHeadersWithAgent().forEach(header -> {
+ assertTrue(capturedHeaders.contains(header));
});
}
@Test
void getTerminologyCapabilities() throws IOException {
+
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new TerminologyCapabilities(), 200, "location"));
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getTerminologyCapabilities();
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getTerminologyCapabilitiesNotSupported() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
Exception exception = assertThrows(FHIRException.class, () -> {
toolingClient.getTerminologyCapabilities();
@@ -180,24 +189,23 @@ class FHIRToolingClientTest {
@Test
void getTerminologyCapabilitiesFailsForJSON() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
.thenThrow(new FHIRFormatError("dummy error"))
.thenReturn(new ResourceRequest<>(new TerminologyCapabilities(), 200, "location"));
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getTerminologyCapabilities();
Mockito.verify(mockClient, times(2)).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getTerminologyCapabilitiesStatementFailsForJSONandXML() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("TerminologyCapabilities"), Mockito.anyLong()))
.thenThrow(new FHIRFormatError("dummy error"))
.thenThrow(new FHIRFormatError("dummy error 2"));
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
@@ -210,10 +218,9 @@ class FHIRToolingClientTest {
@Test
void getCapabilitiesStatement() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
toolingClient.setClientHeaders(getHeaders());
@@ -222,18 +229,17 @@ class FHIRToolingClientTest {
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getCapabilitiesStatementFailsForJSON() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
.thenThrow(new FHIRFormatError("dummy error"))
.thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
toolingClient.setClientHeaders(getHeaders());
toolingClient.getCapabilitiesStatement();
@@ -241,14 +247,14 @@ class FHIRToolingClientTest {
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
assertEquals(ResourceFormat.RESOURCE_XML.getHeader(), toolingClient.getPreferredResourceFormat());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getCapabilitiesStatementFailsForJSONandXML() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement"), Mockito.anyLong()))
.thenThrow(new FHIRFormatError("dummy error"))
.thenThrow(new FHIRFormatError("dummy error 2"));
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
@@ -261,17 +267,16 @@ class FHIRToolingClientTest {
@Test
void getCapabilitiesStatementQuick() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
.thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getCapabilitiesStatementQuick();
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
@@ -280,19 +285,18 @@ class FHIRToolingClientTest {
@Test
void getCapabilitiesStatementQuickFailsForJSON() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
.thenThrow(new FHIRFormatError("dummy error"))
.thenReturn(new ResourceRequest<>(new CapabilityStatement(), 200, "location"));
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getCapabilitiesStatementQuick();
Mockito.verify(mockClient, times(2)).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
assertEquals(ResourceFormat.RESOURCE_XML.getHeader(), toolingClient.getPreferredResourceFormat());
@@ -301,11 +305,10 @@ class FHIRToolingClientTest {
@Test
void getCapabilitiesStatementQuickFailsForJSONandXML() throws IOException {
Mockito.when(mockClient.issueGetResourceRequest(Mockito.any(URI.class), Mockito.anyString(),
- Mockito.any(Headers.class), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
+ ArgumentMatchers.any(), Mockito.eq("CapabilitiesStatement-Quick"), Mockito.anyLong()))
.thenThrow(new FHIRFormatError("dummy error"))
.thenThrow(new FHIRFormatError("dummy error 2"));
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
assertEquals(ResourceFormat.RESOURCE_JSON.getHeader(), toolingClient.getPreferredResourceFormat());
Exception exception = assertThrows(FHIRException.class, () -> {
@@ -317,63 +320,58 @@ class FHIRToolingClientTest {
@Test
void read() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.read(Patient.class, "id");
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void vread() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.vread(Patient.class, "id", "version");
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void getCanonical() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.getCanonical(Patient.class, "canonicalURL");
Mockito.verify(mockClient).issueGetResourceRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.anyString(),
headersArgumentCaptor.capture(), ArgumentMatchers.anyString(), ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void update() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.update(generatePatient());
Mockito.verify(mockClient).issuePutRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
@Test
void validate() throws IOException {
- ArgumentCaptor headersArgumentCaptor = ArgumentCaptor.forClass(Headers.class);
toolingClient.setClientHeaders(getHeaders());
toolingClient.validate(Patient.class, generatePatient(), "id");
Mockito.verify(mockClient).issuePostRequest(ArgumentMatchers.any(URI.class), ArgumentMatchers.any(byte[].class),
ArgumentMatchers.anyString(), headersArgumentCaptor.capture(), ArgumentMatchers.anyString(),
ArgumentMatchers.anyLong());
- Headers argumentCaptorValue = headersArgumentCaptor.getValue();
+ Iterable argumentCaptorValue = headersArgumentCaptor.getValue();
checkHeaders(argumentCaptorValue);
}
}
\ No newline at end of file
diff --git a/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/ClientHeadersTest.java b/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/ClientHeadersTest.java
index 7f102bf73..2dd985409 100644
--- a/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/ClientHeadersTest.java
+++ b/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/ClientHeadersTest.java
@@ -4,20 +4,19 @@ import java.util.Arrays;
import java.util.List;
import org.hl7.fhir.exceptions.FHIRException;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
-import okhttp3.internal.http2.Header;
-
class ClientHeadersTest {
ClientHeaders clientHeaders;
- Header h1 = new Header("header1", "value1");
- Header h2 = new Header("header2", "value2");
- Header h3 = new Header("header3", "value3");
+ HTTPHeader h1 = new HTTPHeader("header1", "value1");
+ HTTPHeader h2 = new HTTPHeader("header2", "value2");
+ HTTPHeader h3 = new HTTPHeader("header3", "value3");
@BeforeEach
void setUp() {
@@ -43,7 +42,7 @@ class ClientHeadersTest {
@Test
@DisplayName("Happy path add headers as list.")
void addHeaders() {
- List headersList = Arrays.asList(h1, h2, h3);
+ List headersList = Arrays.asList(h1, h2, h3);
clientHeaders.addHeaders(headersList);
Assertions.assertEquals(3, clientHeaders.headers().size());
Assertions.assertEquals(headersList, clientHeaders.headers());
@@ -52,7 +51,7 @@ class ClientHeadersTest {
@Test
@DisplayName("Happy path add headers as list.")
void addHeadersDuplicateAdd() {
- List headersList = Arrays.asList(h1, h2, h1);
+ List headersList = Arrays.asList(h1, h2, h1);
Assertions.assertThrows(FHIRException.class, () -> clientHeaders.addHeaders(headersList));
}
@@ -64,7 +63,7 @@ class ClientHeadersTest {
clientHeaders.addHeader(h3);
clientHeaders.removeHeader(h2);
Assertions.assertEquals(2, clientHeaders.headers().size());
- clientHeaders.removeHeader(new Header("header3", "value3"));
+ clientHeaders.removeHeader(new HTTPHeader("header3", "value3"));
Assertions.assertEquals(1, clientHeaders.headers().size());
}
@@ -79,8 +78,8 @@ class ClientHeadersTest {
@Test
@DisplayName("Happy path remove list of existing headers.")
void removeHeaders() {
- List headersToAddList = Arrays.asList(h1, h2, h3);
- List headersToRemoveList = Arrays.asList(h2, h3);
+ List headersToAddList = Arrays.asList(h1, h2, h3);
+ List headersToRemoveList = Arrays.asList(h2, h3);
clientHeaders.addHeaders(headersToAddList);
clientHeaders.removeHeaders(headersToRemoveList);
Assertions.assertEquals(1, clientHeaders.headers().size());
@@ -89,15 +88,15 @@ class ClientHeadersTest {
@Test
@DisplayName("Remove list containing unknown header.")
void removeHeadersUnknown() {
- List headersToAddList = Arrays.asList(h1, h3);
- List headersToRemoveList = Arrays.asList(h2, h3);
+ List headersToAddList = Arrays.asList(h1, h3);
+ List headersToRemoveList = Arrays.asList(h2, h3);
clientHeaders.addHeaders(headersToAddList);
Assertions.assertThrows(FHIRException.class, () -> clientHeaders.removeHeaders(headersToRemoveList));
}
@Test
void clearHeaders() {
- List headersToAddList = Arrays.asList(h1, h3);
+ List headersToAddList = Arrays.asList(h1, h3);
clientHeaders.addHeaders(headersToAddList);
Assertions.assertEquals(2, clientHeaders.headers().size());
clientHeaders.clearHeaders();
diff --git a/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilderTest.java b/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilderTest.java
index 7be6d672a..415edf1fd 100644
--- a/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilderTest.java
+++ b/org.hl7.fhir.r5/src/test/java/org/hl7/fhir/r5/utils/client/network/FhirRequestBuilderTest.java
@@ -1,39 +1,29 @@
package org.hl7.fhir.r5.utils.client.network;
+import java.util.Collections;
import java.util.List;
import java.util.Map;
import org.hl7.fhir.r5.model.OperationOutcome;
+import org.hl7.fhir.utilities.http.HTTPHeaderUtil;
+import org.hl7.fhir.utilities.http.HTTPRequest;
+import org.hl7.fhir.utilities.http.HTTPHeader;
+import org.hl7.fhir.utilities.http.HTTPResult;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.DisplayName;
import org.junit.jupiter.api.Test;
-import okhttp3.Headers;
-import okhttp3.Request;
-
class FhirRequestBuilderTest {
- @Test
- @DisplayName("Test default headers are added correctly.")
- void addDefaultHeaders() {
- Request.Builder request = new Request.Builder().url("http://www.google.com");
- FhirRequestBuilder.addDefaultHeaders(request, null);
-
- Map> headersMap = request.build().headers().toMultimap();
- Assertions.assertNotNull(headersMap.get("User-Agent"), "User-Agent header null.");
- Assertions.assertEquals("hapi-fhir-tooling-client", headersMap.get("User-Agent").get(0),
- "User-Agent header not populated with expected value \"hapi-fhir-tooling-client\".");
- }
-
@Test
@DisplayName("Test resource format headers are added correctly (GET).")
void addResourceFormatHeadersGET() {
String testFormat = "yaml";
- Request.Builder request = new Request.Builder().url("http://www.google.com");
- request.setMethod$okhttp("GET");
- FhirRequestBuilder.addResourceFormatHeaders(request, testFormat);
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.GET);
- Map> headersMap = request.build().headers().toMultimap();
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
"Accept header not populated with expected value " + testFormat + ".");
@@ -45,11 +35,11 @@ class FhirRequestBuilderTest {
@DisplayName("Test resource format headers are added correctly (POST).")
void addResourceFormatHeadersPOST() {
String testFormat = "yaml";
- Request.Builder request = new Request.Builder().url("http://www.google.com");
- request.setMethod$okhttp("POST");
- FhirRequestBuilder.addResourceFormatHeaders(request, testFormat);
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com").withMethod(HTTPRequest.HttpMethod.POST);
- Map> headersMap = request.build().headers().toMultimap();
+ Iterable headers = FhirRequestBuilder.getResourceFormatHeaders(request, testFormat);
+
+ Map> headersMap = HTTPHeaderUtil.getMultimap(headers);
Assertions.assertNotNull(headersMap.get("Accept"), "Accept header null.");
Assertions.assertEquals(testFormat, headersMap.get("Accept").get(0),
"Accept header not populated with expected value " + testFormat + ".");
@@ -59,31 +49,6 @@ class FhirRequestBuilderTest {
"Content-Type header not populated with expected value \"" + testFormat + ";charset=" + FhirRequestBuilder.DEFAULT_CHARSET + "\".");
}
- @Test
- @DisplayName("Test a list of provided headers are added correctly.")
- void addHeaders() {
- String headerName1 = "headerName1";
- String headerValue1 = "headerValue1";
- String headerName2 = "headerName2";
- String headerValue2 = "headerValue2";
-
- Headers headers = new Headers.Builder()
- .add(headerName1, headerValue1)
- .add(headerName2, headerValue2)
- .build();
-
- Request.Builder request = new Request.Builder().url("http://www.google.com");
- FhirRequestBuilder.addHeaders(request, headers);
-
- Map> headersMap = request.build().headers().toMultimap();
- Assertions.assertNotNull(headersMap.get(headerName1), headerName1 + " header null.");
- Assertions.assertEquals(headerValue1, headersMap.get(headerName1).get(0),
- headerName1 + " header not populated with expected value " + headerValue1 + ".");
- Assertions.assertNotNull(headersMap.get(headerName2), headerName2 + " header null.");
- Assertions.assertEquals(headerValue2, headersMap.get(headerName2).get(0),
- headerName2 + " header not populated with expected value " + headerValue2 + ".");
- }
-
@Test
@DisplayName("Test that FATAL issue severity triggers error.")
void hasErrorTestFatal() {
@@ -120,19 +85,22 @@ class FhirRequestBuilderTest {
@DisplayName("Test that getLocationHeader returns header for 'location'.")
void getLocationHeaderWhenOnlyLocationIsSet() {
final String expectedLocationHeader = "location_header_value";
- Headers headers = new Headers.Builder()
- .add(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader)
- .build();
- Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
+ HTTPResult result = new HTTPResult("source",
+ 200,
+ "message",
+ "contentType",
+ new byte[0],
+ List.of(new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader)));
+
+ Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(result.getHeaders()));
}
@Test
@DisplayName("Test that getLocationHeader returns header for 'content-location'.")
void getLocationHeaderWhenOnlyContentLocationIsSet() {
final String expectedContentLocationHeader = "content_location_header_value";
- Headers headers = new Headers.Builder()
- .add(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader)
- .build();
+ Iterable headers = List.of(new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader));
+
Assertions.assertEquals(expectedContentLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
}
@@ -141,18 +109,18 @@ class FhirRequestBuilderTest {
void getLocationHeaderWhenLocationAndContentLocationAreSet() {
final String expectedLocationHeader = "location_header_value";
final String expectedContentLocationHeader = "content_location_header_value";
- Headers headers = new Headers.Builder()
- .add(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader)
- .add(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader)
- .build();
+
+ Iterable headers = List.of(
+ new HTTPHeader(FhirRequestBuilder.LOCATION_HEADER, expectedLocationHeader),
+ new HTTPHeader(FhirRequestBuilder.CONTENT_LOCATION_HEADER, expectedContentLocationHeader)
+ );
+
Assertions.assertEquals(expectedLocationHeader, FhirRequestBuilder.getLocationHeader(headers));
}
@Test
@DisplayName("Test that getLocationHeader returns null when no location available.")
void getLocationHeaderWhenNoLocationSet() {
- Headers headers = new Headers.Builder()
- .build();
- Assertions.assertNull(FhirRequestBuilder.getLocationHeader(headers));
+ Assertions.assertNull(FhirRequestBuilder.getLocationHeader(Collections.emptyList()));
}
}
\ No newline at end of file
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPHeader.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPHeader.java
new file mode 100644
index 000000000..b45e9ff7f
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPHeader.java
@@ -0,0 +1,21 @@
+package org.hl7.fhir.utilities.http;
+
+import lombok.AllArgsConstructor;
+import lombok.EqualsAndHashCode;
+import lombok.Getter;
+import lombok.With;
+
+import javax.annotation.Nonnull;
+
+@EqualsAndHashCode
+public class HTTPHeader {
+ @With @Getter @Nonnull
+ private final String name;
+ @With @Getter
+ private final String value;
+
+ public HTTPHeader(@Nonnull String name, String value) {
+ this.name = name;
+ this.value = value;
+ }
+}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPHeaderUtil.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPHeaderUtil.java
new file mode 100644
index 000000000..06bf48242
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPHeaderUtil.java
@@ -0,0 +1,45 @@
+package org.hl7.fhir.utilities.http;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+public class HTTPHeaderUtil {
+
+ public static final String USER_AGENT = "User-Agent";
+
+
+ public static Map> getMultimap(Iterable headers) {
+ Map> result = new HashMap<>();
+ if (headers != null) {
+ for (HTTPHeader header : headers) {
+ List values = result.getOrDefault(header.getName(), new ArrayList<>());
+ values.add(header.getValue());
+ result.put(header.getName(), values);
+ }
+ }
+ return result;
+ }
+
+ public static Iterable getHeaders(Iterable headers, String key) {
+ List result = new ArrayList<>();
+ if (headers != null) {
+ for (HTTPHeader header : headers) {
+ if (header.getName().equalsIgnoreCase(key)) {
+ result.add(header.getValue());
+ }
+ }
+ }
+ return result;
+ }
+
+ public static String getSingleHeader(Iterable headers, String key) {
+ for (HTTPHeader header : headers) {
+ if (header.getName().equalsIgnoreCase(key)) {
+ return header.getValue();
+ }
+ }
+ return null;
+ }
+}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPRequest.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPRequest.java
new file mode 100644
index 000000000..19bfa9d36
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPRequest.java
@@ -0,0 +1,54 @@
+package org.hl7.fhir.utilities.http;
+
+import lombok.AllArgsConstructor;
+import lombok.Getter;
+import lombok.With;
+
+import javax.annotation.Nonnull;
+import javax.annotation.Nullable;
+import java.net.MalformedURLException;
+import java.net.URL;
+import java.util.Collections;
+
+@AllArgsConstructor
+public class HTTPRequest {
+
+ public HTTPRequest() {
+ url = null;
+ method = HttpMethod.GET;
+ body = null;
+ contentType = null;
+ headers = Collections.emptyList();
+ }
+
+ public enum HttpMethod {
+ GET, POST, PUT, DELETE, OPTIONS, HEAD, PATCH
+ }
+
+ @Getter @Nullable
+ private final URL url;
+
+ public HTTPRequest withUrl(URL url) {
+ return new HTTPRequest(url, method, body, contentType, headers);
+ }
+
+ public HTTPRequest withUrl(String url) {
+ try {
+ return withUrl(new URL(url));
+ } catch (MalformedURLException e) {
+ throw new IllegalArgumentException("Invalid URL: " + url, e);
+ }
+ }
+
+ @With @Getter
+ private final HttpMethod method;
+
+ @With @Getter @Nullable
+ private final byte[] body;
+
+ @With @Getter @Nullable
+ private final String contentType;
+
+ @With @Getter @Nonnull
+ private final Iterable headers;
+}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPResult.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPResult.java
index 0a3986ec1..3946442b8 100644
--- a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPResult.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/HTTPResult.java
@@ -2,39 +2,40 @@ package org.hl7.fhir.utilities.http;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
+import java.util.*;
+import lombok.Getter;
import org.hl7.fhir.utilities.TextFile;
import org.hl7.fhir.utilities.Utilities;
public class HTTPResult {
- private int code;
- private String contentType;
- private byte[] content;
- private String source;
- private String message;
-
-
+
+ @Getter
+ private final int code;
+ @Getter
+ private final String contentType;
+ @Getter
+ private final byte[] content;
+ @Getter
+ private final String source;
+ @Getter
+ private final String message;
+ @Getter
+ private final Iterable headers;
+
public HTTPResult(String source, int code, String message, String contentType, byte[] content) {
+ this(source, code, message, contentType, content, Collections.emptyList());
+ }
+
+
+ public HTTPResult(String source, int code, String message, String contentType, byte[] content, Iterable headers) {
super();
this.source = source;
this.code = code;
this.contentType = contentType;
this.content = content;
this.message = message;
- }
-
- public int getCode() {
- return code;
- }
- public String getContentType() {
- return contentType;
- }
- public byte[] getContent() {
- return content;
- }
-
- public String getSource() {
- return source;
+ this.headers = headers;
}
public void checkThrowException() throws IOException {
@@ -52,11 +53,7 @@ public class HTTPResult {
}
}
- public String getMessage() {
- return message;
- }
-
public String getContentAsString() {
return new String(content, StandardCharsets.UTF_8);
- }
+ }
}
\ No newline at end of file
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedFhirWebAccessor.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedFhirWebAccessor.java
new file mode 100644
index 000000000..b12c84175
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedFhirWebAccessor.java
@@ -0,0 +1,170 @@
+package org.hl7.fhir.utilities.http;
+
+import okhttp3.*;
+import org.hl7.fhir.utilities.ToolingClientLogger;
+import org.hl7.fhir.utilities.http.okhttpimpl.LoggingInterceptor;
+import org.hl7.fhir.utilities.http.okhttpimpl.ProxyAuthenticator;
+import org.hl7.fhir.utilities.http.okhttpimpl.RetryInterceptor;
+import org.hl7.fhir.utilities.settings.ServerDetailsPOJO;
+
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Map;
+import java.util.concurrent.TimeUnit;
+
+public class ManagedFhirWebAccessor extends ManagedWebAccessorBase {
+
+ /**
+ * The singleton instance of the HttpClient, used for all requests.
+ */
+ private static OkHttpClient okHttpClient;
+
+ private long timeout;
+ private TimeUnit timeoutUnit;
+ private int retries;
+ private ToolingClientLogger logger;
+ private LoggingInterceptor loggingInterceptor;
+
+ public ManagedFhirWebAccessor withTimeout(long timeout, TimeUnit timeoutUnit) {
+ this.timeout = timeout;
+ this.timeoutUnit = timeoutUnit;
+ return this;
+ }
+
+ public ManagedFhirWebAccessor withRetries(int retries) {
+ this.retries = retries;
+ return this;
+ }
+
+ public ManagedFhirWebAccessor withLogger(ToolingClientLogger logger) {
+ this.logger = logger;
+ this.loggingInterceptor = new LoggingInterceptor(logger);
+ return this;
+ }
+
+
+ public ManagedFhirWebAccessor(String userAgent, List serverAuthDetails) {
+ super(userAgent, serverAuthDetails);
+ this.timeout = 5000;
+ this.timeoutUnit = TimeUnit.MILLISECONDS;
+ }
+
+ protected HTTPRequest httpRequestWithDefaultHeaders(HTTPRequest request) {
+ List headers = new ArrayList<>();
+ if (HTTPHeaderUtil.getSingleHeader(request.getHeaders(), HTTPHeaderUtil.USER_AGENT) == null
+ && getUserAgent() != null) {
+ headers.add(new HTTPHeader(HTTPHeaderUtil.USER_AGENT, getUserAgent()));
+ }
+ request.getHeaders().forEach(headers::add);
+ return request.withHeaders(headers);
+ }
+
+ protected HTTPRequest requestWithManagedHeaders(HTTPRequest httpRequest) {
+ HTTPRequest requestWithDefaultHeaders = httpRequestWithDefaultHeaders(httpRequest);
+
+ List headers = new ArrayList<>();
+ requestWithDefaultHeaders.getHeaders().forEach(headers::add);
+
+ for (Map.Entry entry : this.getHeaders().entrySet()) {
+ headers.add(new HTTPHeader(entry.getKey(), entry.getValue()));
+ }
+
+ if (getAuthenticationMode() != null) {
+ if (getAuthenticationMode() != HTTPAuthenticationMode.NONE) {
+ switch (getAuthenticationMode()) {
+ case BASIC:
+ final String basicCredential = Credentials.basic(getUsername(), getPassword());
+ headers.add(new HTTPHeader("Authorization", basicCredential));
+ break;
+ case TOKEN:
+ String tokenCredential = "Bearer " + getToken();
+ headers.add(new HTTPHeader("Authorization", tokenCredential));
+ break;
+ case APIKEY:
+ String apiKeyCredential = getToken();
+ headers.add(new HTTPHeader("Api-Key", apiKeyCredential));
+ break;
+ }
+ }
+ } else {
+ ServerDetailsPOJO settings = ManagedWebAccessUtils.getServer(httpRequest.getUrl().toString(), getServerAuthDetails());
+ if (settings != null) {
+ switch (settings.getAuthenticationType()) {
+ case "basic":
+ final String basicCredential = Credentials.basic(settings.getUsername(), settings.getPassword());
+ headers.add(new HTTPHeader("Authorization", basicCredential));
+ break;
+ case "token":
+ String tokenCredential = "Bearer " + settings.getToken();
+ headers.add(new HTTPHeader("Authorization", tokenCredential));
+ break;
+ case "apikey":
+ String apiKeyCredential = settings.getApikey();
+ headers.add(new HTTPHeader("Api-Key", apiKeyCredential));
+ break;
+ }
+ }
+ }
+ return httpRequest.withHeaders(headers);
+ }
+
+ public HTTPResult httpCall(HTTPRequest httpRequest) throws IOException {
+ switch (ManagedWebAccess.getAccessPolicy()) {
+ case DIRECT:
+
+ HTTPRequest httpRequestWithDirectHeaders = requestWithManagedHeaders(httpRequest);
+ assert httpRequestWithDirectHeaders.getUrl() != null;
+
+ RequestBody body = httpRequestWithDirectHeaders.getBody() == null ? null : RequestBody.create(httpRequestWithDirectHeaders.getBody());
+ Request.Builder requestBuilder = new Request.Builder()
+ .url(httpRequestWithDirectHeaders.getUrl())
+ .method(httpRequestWithDirectHeaders.getMethod().name(), body);
+
+ for (HTTPHeader header : httpRequestWithDirectHeaders.getHeaders()) {
+ requestBuilder.addHeader(header.getName(), header.getValue());
+ }
+ OkHttpClient okHttpClient = getOkHttpClient();
+ //TODO check and throw based on httpRequest:
+
+ if (!ManagedWebAccess.inAllowedPaths(httpRequestWithDirectHeaders.getUrl().toString())) {
+ throw new IOException("The pathname '"+httpRequestWithDirectHeaders.getUrl().toString()+"' cannot be accessed by policy");}
+ Response response = okHttpClient.newCall(requestBuilder.build()).execute();
+ return getHTTPResult(response);
+ case MANAGED:
+ HTTPRequest httpRequestWithManagedHeaders = requestWithManagedHeaders(httpRequest);
+ assert httpRequestWithManagedHeaders.getUrl() != null;
+ return ManagedWebAccess.getFhirWebAccessor().httpCall(httpRequestWithManagedHeaders);
+ case PROHIBITED:
+ throw new IOException("Access to the internet is not allowed by local security policy");
+ default:
+ throw new IOException("Internal Error");
+ }
+ }
+
+ private HTTPResult getHTTPResult(Response execute) throws IOException {
+ return new HTTPResult(execute.request().url().toString(), execute.code(), execute.message(), execute.header("Content-Type"), execute.body() != null && execute.body().contentLength() != 0 ? execute.body().bytes() : null, getHeadersFromResponse(execute));
+ }
+
+ private Iterable getHeadersFromResponse(Response response) {
+ List headers = new ArrayList<>();
+ for (String name : response.headers().names()) {
+ headers.add(new HTTPHeader(name, response.header(name)));
+ }
+ return headers;
+ }
+
+ private OkHttpClient getOkHttpClient() {
+ if (okHttpClient == null) {
+ okHttpClient = new OkHttpClient();
+ }
+ OkHttpClient.Builder builder = okHttpClient.newBuilder();
+ if (logger != null) builder.addInterceptor(loggingInterceptor);
+ builder.addInterceptor(new RetryInterceptor(retries));
+ builder.proxyAuthenticator(new ProxyAuthenticator());
+ return builder.connectTimeout(timeout, timeoutUnit)
+ .writeTimeout(timeout, timeoutUnit)
+ .readTimeout(timeout, timeoutUnit).build();
+ }
+
+}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccess.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccess.java
index f1afac25d..3ebb85c20 100644
--- a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccess.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccess.java
@@ -31,23 +31,16 @@ package org.hl7.fhir.utilities.http;
-import java.io.File;
-import java.io.FileInputStream;
-import java.io.FileNotFoundException;
-import java.io.FileOutputStream;
import java.io.IOException;
-import java.io.InputStream;
-import java.nio.file.Path;
-import java.util.ArrayList;
-import java.util.List;
-import java.util.Map;
+import java.util.*;
-import org.hl7.fhir.utilities.Utilities;
+import lombok.Getter;
+import org.hl7.fhir.utilities.settings.FhirSettings;
import org.hl7.fhir.utilities.settings.ServerDetailsPOJO;
/**
- * see security.md - manages access to the local file system by the FHIR HAPI Core library
- *
+ * see security.md - manages web access by the FHIR HAPI Core library
+ *
* By using accessPolicy, allowedDomains and accessor, a host java application can control
* whether this library has direct access to the web (and which domains it is allowed to access),
* or whether the host application provides controlled access, or whether no access is allowed at all
@@ -57,13 +50,17 @@ import org.hl7.fhir.utilities.settings.ServerDetailsPOJO;
*
*/
public class ManagedWebAccess {
-
+
public interface IWebAccessor {
HTTPResult get(String url, String accept, Map headers) throws IOException;
HTTPResult post(String url, byte[] bytes, String contentType, String accept, Map headers) throws IOException;
HTTPResult put(String url, byte[] bytes, String contentType, String accept, Map headers) throws IOException;
}
+ public interface IFhirWebAccessor {
+ HTTPResult httpCall(HTTPRequest httpRequest);
+ }
+
public enum WebAccessPolicy {
DIRECT, // open access to the web, though access can be restricted only to domains in AllowedDomains
MANAGED, // no access except by the IWebAccessor
@@ -71,12 +68,18 @@ public class ManagedWebAccess {
}
private static WebAccessPolicy accessPolicy = WebAccessPolicy.DIRECT; // for legacy reasons
+ //TODO get this from fhir settings
private static List allowedDomains = new ArrayList<>();
+ @Getter
private static IWebAccessor accessor;
+
+ @Getter
+ private static IFhirWebAccessor fhirWebAccessor;
+
+ @Getter
private static String userAgent;
private static List serverAuthDetails;
-
-
+
public static WebAccessPolicy getAccessPolicy() {
return accessPolicy;
}
@@ -97,37 +100,51 @@ public class ManagedWebAccess {
return false;
}
- public static String getUserAgent() {
- return userAgent;
- }
-
public static void setUserAgent(String userAgent) {
ManagedWebAccess.userAgent = userAgent;
}
- public static IWebAccessor getAccessor() {
- return accessor;
+ public static ManagedWebAccessor accessor() {
+ return new ManagedWebAccessor(userAgent, serverAuthDetails);
}
- public static ManagedWebAccessBuilder builder() {
- return new ManagedWebAccessBuilder(userAgent, serverAuthDetails);
+ public static ManagedFhirWebAccessor fhirAccessor() {
+ return new ManagedFhirWebAccessor(userAgent, serverAuthDetails);
}
public static HTTPResult get(String url) throws IOException {
- return builder().get(url);
+ return accessor().get(url);
}
public static HTTPResult get(String url, String accept) throws IOException {
- return builder().withAccept(accept).get(url);
+ return accessor().get(url, accept);
}
public static HTTPResult post(String url, byte[] content, String contentType, String accept) throws IOException {
- return builder().withAccept(accept).post(url, content, contentType);
+ return accessor().post(url, content, contentType, accept);
}
-
public static HTTPResult put(String url, byte[] content, String contentType, String accept) throws IOException {
- return builder().withAccept(accept).put(url, content, contentType);
+ return accessor().put(url, content, contentType, accept);
}
+ public static HTTPResult httpCall(HTTPRequest httpRequest) throws IOException {
+ return fhirAccessor().httpCall(httpRequest);
+ }
+
+ public static void loadFromFHIRSettings() {
+ setAccessPolicy(FhirSettings.isProhibitNetworkAccess() ? WebAccessPolicy.PROHIBITED : WebAccessPolicy.DIRECT);
+ setUserAgent("hapi-fhir-tooling-client");
+ serverAuthDetails = new ArrayList<>();
+ serverAuthDetails.addAll(FhirSettings.getPackageServers());
+ serverAuthDetails.addAll(FhirSettings.getTerminologyServers());
+ }
+
+ public static void loadFromFHIRSettings(FhirSettings settings) {
+ setAccessPolicy(settings.isProhibitNetworkAccess() ? WebAccessPolicy.PROHIBITED : WebAccessPolicy.DIRECT);
+ setUserAgent("hapi-fhir-tooling-client");
+ serverAuthDetails = new ArrayList<>();
+ serverAuthDetails.addAll(settings.getPackageServers());
+ serverAuthDetails.addAll(settings.getTerminologyServers());
+ }
}
\ No newline at end of file
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessUtils.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessUtils.java
new file mode 100644
index 000000000..a97353633
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessUtils.java
@@ -0,0 +1,17 @@
+package org.hl7.fhir.utilities.http;
+
+import org.hl7.fhir.utilities.settings.ServerDetailsPOJO;
+
+public class ManagedWebAccessUtils {
+
+ public static ServerDetailsPOJO getServer(String url, Iterable serverAuthDetails) {
+ if (serverAuthDetails != null) {
+ for (ServerDetailsPOJO t : serverAuthDetails) {
+ if (url.startsWith(t.getUrl())) {
+ return t;
+ }
+ }
+ }
+ return null;
+ }
+}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessBuilder.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessor.java
similarity index 54%
rename from org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessBuilder.java
rename to org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessor.java
index 79f27add7..4c364a2f5 100644
--- a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessBuilder.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessor.java
@@ -9,61 +9,30 @@ import java.util.Map;
import org.hl7.fhir.utilities.settings.ServerDetailsPOJO;
+/**
+ * Simple HTTP client for making requests to a server.
+ */
+public class ManagedWebAccessor extends ManagedWebAccessorBase {
-public class ManagedWebAccessBuilder {
-
- private String userAgent;
- private HTTPAuthenticationMode authenticationMode;
- private String username;
- private String password;
- private String token;
- private String accept;
- private List serverAuthDetails;
- private Map headers = new HashMap();
-
- public ManagedWebAccessBuilder(String userAgent, List serverAuthDetails) {
- this.userAgent = userAgent;
- this.serverAuthDetails = serverAuthDetails;
- }
-
- public ManagedWebAccessBuilder withAccept(String accept) {
- this.accept = accept;
- return this;
- }
-
- public ManagedWebAccessBuilder withHeader(String name, String value) {
- headers.put(name, value);
- return this;
- }
-
- public ManagedWebAccessBuilder withBasicAuth(String username, String password) {
- this.authenticationMode = HTTPAuthenticationMode.BASIC;
- this.username = username;
- this.password = password;
- return this;
+ public ManagedWebAccessor(String userAgent, List serverAuthDetails) {
+ super(userAgent, serverAuthDetails);
}
- public ManagedWebAccessBuilder withToken(String token) {
- this.authenticationMode = HTTPAuthenticationMode.TOKEN;
- this.token = token;
- return this;
- }
-
- private Map headers() {
- Map headers = new HashMap();
- headers.putAll(this.headers);
- if (authenticationMode == HTTPAuthenticationMode.TOKEN) {
- headers.put("Authorization", "Bearer " + token);
- } else if (authenticationMode == HTTPAuthenticationMode.BASIC) {
- String auth = username+":"+password;
+ private Map newHeaders() {
+ Map headers = new HashMap(this.getHeaders());
+ if (getAuthenticationMode() == HTTPAuthenticationMode.TOKEN) {
+ headers.put("Authorization", "Bearer " + getToken());
+ } else if (getAuthenticationMode() == HTTPAuthenticationMode.BASIC) {
+ String auth = getUsername() +":"+ getPassword();
byte[] encodedAuth = Base64.getEncoder().encode(auth.getBytes(StandardCharsets.UTF_8));
headers.put("Authorization", "Basic " + new String(encodedAuth));
+ } else if (getAuthenticationMode() == HTTPAuthenticationMode.APIKEY) {
+ headers.put("Api-Key", getToken());
}
- if (userAgent != null) {
- headers.put("User-Agent", userAgent);
+ if (getUserAgent() != null) {
+ headers.put("User-Agent", getUserAgent());
}
-
return headers;
}
@@ -72,25 +41,32 @@ public class ManagedWebAccessBuilder {
throw new IOException("The pathname '"+url+"' cannot be accessed by policy");
}
SimpleHTTPClient client = new SimpleHTTPClient();
- if (userAgent != null) {
- client.addHeader("User-Agent", userAgent);
+
+ for (Map.Entry entry : this.getHeaders().entrySet()) {
+ client.addHeader(entry.getKey(), entry.getValue());
}
- if (authenticationMode != null && authenticationMode != HTTPAuthenticationMode.NONE) {
- client.setAuthenticationMode(authenticationMode);
- switch (authenticationMode) {
+
+ if (getUserAgent() != null) {
+ client.addHeader("User-Agent", getUserAgent());
+ }
+ if (getAuthenticationMode() != null) {
+ if (getAuthenticationMode() != HTTPAuthenticationMode.NONE) {
+ client.setAuthenticationMode(getAuthenticationMode());
+ switch (getAuthenticationMode()) {
case BASIC :
- client.setUsername(username);
- client.setPassword(password);
+ client.setUsername(getUsername());
+ client.setPassword(getPassword());
break;
case TOKEN :
- client.setToken(token);
+ client.setToken(getToken());
break;
case APIKEY :
- client.setApiKey(token);
+ client.setApiKey(getToken());
break;
}
+ }
} else {
- ServerDetailsPOJO settings = getServer(url);
+ ServerDetailsPOJO settings = ManagedWebAccessUtils.getServer(url, getServerAuthDetails());
if (settings != null) {
switch (settings.getAuthenticationType()) {
case "basic" :
@@ -109,32 +85,23 @@ public class ManagedWebAccessBuilder {
}
}
}
- if (username != null || token != null) {
-
- client.setAuthenticationMode(authenticationMode);
+ if (getUsername() != null || getToken() != null) {
+ client.setAuthenticationMode(getAuthenticationMode());
}
return client;
}
-
- private ServerDetailsPOJO getServer(String url) {
- if (serverAuthDetails != null) {
- for (ServerDetailsPOJO t : serverAuthDetails) {
- if (url.startsWith(t.getUrl())) {
- return t;
- }
- }
- }
- return null;
+ public HTTPResult get(String url) throws IOException {
+ return get(url, null);
}
- public HTTPResult get(String url) throws IOException {
+ public HTTPResult get(String url, String accept) throws IOException {
switch (ManagedWebAccess.getAccessPolicy()) {
case DIRECT:
SimpleHTTPClient client = setupClient(url);
return client.get(url, accept);
case MANAGED:
- return ManagedWebAccess.getAccessor().get(url, accept, headers());
+ return ManagedWebAccess.getAccessor().get(url, accept, newHeaders());
case PROHIBITED:
throw new IOException("Access to the internet is not allowed by local security policy");
default:
@@ -142,14 +109,17 @@ public class ManagedWebAccessBuilder {
}
}
-
public HTTPResult post(String url, byte[] content, String contentType) throws IOException {
+ return post(url, content, contentType, null);
+ }
+
+ public HTTPResult post(String url, byte[] content, String contentType, String accept) throws IOException {
switch (ManagedWebAccess.getAccessPolicy()) {
case DIRECT:
SimpleHTTPClient client = setupClient(url);
return client.post(url, contentType, content, accept);
case MANAGED:
- return ManagedWebAccess.getAccessor().post(url, content, contentType, accept, headers());
+ return ManagedWebAccess.getAccessor().post(url, content, contentType, accept, newHeaders());
case PROHIBITED:
throw new IOException("Access to the internet is not allowed by local security policy");
default:
@@ -158,17 +128,20 @@ public class ManagedWebAccessBuilder {
}
public HTTPResult put(String url, byte[] content, String contentType) throws IOException {
+ return put(url, content, contentType, null);
+ }
+
+ public HTTPResult put(String url, byte[] content, String contentType, String accept) throws IOException {
switch (ManagedWebAccess.getAccessPolicy()) {
case DIRECT:
SimpleHTTPClient client = setupClient(url);
return client.put(url, contentType, content, accept);
case MANAGED:
- return ManagedWebAccess.getAccessor().put(url, content, contentType, accept, headers());
+ return ManagedWebAccess.getAccessor().put(url, content, contentType, accept, newHeaders());
case PROHIBITED:
throw new IOException("Access to the internet is not allowed by local security policy");
default:
throw new IOException("Internal Error");
}
}
-
}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessorBase.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessorBase.java
new file mode 100644
index 000000000..67e4df641
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/ManagedWebAccessorBase.java
@@ -0,0 +1,78 @@
+package org.hl7.fhir.utilities.http;
+
+import lombok.Getter;
+import org.hl7.fhir.utilities.settings.ServerDetailsPOJO;
+
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+
+public abstract class ManagedWebAccessorBase> {
+ @Getter
+ private final String userAgent;
+ @Getter
+ private HTTPAuthenticationMode authenticationMode;
+ @Getter
+ private String username;
+ @Getter
+ private String password;
+ @Getter
+ private String token;
+
+ @Getter
+ private final List serverAuthDetails;
+ @Getter
+ private final Map headers = new HashMap<>();
+
+ public ManagedWebAccessorBase(String userAgent, List serverAuthDetails) {
+ this.userAgent = userAgent;
+ this.serverAuthDetails = serverAuthDetails;
+ }
+
+ @SuppressWarnings("unchecked")
+ final B self() {
+ return (B) this;
+ }
+
+ public B withHeader(String name, String value) {
+ headers.put(name, value);
+ return self();
+ }
+
+ public B withBasicAuth(String username, String password) {
+ this.authenticationMode = HTTPAuthenticationMode.BASIC;
+ this.username = username;
+ this.password = password;
+ return self();
+ }
+
+ public B withToken(String token) {
+ this.authenticationMode = HTTPAuthenticationMode.TOKEN;
+ this.token = token;
+ return self();
+ }
+
+ public B withApiKey(String apiKey) {
+ this.authenticationMode = HTTPAuthenticationMode.APIKEY;
+ this.token = apiKey;
+ return self();
+ }
+
+ public B withNoneAuth() {
+ this.authenticationMode = HTTPAuthenticationMode.NONE;
+ setAllAuthHeadersToNull();
+ return self();
+ }
+
+ public B withServerSpecificAuth() {
+ this.authenticationMode = null;
+ setAllAuthHeadersToNull();
+ return self();
+ }
+
+ private void setAllAuthHeadersToNull() {
+ this.token = null;
+ this.username = null;
+ this.password = null;
+ }
+}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/SimpleHTTPClient.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/SimpleHTTPClient.java
index 0e448293b..bf35ee906 100644
--- a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/SimpleHTTPClient.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/SimpleHTTPClient.java
@@ -20,27 +20,10 @@ import lombok.Setter;
public class SimpleHTTPClient {
-
- public static class Header {
- private String name;
- private String value;
- public Header(String name, String value) {
- super();
- this.name = name;
- this.value = value;
- }
- public String getName() {
- return name;
- }
- public String getValue() {
- return value;
- }
- }
-
private static final int MAX_REDIRECTS = 5;
private static int counter = 1;
- private List headers = new ArrayList<>();
+ private List headers = new ArrayList<>();
@Getter @Setter
private HTTPAuthenticationMode authenticationMode;
@@ -58,7 +41,7 @@ public class SimpleHTTPClient {
private String apiKey;
public void addHeader(String name, String value) {
- headers.add(new Header(name, value));
+ headers.add(new HTTPHeader(name, value));
}
public HTTPResult get(String url) throws IOException {
@@ -114,8 +97,8 @@ public class SimpleHTTPClient {
private void setHeaders(HttpURLConnection c) {
if (headers != null) {
- for (Header h : headers) {
- c.setRequestProperty(h.getName(), h.getValue());
+ for (HTTPHeader header : headers) {
+ c.setRequestProperty(header.getName(), header.getValue());
}
}
c.setConnectTimeout(15000);
@@ -126,7 +109,7 @@ public class SimpleHTTPClient {
private void setAuthenticationHeader(HttpURLConnection c) {
String authHeaderValue = null;
if (authenticationMode == HTTPAuthenticationMode.TOKEN) {
- authHeaderValue = "Bearer " + new String(token);
+ authHeaderValue = "Bearer " + token;
} else if (authenticationMode == HTTPAuthenticationMode.BASIC) {
String auth = username+":"+password;
byte[] encodedAuth = Base64.getEncoder().encode(auth.getBytes(StandardCharsets.UTF_8));
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/LoggingInterceptor.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/LoggingInterceptor.java
new file mode 100644
index 000000000..35dd0f947
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/LoggingInterceptor.java
@@ -0,0 +1,69 @@
+package org.hl7.fhir.utilities.http.okhttpimpl;
+
+import okhttp3.*;
+import okio.Buffer;
+import org.hl7.fhir.utilities.ToolingClientLogger;
+
+import javax.annotation.Nonnull;
+import java.io.IOException;
+import java.util.ArrayList;
+import java.util.List;
+import java.util.Map;
+
+public class LoggingInterceptor implements Interceptor{
+
+ private ToolingClientLogger logger;
+
+ public LoggingInterceptor(ToolingClientLogger logger) {
+ this.logger = logger;
+ }
+
+ public LoggingInterceptor setLogger(ToolingClientLogger logger) {
+ this.logger = logger;
+ return this;
+ }
+
+ @Override
+ public Response intercept(@Nonnull Interceptor.Chain chain) throws IOException {
+ // Log Request
+ Request request = chain.request();
+ List hdrs = new ArrayList<>();
+ for (String s : request.headers().toString().split("\\n")) {
+ hdrs.add(s.trim());
+ }
+ byte[] cnt = null;
+ if (request.body() != null) {
+ Buffer buf = new Buffer();
+ request.body().writeTo(buf);
+ cnt = buf.readByteArray();
+ }
+ if (logger != null) {
+ logger.logRequest(request.method(), request.url().toString(), hdrs, cnt);
+ }
+
+ // Log Response
+ Response response = null;
+ response = chain.proceed(chain.request());
+
+ MediaType contentType = null;
+ byte[] bodyBytes = null;
+ if (response.body() != null) {
+ contentType = response.body().contentType();
+ bodyBytes = response.body().bytes();
+ }
+
+ // Get Headers as List
+ List headerList = new ArrayList<>();
+ Map> headerMap = response.headers().toMultimap();
+ headerMap.keySet().forEach(key -> headerMap.get(key).forEach(value -> headerList.add(key + ":" + value)));
+
+ if (logger != null) {
+ long responseTimeInMillis = response.receivedResponseAtMillis() - response.sentRequestAtMillis();
+ logger.logResponse(Integer.toString(response.code()), headerList, bodyBytes, responseTimeInMillis);
+ }
+
+ // Reading byte[] clears body. Need to recreate.
+ ResponseBody body = ResponseBody.create(bodyBytes, contentType);
+ return response.newBuilder().body(body).build();
+ }
+}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/ProxyAuthenticator.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/ProxyAuthenticator.java
new file mode 100644
index 000000000..1f59167d3
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/ProxyAuthenticator.java
@@ -0,0 +1,24 @@
+package org.hl7.fhir.utilities.http.okhttpimpl;
+
+import okhttp3.*;
+
+import java.io.IOException;
+
+public class ProxyAuthenticator implements Authenticator {
+ protected static final String HTTP_PROXY_USER = "http.proxyUser";
+ protected static final String HTTP_PROXY_PASS = "http.proxyPassword";
+ protected static final String HEADER_PROXY_AUTH = "Proxy-Authorization";
+
+ @Override
+ public Request authenticate(Route route, Response response) throws IOException {
+ final String httpProxyUser = System.getProperty(HTTP_PROXY_USER);
+ final String httpProxyPass = System.getProperty(HTTP_PROXY_PASS);
+ if (httpProxyUser != null && httpProxyPass != null) {
+ String credential = Credentials.basic(httpProxyUser, httpProxyPass);
+ return response.request().newBuilder()
+ .header(HEADER_PROXY_AUTH, credential)
+ .build();
+ }
+ return response.request().newBuilder().build();
+ }
+}
diff --git a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/RetryInterceptor.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/RetryInterceptor.java
similarity index 94%
rename from org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/RetryInterceptor.java
rename to org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/RetryInterceptor.java
index e3b6ec084..ca71c9a01 100644
--- a/org.hl7.fhir.dstu3/src/main/java/org/hl7/fhir/dstu3/utils/client/network/RetryInterceptor.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/http/okhttpimpl/RetryInterceptor.java
@@ -1,11 +1,11 @@
-package org.hl7.fhir.dstu3.utils.client.network;
-
-import java.io.IOException;
+package org.hl7.fhir.utilities.http.okhttpimpl;
import okhttp3.Interceptor;
import okhttp3.Request;
import okhttp3.Response;
+import java.io.IOException;
+
/**
* An {@link Interceptor} for {@link okhttp3.OkHttpClient} that controls the number of times we retry a to execute a
* given request, before reporting a failure. This includes unsuccessful return codes and timeouts.
@@ -26,7 +26,7 @@ public class RetryInterceptor implements Interceptor {
}
@Override
- public Response intercept(Interceptor.Chain chain) throws IOException {
+ public Response intercept(Chain chain) throws IOException {
Request request = chain.request();
Response response = null;
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/i18n/I18nConstants.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/i18n/I18nConstants.java
index 78309dfda..c36b67897 100644
--- a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/i18n/I18nConstants.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/i18n/I18nConstants.java
@@ -178,7 +178,6 @@ public class I18nConstants {
public static final String DIFFERENTIAL_WALKS_INTO____BUT_THE_BASE_DOES_NOT_AND_THERE_IS_NOT_A_SINGLE_FIXED_TYPE_THE_TYPE_IS__THIS_IS_NOT_HANDLED_YET = "Differential_walks_into____but_the_base_does_not_and_there_is_not_a_single_fixed_type_The_type_is__This_is_not_handled_yet";
public static final String DISCRIMINATOR_BAD_PATH = "DISCRIMINATOR_BAD_PATH";
public static final String DISCRIMINATOR__IS_BASED_ON_ELEMENT_EXISTENCE_BUT_SLICE__NEITHER_SETS_MIN1_OR_MAX0 = "Discriminator__is_based_on_element_existence_but_slice__neither_sets_min1_or_max0";
- public static final String DISCRIMINATOR__IS_BASED_ON_TYPE_BUT_SLICE__IN__HAS_MULTIPLE_TYPES = "Discriminator__is_based_on_type_but_slice__in__has_multiple_types";
public static final String DISCRIMINATOR__IS_BASED_ON_TYPE_BUT_SLICE__IN__HAS_NO_TYPES = "Discriminator__is_based_on_type_but_slice__in__has_no_types";
public static final String DISPLAY_NAME_FOR__SHOULD_BE_ONE_OF__INSTEAD_OF = "Display_Name_for__should_be_one_of__instead_of";
public static final String DISPLAY_NAME_WS_FOR__SHOULD_BE_ONE_OF__INSTEAD_OF = "Display_Name_WS_for__should_be_one_of__instead_of";
@@ -475,7 +474,6 @@ public class I18nConstants {
public static final String PROBLEM_PROCESSING_EXPRESSION__IN_PROFILE__PATH__ = "Problem_processing_expression__in_profile__path__";
public static final String PROFILE_BASED_DISCRIMINATORS_MUST_HAVE_A_TYPE_WITH_A_PROFILE__IN_PROFILE_ = "Profile_based_discriminators_must_have_a_type_with_a_profile__in_profile_";
public static final String PROFILE_BASED_DISCRIMINATORS_MUST_HAVE_A_TYPE__IN_PROFILE_ = "Profile_based_discriminators_must_have_a_type__in_profile_";
- public static final String PROFILE_BASED_DISCRIMINATORS_MUST_HAVE_ONLY_ONE_TYPE__IN_PROFILE = "Profile_based_discriminators_must_have_only_one_type__in_profile";
public static final String PROFILE_EXT_NOT_HERE = "Profile_EXT_Not_Here";
public static final String PROFILE_VAL_MISSINGELEMENT = "Profile_VAL_MissingElement";
public static final String PROFILE_VAL_NOTALLOWED = "Profile_VAL_NotAllowed";
@@ -1123,4 +1121,9 @@ public class I18nConstants {
public static final String CODESYSTEM_CS_COMPLETE_AND_EMPTY = "CODESYSTEM_CS_COMPLETE_AND_EMPTY";
public static final String VALIDATION_VAL_VERSION_NOHASH = "VALIDATION_VAL_VERSION_NOHASH";
public static final String PRIMITIVE_TOO_SHORT = "PRIMITIVE_TOO_SHORT";
+ public static final String CANONICAL_MULTIPLE_VERSIONS_KNOWN = "CANONICAL_MULTIPLE_VERSIONS_KNOWN";
+ public static final String SD_PATH_NO_SLICING = "SD_PATH_NO_SLICING";
+ public static final String SD_PATH_SLICING_DEPRECATED = "SD_PATH_SLICING_DEPRECATED";
+ public static final String SD_PATH_NOT_VALID = "SD_PATH_NOT_VALID";
+ public static final String SD_PATH_ERROR = "SD_PATH_ERROR";
}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/npm/PackageClient.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/npm/PackageClient.java
index 4bf827356..6f6feeb84 100644
--- a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/npm/PackageClient.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/npm/PackageClient.java
@@ -14,7 +14,6 @@ import java.util.Date;
import java.util.List;
import java.util.Set;
-import javax.annotation.Nonnull;
import javax.annotation.Nullable;
import org.hl7.fhir.utilities.CommaSeparatedStringBuilder;
@@ -24,7 +23,7 @@ import org.hl7.fhir.utilities.VersionUtilities;
import org.hl7.fhir.utilities.http.HTTPAuthenticationMode;
import org.hl7.fhir.utilities.http.HTTPResult;
import org.hl7.fhir.utilities.http.ManagedWebAccess;
-import org.hl7.fhir.utilities.http.ManagedWebAccessBuilder;
+import org.hl7.fhir.utilities.http.ManagedWebAccessor;
import org.hl7.fhir.utilities.json.model.JsonArray;
import org.hl7.fhir.utilities.json.model.JsonObject;
import org.hl7.fhir.utilities.json.model.JsonProperty;
@@ -176,13 +175,13 @@ public class PackageClient {
}
private InputStream fetchUrl(String source, String accept) throws IOException {
- ManagedWebAccessBuilder client = ManagedWebAccess.builder().withAccept(accept);
+ ManagedWebAccessor webAccessor = ManagedWebAccess.accessor();
if (server.getAuthenticationMode() == HTTPAuthenticationMode.TOKEN) {
- client.withToken(server.getToken());
+ webAccessor.withToken(server.getToken());
} else if (server.getAuthenticationMode() == HTTPAuthenticationMode.BASIC) {
- client.withBasicAuth(server.getUsername(), server.getPassword());
+ webAccessor.withBasicAuth(server.getUsername(), server.getPassword());
}
- HTTPResult res = client.get(source);
+ HTTPResult res = webAccessor.get(source, accept);
res.checkThrowException();
return new ByteArrayInputStream(res.getContent());
}
diff --git a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/settings/FhirSettings.java b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/settings/FhirSettings.java
index dc69f3fd0..060ce2195 100644
--- a/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/settings/FhirSettings.java
+++ b/org.hl7.fhir.utilities/src/main/java/org/hl7/fhir/utilities/settings/FhirSettings.java
@@ -4,6 +4,7 @@ import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
+import java.util.Arrays;
import java.util.Collections;
import java.util.List;
@@ -232,6 +233,14 @@ public class FhirSettings {
if (instance.fhirSettings.getPackageManagement() == null) {
return Collections.emptyList();
}
- return List.of(instance.fhirSettings.getPackageManagement().getServers().toArray(new ServerDetailsPOJO[]{}));
+ return Arrays.asList(instance.fhirSettings.getPackageManagement().getServers().toArray(new ServerDetailsPOJO[]{}));
+ }
+
+ public static List getTerminologyServers() {
+ getInstance();
+ if (instance.fhirSettings.getTerminologyServers() == null) {
+ return Collections.emptyList();
+ }
+ return Arrays.asList(instance.fhirSettings.getTerminologyServers().getServers().toArray(new ServerDetailsPOJO[]{}));
}
}
diff --git a/org.hl7.fhir.utilities/src/main/resources/Messages.properties b/org.hl7.fhir.utilities/src/main/resources/Messages.properties
index 6bbe79598..2e63e64e5 100644
--- a/org.hl7.fhir.utilities/src/main/resources/Messages.properties
+++ b/org.hl7.fhir.utilities/src/main/resources/Messages.properties
@@ -179,8 +179,6 @@ Did_not_find_type_root_ = Did not find type root: {0}
Differential_does_not_have_a_slice__b_of_____in_profile_ = Differential in profile {5} does not have a slice at {6} (on {0}, position {1} of {2} / {3} / {4})
Differential_walks_into____but_the_base_does_not_and_there_is_not_a_single_fixed_type_The_type_is__This_is_not_handled_yet = Differential walks into ''{0} (@ {1})'', but the base does not, and there is not a single fixed type. The type is {2}. This is not handled yet
Discriminator__is_based_on_element_existence_but_slice__neither_sets_min1_or_max0 = Discriminator ({0}) is based on element existence, but slice {1} neither sets min>=1 or max=0
-Discriminator__is_based_on_type_but_slice__in__has_multiple_types_one =
-Discriminator__is_based_on_type_but_slice__in__has_multiple_types_other = Discriminator ({1}) is based on type, but slice {2} in {3} has {0} types: {4}
Discriminator__is_based_on_type_but_slice__in__has_no_types = Discriminator ({0}) is based on type, but slice {1} in {2} has no types
Display_Name_WS_for__should_be_one_of__instead_of_one = Wrong whitespace in Display Name ''{4}'' for {1}#{2}. Valid display is {3} (for the language(s) ''{5}'')
Display_Name_WS_for__should_be_one_of__instead_of_other = Wrong whitespace in Display Name ''{4}'' for {1}#{2}. Valid display is one of {0} choices: {3} (for the language(s) ''{5}'')
@@ -486,8 +484,6 @@ Profile___has_no_base_and_no_snapshot = Profile {0} ({1}) has no base and no sna
Profile__does_not_match_for__because_of_the_following_profile_issues__ = Profile {0} does not match for {1} because of the following profile issues: {2}
Profile_based_discriminators_must_have_a_type__in_profile_ = Profile based discriminators must have a type ({0} in profile {1})
Profile_based_discriminators_must_have_a_type_with_a_profile__in_profile_ = Profile based discriminators must have a type with a profile ({0} in profile {1})
-Profile_based_discriminators_must_have_only_one_type__in_profile_one =
-Profile_based_discriminators_must_have_only_one_type__in_profile_other = Profile based discriminators must have only one type ({1} in profile {2}) but found {0} types
QUESTIONNAIRE_QR_ITEM_BADOPTION_CS = The code provided {1} cannot be validated in the options value set ({2}) in the questionnaire because the system {0} could not be found
QUESTIONNAIRE_Q_DERIVATION_TYPE_IGNORED = The derivation type ''{0}'' means that no derivation checking has been performed against this questionnaire
QUESTIONNAIRE_Q_DERIVATION_TYPE_UNKNOWN = The derivation type ''{0}'' is unknown, which means that no derivation checking has been performed against this questionnaire
@@ -1091,7 +1087,7 @@ XHTML_XHTML_NS_InValid = Wrong namespace on the XHTML (''{0}'', should be ''{1}'
XHTML_XHTML_Name_Invalid = Wrong name on the XHTML (''{0}'') - must start with div
XSI_TYPE_UNNECESSARY = xsi:type is unnecessary at this point
XSI_TYPE_WRONG = The xsi:type value ''{0}'' is wrong (should be ''{1}''). Note that xsi:type is unnecessary at this point
-_DT_Fixed_Wrong = Value is ''{0}'' but must be ''{1}''{2}
+_DT_Fixed_Wrong = Value is ''{0}'' but is fixed to ''{1}'' in the profile {2}
_has_children__and_multiple_types__in_profile_ = {0} has children ({1}) and multiple types ({2}) in profile {3}
_has_children__for_type__in_profile__but_cant_find_type = {0} has children ({1}) for type {2} in profile {3}, but can''t find type
_has_no_children__and_no_types_in_profile_ = {0} has no children ({1}) and no types in profile {2}
@@ -1156,4 +1152,8 @@ SD_ED_ADDITIONAL_BINDING_USAGE_INVALID_TYPE = The Usage Context value must be of
CODESYSTEM_CS_COMPLETE_AND_EMPTY = When a CodeSystem has content = ''complete'', it doesn't make sense for there to be no concepts defined
VALIDATION_VAL_VERSION_NOHASH = Version ''{0}'' contains a ''#'', which as this character is used in some URLs to separate the version and the fragment id. When version does include '#', systems will not be able to parse the URL
PRIMITIVE_TOO_SHORT = Value ''{0}'' is shorter than permitted minimum length of {1}
-
\ No newline at end of file
+CANONICAL_MULTIPLE_VERSIONS_KNOWN = The version {2} for the {0} {1} is not known. These versions are known: {3}
+SD_PATH_SLICING_DEPRECATED = The discriminator type ''{0}'' has been deprecated. Use type=fixed with a pattern[x] instead
+SD_PATH_NOT_VALID = The discriminator path ''{0}'' does not appear to be valid for the element that is being sliced ''{1}''
+SD_PATH_ERROR = The discriminator path ''{0}'' does not appear to be valid for the element that is being sliced ''{1}'': {2}
+
diff --git a/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/HTTPHeaderTests.java b/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/HTTPHeaderTests.java
new file mode 100644
index 000000000..ad86e35e8
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/HTTPHeaderTests.java
@@ -0,0 +1,20 @@
+package org.hl7.fhir.utilities.http;
+
+import org.junit.jupiter.api.Test;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class HTTPHeaderTests {
+ @Test
+ void testHTTPHeaderEquals() {
+ HTTPHeader header1 = new HTTPHeader("name", "value1");
+ HTTPHeader sameAsHeader1 = new HTTPHeader("name", "value1");
+ assertThat(header1).isEqualTo(sameAsHeader1);
+
+ HTTPHeader notSameNameAsHeader1 = new HTTPHeader("name2", "value1");
+ assertThat(header1).isNotEqualTo(notSameNameAsHeader1);
+
+ HTTPHeader notSameValueAsHeader1 = new HTTPHeader("name", "value2");
+ assertThat(header1).isNotEqualTo(notSameValueAsHeader1);
+ }
+}
diff --git a/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/ManagedFhirWebAccessorTests.java b/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/ManagedFhirWebAccessorTests.java
new file mode 100644
index 000000000..be68a2571
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/ManagedFhirWebAccessorTests.java
@@ -0,0 +1,40 @@
+package org.hl7.fhir.utilities.http;
+
+import org.junit.jupiter.api.Assertions;
+import org.junit.jupiter.api.DisplayName;
+import org.junit.jupiter.api.Test;
+
+public class ManagedFhirWebAccessorTests {
+ final String expectedUserAgent = "dummy-agent";
+
+ @Test
+ @DisplayName("Test default headers are added correctly.")
+ void addDefaultAgentHeader() {
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com");
+
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor(expectedUserAgent, null);
+
+ HTTPRequest requestWithDefaultHeaders = builder.httpRequestWithDefaultHeaders(request);
+ assertRequestContainsExpectedAgentHeader(requestWithDefaultHeaders);
+ }
+
+ @Test
+ @DisplayName("Test default headers are added correctly.")
+ void addDefaultBasicHeader() {
+ HTTPRequest request = new HTTPRequest().withUrl("http://www.google.com");
+
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor(expectedUserAgent, null)
+ .withBasicAuth("dummy-user", "dummy-password");
+
+ HTTPRequest requestWithManagedHeaders = builder.requestWithManagedHeaders(request);
+ assertRequestContainsExpectedAgentHeader(requestWithManagedHeaders);
+
+ Assertions.assertNotNull(HTTPHeaderUtil.getSingleHeader(requestWithManagedHeaders.getHeaders(),"Authorization"), "Authorization header null.");
+ }
+
+ private void assertRequestContainsExpectedAgentHeader(HTTPRequest requestWithDefaultHeaders) {
+ Assertions.assertNotNull(HTTPHeaderUtil.getSingleHeader(requestWithDefaultHeaders.getHeaders(),"User-Agent"), "User-Agent header null.");
+ Assertions.assertEquals(expectedUserAgent, HTTPHeaderUtil.getSingleHeader(requestWithDefaultHeaders.getHeaders(),"User-Agent"),
+ "User-Agent header not populated with expected value \""+expectedUserAgent+"\".");
+ }
+}
diff --git a/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/ManagedWebAccessAuthTests.java b/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/ManagedWebAccessAuthTests.java
new file mode 100644
index 000000000..f57d93988
--- /dev/null
+++ b/org.hl7.fhir.utilities/src/test/java/org/hl7/fhir/utilities/http/ManagedWebAccessAuthTests.java
@@ -0,0 +1,216 @@
+package org.hl7.fhir.utilities.http;
+
+import okhttp3.HttpUrl;
+import okhttp3.mockwebserver.MockResponse;
+import okhttp3.mockwebserver.MockWebServer;
+import okhttp3.mockwebserver.RecordedRequest;
+import org.apache.commons.net.util.Base64;
+import org.hl7.fhir.utilities.settings.ServerDetailsPOJO;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+
+import java.io.IOException;
+import java.nio.charset.StandardCharsets;
+import java.util.List;
+
+import static org.assertj.core.api.Assertions.assertThat;
+
+public class ManagedWebAccessAuthTests {
+
+ public static final String DUMMY_AGENT = "dummyAgent";
+ public static final String DUMMY_USERNAME = "dummy1";
+ public static final String DUMMY_PASSWORD = "pass1";
+
+ public static final String DUMMY_TOKEN = "dummyToken";
+ private static final String DUMMY_API_KEY = "dummyApiKey";
+ private MockWebServer server;
+
+ @BeforeEach
+ void setup() {
+ setupMockServer();
+ }
+
+ void setupMockServer() {
+ server = new MockWebServer();
+ }
+
+ @Test
+ public void testBaseCase() throws IOException, InterruptedException {
+ HttpUrl serverUrl = server.url("blah/blah/blah?arg=blah");
+
+ server.enqueue(
+ new MockResponse()
+ .setBody("Dummy Response").setResponseCode(200)
+ );
+
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor("dummyAgent", null);
+ HTTPResult result = builder.httpCall(new HTTPRequest().withUrl(serverUrl.toString()).withMethod(HTTPRequest.HttpMethod.GET));
+
+ assertThat(result.getCode()).isEqualTo(200);
+ assertThat(result.getContentAsString()).isEqualTo("Dummy Response");
+
+ RecordedRequest packageRequest = server.takeRequest();
+
+ assert packageRequest.getRequestUrl() != null;
+ assertExpectedHeaders(packageRequest, serverUrl.url().toString(), "GET");
+
+ }
+
+
+ @Test
+ public void testBasicAuthCase() throws IOException, InterruptedException {
+
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor("dummyAgent", null).withBasicAuth("dummy1", "pass1");
+
+ testBasicServerAuth(builder);
+ }
+
+ private void testBasicServerAuth(ManagedFhirWebAccessor builder) throws IOException, InterruptedException {
+ HttpUrl serverUrl = server.url("blah/blah/blah?arg=blah");
+
+ server.enqueue(
+ new MockResponse()
+ .setBody("Dummy Response").setResponseCode(200)
+ );
+ HTTPResult result = builder.httpCall(new HTTPRequest().withUrl(serverUrl.toString()).withMethod(HTTPRequest.HttpMethod.GET));
+
+ assertThat(result.getCode()).isEqualTo(200);
+ assertThat(result.getContentAsString()).isEqualTo("Dummy Response");
+
+ RecordedRequest packageRequest = server.takeRequest();
+
+ assert packageRequest.getRequestUrl() != null;
+ assertExpectedHeaders(packageRequest, serverUrl.url().toString(), "GET");
+
+ byte[] b = Base64.encodeBase64((DUMMY_USERNAME + ":" + DUMMY_PASSWORD).getBytes(StandardCharsets.US_ASCII));
+ String b64 = new String(b, StandardCharsets.US_ASCII);
+
+ assertThat(packageRequest.getHeader("Authorization")).isEqualTo("Basic " + b64);
+ }
+
+ @Test
+ public void testTokenAuthCase() throws IOException, InterruptedException {
+
+
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor("dummyAgent", null).withToken(DUMMY_TOKEN);
+
+ testTokenAuthCase(builder);
+ }
+
+ private void testTokenAuthCase(ManagedFhirWebAccessor builder) throws IOException, InterruptedException {
+ HttpUrl serverUrl = server.url("blah/blah/blah?arg=blah");
+
+ server.enqueue(
+ new MockResponse()
+ .setBody("Dummy Response").setResponseCode(200)
+ );
+ HTTPResult result = builder.httpCall(new HTTPRequest().withUrl(serverUrl.toString()).withMethod(HTTPRequest.HttpMethod.GET));
+
+ assertThat(result.getCode()).isEqualTo(200);
+ assertThat(result.getContentAsString()).isEqualTo("Dummy Response");
+
+ RecordedRequest packageRequest = server.takeRequest();
+
+ assert packageRequest.getRequestUrl() != null;
+ assertExpectedHeaders(packageRequest, serverUrl.url().toString(), "GET");
+
+ assertThat(packageRequest.getHeader("Authorization")).isEqualTo("Bearer " + DUMMY_TOKEN);
+ }
+
+ private void assertExpectedHeaders(RecordedRequest packageRequest, String expectedUrl, String expectedHttpMethod) {
+ assertThat(packageRequest.getRequestUrl().toString()).isEqualTo(expectedUrl);
+ assertThat(packageRequest.getMethod()).isEqualTo(expectedHttpMethod);
+ assertThat(packageRequest.getHeader("User-Agent")).isEqualTo(DUMMY_AGENT);
+ }
+
+ @Test
+ public void testApiKeyAuthCase() throws IOException, InterruptedException {
+
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor("dummyAgent", null).withApiKey(DUMMY_API_KEY);
+
+ testApiKeyAuthCase(builder);
+ }
+
+ private void testApiKeyAuthCase(ManagedFhirWebAccessor builder) throws IOException, InterruptedException {
+ HttpUrl serverUrl = server.url("blah/blah/blah?arg=blah");
+
+ server.enqueue(
+ new MockResponse()
+ .setBody("Dummy Response").setResponseCode(200)
+ );
+ HTTPResult result = builder.httpCall(new HTTPRequest().withUrl(serverUrl.toString()).withMethod(HTTPRequest.HttpMethod.GET));
+
+ assertThat(result.getCode()).isEqualTo(200);
+ assertThat(result.getContentAsString()).isEqualTo("Dummy Response");
+
+ RecordedRequest packageRequest = server.takeRequest();
+
+ assert packageRequest.getRequestUrl() != null;
+ assertExpectedHeaders(packageRequest, serverUrl.url().toString(), "GET");
+
+ assertThat(packageRequest.getHeader("Api-Key")).isEqualTo(DUMMY_API_KEY);
+ }
+
+ @Test
+ public void testBasicAuthFromSettings() throws IOException, InterruptedException {
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor(
+ "dummyAgent",
+ List.of(getBasicAuthServerPojo()));
+
+ testBasicServerAuth(builder);
+ }
+
+ private ServerDetailsPOJO getBasicAuthServerPojo() {
+ return new ServerDetailsPOJO(
+ server.url("").toString(),
+ "basic",
+ "dummyServerType",
+ DUMMY_USERNAME,
+ DUMMY_PASSWORD,
+ null, null);
+ }
+
+@Test
+public void testTokenAuthFromSettings() throws IOException, InterruptedException {
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor(
+ "dummyAgent",
+ List.of(getTokenAuthServerPojo()));
+
+ testTokenAuthCase(builder);
+}
+
+ private ServerDetailsPOJO getTokenAuthServerPojo() {
+ return new ServerDetailsPOJO(
+ server.url("").toString(),
+ "token",
+ "dummyServerType",
+ null,
+ null,
+ DUMMY_TOKEN, null);
+ }
+
+ @Test
+ public void testApiKeyAuthFromSettings() throws IOException, InterruptedException {
+ ManagedFhirWebAccessor builder = new ManagedFhirWebAccessor(
+ "dummyAgent",
+ List.of(getApiKeyAuthServerPojo()));
+
+ testApiKeyAuthCase(builder);
+ }
+
+ private ServerDetailsPOJO getApiKeyAuthServerPojo() {
+ return new ServerDetailsPOJO(
+ server.url("").toString(),
+ "apikey",
+ "dummyServerType",
+ null,
+ null,
+ null, DUMMY_API_KEY);
+ }
+
+ @Test
+ public void verifyAllowedPaths() {
+ //TODO the allowed paths cannot be set for now, meaning all will be allowed.
+ ManagedWebAccess.inAllowedPaths("http://www.anywhere.com");
+ }
+}
\ No newline at end of file
diff --git a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/ValidatorCli.java b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/ValidatorCli.java
index 3684923b3..ff1d9d4de 100644
--- a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/ValidatorCli.java
+++ b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/ValidatorCli.java
@@ -71,6 +71,7 @@ import org.hl7.fhir.utilities.SystemExitManager;
import org.hl7.fhir.utilities.TimeTracker;
import org.hl7.fhir.utilities.Utilities;
import org.hl7.fhir.utilities.VersionUtilities;
+import org.hl7.fhir.utilities.http.ManagedWebAccess;
import org.hl7.fhir.utilities.settings.FhirSettings;
import org.hl7.fhir.validation.cli.model.CliContext;
import org.hl7.fhir.validation.cli.services.ValidationService;
@@ -151,6 +152,7 @@ public class ValidatorCli {
if (cliContext.getFhirSettingsFile() != null) {
FhirSettings.setExplicitFilePath(cliContext.getFhirSettingsFile());
}
+ ManagedWebAccess.loadFromFHIRSettings();
FileFormat.checkCharsetAndWarnIfNotUTF8(System.out);
diff --git a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/InstanceValidator.java b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/InstanceValidator.java
index 48e0a900e..e8d0912d7 100644
--- a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/InstanceValidator.java
+++ b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/InstanceValidator.java
@@ -1047,7 +1047,8 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
}
private boolean check(String v1, String v2) {
- return v1 == null ? Utilities.noString(v1) : v1.equals(v2);
+ boolean res = v1 == null ? Utilities.noString(v1) : v1.equals(v2);
+ return res;
}
private boolean checkAddress(List errors, String path, Element focus, Address fixed, String fixedSource, boolean pattern, String context) {
@@ -3209,10 +3210,10 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
}
if (context.hasFixed()) {
- ok = checkFixedValue(errors, path, e, context.getFixed(), profile.getVersionedUrl(), context.getSliceName(), null, false, "") && ok;
+ ok = checkFixedValue(errors, path, e, context.getFixed(), profile.getVersionedUrl(), context.getSliceName(), null, false, profile.getVersionedUrl()+"#"+context.getId()) && ok;
}
if (context.hasPattern()) {
- ok = checkFixedValue(errors, path, e, context.getPattern(), profile.getVersionedUrl(), context.getSliceName(), null, true, "") && ok;
+ ok = checkFixedValue(errors, path, e, context.getPattern(), profile.getVersionedUrl(), context.getSliceName(), null, true, profile.getVersionedUrl()+"#"+context.getId()) && ok;
}
if (ok && !ID_EXEMPT_LIST.contains(e.fhirType())) { // ids get checked elsewhere
@@ -5129,12 +5130,13 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
if ("0".equals(criteriaElement.getMax())) {
expression.append(" and " + discriminator + ".empty()");
} else if (s.getType() == DiscriminatorType.TYPE) {
- String type = null;
if (!criteriaElement.getPath().contains("[") && discriminator.contains("[")) {
discriminator = discriminator.substring(0, discriminator.indexOf('['));
String lastNode = tail(discriminator);
- type = makeTypeForFHIRPath(criteriaElement.getPath()).substring(lastNode.length());
+ String type = makeTypeForFHIRPath(criteriaElement.getPath()).substring(lastNode.length());
+ expression.append(" and " + discriminator + " is " + type);
} else if (!criteriaElement.hasType() || criteriaElement.getType().size() == 1) {
+ String type = null;
if (discriminator.contains("["))
discriminator = discriminator.substring(0, discriminator.indexOf('['));
if (criteriaElement.hasType()) {
@@ -5144,23 +5146,25 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
} else {
throw new DefinitionException(context.formatMessage(I18nConstants.DISCRIMINATOR__IS_BASED_ON_TYPE_BUT_SLICE__IN__HAS_NO_TYPES, discriminator, ed.getId(), profile.getVersionedUrl()));
}
+ expression.append(" and " + discriminator + " is " + type);
} else if (criteriaElement.getType().size() > 1) {
- throw new DefinitionException(context.formatMessagePlural(criteriaElement.getType().size(), I18nConstants.DISCRIMINATOR__IS_BASED_ON_TYPE_BUT_SLICE__IN__HAS_MULTIPLE_TYPES, discriminator, ed.getId(), profile.getVersionedUrl(), criteriaElement.typeSummary()));
+ CommaSeparatedStringBuilder cb = new CommaSeparatedStringBuilder(" or ");
+ for (TypeRefComponent tr : criteriaElement.getType()) {
+ String type = makeTypeForFHIRPath(tr.getWorkingCode());
+ cb.append(discriminator + " is " + type);
+ }
+ expression.append(" and (" + cb.toString()+")");
} else
throw new DefinitionException(context.formatMessage(I18nConstants.DISCRIMINATOR__IS_BASED_ON_TYPE_BUT_SLICE__IN__HAS_NO_TYPES, discriminator, ed.getId(), profile.getVersionedUrl()));
- if (discriminator.isEmpty()) {
- expression.append(" and $this is " + type);
- } else {
- expression.append(" and " + discriminator + " is " + type);
- }
} else if (s.getType() == DiscriminatorType.PROFILE) {
if (criteriaElement.getType().size() == 0) {
throw new DefinitionException(context.formatMessage(I18nConstants.PROFILE_BASED_DISCRIMINATORS_MUST_HAVE_A_TYPE__IN_PROFILE_, criteriaElement.getId(), profile.getVersionedUrl()));
}
- if (criteriaElement.getType().size() != 1) {
- throw new DefinitionException(context.formatMessagePlural(criteriaElement.getType().size(), I18nConstants.PROFILE_BASED_DISCRIMINATORS_MUST_HAVE_ONLY_ONE_TYPE__IN_PROFILE, criteriaElement.getId(), profile.getVersionedUrl()));
+ List list = new ArrayList<>();
+ boolean ref = discriminator.endsWith(".resolve()") || discriminator.equals("resolve()");
+ for (TypeRefComponent tr : criteriaElement.getType()) {
+ list.addAll(ref ? tr.getTargetProfile() : tr.getProfile());
}
- List list = discriminator.endsWith(".resolve()") || discriminator.equals("resolve()") ? criteriaElement.getType().get(0).getTargetProfile() : criteriaElement.getType().get(0).getProfile();
if (list.size() == 0) {
// we don't have to find something
// throw new DefinitionException(context.formatMessage(I18nConstants.PROFILE_BASED_DISCRIMINATORS_MUST_HAVE_A_TYPE_WITH_A_PROFILE__IN_PROFILE_, criteriaElement.getId(), profile.getVersionedUrl()));
@@ -6357,10 +6361,10 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
ValidationInfo vi = element.addDefinition(profile, definition, mode);
if (definition.getFixed() != null) {
- ok = checkFixedValue(errors, stack.getLiteralPath(), element, definition.getFixed(), profile.getVersionedUrl(), definition.getSliceName(), null, false, "") && ok;
+ ok = checkFixedValue(errors, stack.getLiteralPath(), element, definition.getFixed(), profile.getVersionedUrl(), definition.getSliceName(), null, false, profile.getVersionedUrl()+"#"+definition.getId()) && ok;
}
if (definition.getPattern() != null) {
- ok = checkFixedValue(errors, stack.getLiteralPath(), element, definition.getPattern(), profile.getVersionedUrl(), definition.getSliceName(), null, true, "") && ok;
+ ok = checkFixedValue(errors, stack.getLiteralPath(), element, definition.getPattern(), profile.getVersionedUrl(), definition.getSliceName(), null, true, profile.getVersionedUrl()+"#"+definition.getId()) && ok;
}
// get the list of direct defined children, including slices
@@ -6644,10 +6648,10 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
ok = checkPrimitive(valContext, errors, ei.getPath(), type, checkDefn, ei.getElement(), profile, localStack, stack, valContext.getRootResource()) && ok;
} else {
if (checkDefn.hasFixed()) {
- ok = checkFixedValue(errors, ei.getPath(), ei.getElement(), checkDefn.getFixed(), profile.getVersionedUrl(), checkDefn.getSliceName(), null, false, "") && ok;
+ ok = checkFixedValue(errors, ei.getPath(), ei.getElement(), checkDefn.getFixed(), profile.getVersionedUrl(), checkDefn.getSliceName(), null, false, profile.getVersionedUrl()+"#"+definition.getId()) && ok;
}
if (checkDefn.hasPattern()) {
- ok = checkFixedValue(errors, ei.getPath(), ei.getElement(), checkDefn.getPattern(), profile.getVersionedUrl(), checkDefn.getSliceName(), null, true, "") && ok;
+ ok = checkFixedValue(errors, ei.getPath(), ei.getElement(), checkDefn.getPattern(), profile.getVersionedUrl(), checkDefn.getSliceName(), null, true, profile.getVersionedUrl()+"#"+definition.getId()) && ok;
}
}
if (type.equals("Identifier")) {
@@ -7600,7 +7604,7 @@ public class InstanceValidator extends BaseValidator implements IResourceValidat
private boolean valueMatchesCriteria(Element value, ElementDefinition criteria, StructureDefinition profile) throws FHIRException {
if (criteria.hasFixed()) {
List msgs = new ArrayList();
- checkFixedValue(msgs, "{virtual}", value, criteria.getFixed(), profile.getVersionedUrl(), "value", null, false, "");
+ checkFixedValue(msgs, "{virtual}", value, criteria.getFixed(), profile.getVersionedUrl(), "value", null, false, profile.getVersionedUrl()+"#"+criteria.getId());
return msgs.size() == 0;
} else if (criteria.hasBinding() && criteria.getBinding().getStrength() == BindingStrength.REQUIRED && criteria.getBinding().hasValueSet()) {
throw new FHIRException(context.formatMessage(I18nConstants.UNABLE_TO_RESOLVE_SLICE_MATCHING__SLICE_MATCHING_BY_VALUE_SET_NOT_DONE));
diff --git a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/MeasureValidator.java b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/MeasureValidator.java
index ce526675e..2333cfc78 100644
--- a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/MeasureValidator.java
+++ b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/MeasureValidator.java
@@ -237,6 +237,17 @@ public class MeasureValidator extends BaseValidator {
NodeStack ns = stack.push(m, -1, m.getProperty().getDefinition(), m.getProperty().getDefinition());
hint(errors, NO_RULE_DATE, IssueType.BUSINESSRULE, m.line(), m.col(), ns.getLiteralPath(), Utilities.existsInList(mc.scoring(), "proportion", "ratio", "continuous-variable", "cohort"), I18nConstants.MEASURE_MR_M_SCORING_UNK);
ok = validateMeasureReportGroups(hostContext, mc, errors, element, stack, inComplete) && ok;
+ } else {
+ if (measure.contains("|")) {
+ List versionList = context.fetchResourcesByUrl(Measure.class, measure.substring(0, measure.indexOf("|")));
+ if (versionList != null && !versionList.isEmpty()) {
+ CommaSeparatedStringBuilder b = new CommaSeparatedStringBuilder();
+ for (Measure mm : versionList) {
+ b.append(mm.getVersion());
+ }
+ hint(errors, NO_RULE_DATE, IssueType.INFORMATIONAL, m.line(), m.col(), stack.getLiteralPath(), msrc != null, I18nConstants.CANONICAL_MULTIPLE_VERSIONS_KNOWN, "Measure", measure, measure.substring(measure.indexOf("|")+1), b.toString());
+ }
+ }
}
}
return ok;
diff --git a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/StructureDefinitionValidator.java b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/StructureDefinitionValidator.java
index 91b6d27f0..235a448a8 100644
--- a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/StructureDefinitionValidator.java
+++ b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/instance/type/StructureDefinitionValidator.java
@@ -16,6 +16,7 @@ import org.hl7.fhir.convertors.factory.VersionConvertorFactory_30_50;
import org.hl7.fhir.convertors.factory.VersionConvertorFactory_40_50;
import org.hl7.fhir.exceptions.DefinitionException;
import org.hl7.fhir.exceptions.FHIRException;
+import org.hl7.fhir.exceptions.PathEngineException;
import org.hl7.fhir.r5.conformance.profile.ProfileUtilities;
import org.hl7.fhir.r5.elementmodel.Element;
import org.hl7.fhir.r5.elementmodel.Manager;
@@ -23,6 +24,7 @@ import org.hl7.fhir.r5.elementmodel.Manager.FhirFormat;
import org.hl7.fhir.r5.extensions.ExtensionConstants;
import org.hl7.fhir.r5.fhirpath.ExpressionNode;
import org.hl7.fhir.r5.fhirpath.ExpressionNode.CollectionStatus;
+import org.hl7.fhir.r5.fhirpath.FHIRLexer.FHIRLexerException;
import org.hl7.fhir.r5.fhirpath.FHIRPathEngine;
import org.hl7.fhir.r5.fhirpath.FHIRPathEngine.IssueMessage;
import org.hl7.fhir.r5.fhirpath.TypeDetails;
@@ -193,7 +195,7 @@ public class StructureDefinitionValidator extends BaseValidator {
}
}
} catch (Exception e) {
- //e.printStackTrace();
+ e.printStackTrace();
rule(errors, NO_RULE_DATE, IssueType.EXCEPTION, stack.getLiteralPath(), false, I18nConstants.ERROR_GENERATING_SNAPSHOT, e.getMessage());
ok = false;
}
@@ -445,6 +447,44 @@ public class StructureDefinitionValidator extends BaseValidator {
addCharacteristics(characteristics, path);
characteristicsValid = true;
}
+
+ if (element.hasChild("slicing")) {
+ Element slicing = element.getNamedChild("slicing");
+ NodeStack sStack = stack.push(slicing, -1, null, null);
+
+ // validating slicing.
+ // slicing can only be present if base cardinality > 1, but we can't always know that on the differential - though we can look it up
+ String tn = path.contains(".") ? path.substring(0, path.indexOf(".")) : path;
+ StructureDefinition tsd = context.fetchTypeDefinition(tn);
+ ElementDefinition ted = null;
+ if (tsd != null) {
+ ted = tsd.getSnapshot().getElementByPath(path);
+ if (ted != null) {
+ ok = rule(errors, "2022-11-02", IssueType.NOTFOUND, sStack, canSlice(ted), I18nConstants.SD_PATH_NO_SLICING, path) && ok;
+ }
+ }
+ int i = 0;
+ for (Element discriminator : slicing.getChildren("discriminator")) {
+ NodeStack dStack = sStack.push(discriminator, i, null, null);
+ String type = discriminator.getNamedChildValue("type");
+ warning(errors, "2024-11-06", IssueType.BUSINESSRULE, dStack, !"pattern".equals(type), I18nConstants.SD_PATH_SLICING_DEPRECATED, type);
+ String pathExp = discriminator.getNamedChildValue("path");
+ if (ted != null) {
+ TypeDetails td = getTypesForElement(elements, element, tn, tsd.getUrl());
+ if (!td.isEmpty()) {
+ List warnings = new ArrayList();
+ try {
+ TypeDetails eval = fpe.checkOnTypes(this, tn, td, fpe.parse(pathExp), warnings, true);
+ if (eval.isEmpty()) {
+ ok = rule(errors, "2024-11-06", IssueType.INVALID, dStack, false, I18nConstants.SD_PATH_NOT_VALID, pathExp, path) && ok;
+ }
+ } catch (Exception e) {
+ ok = rule(errors, "2024-11-06", IssueType.INVALID, dStack, false, I18nConstants.SD_PATH_ERROR, pathExp, path, e.getMessage()) && ok;
+ }
+ }
+ }
+ }
+ }
if (!snapshot && (element.hasChild("fixed") || element.hasChild("pattern")) && base != null) {
ElementDefinition ed = getDefinitionFromBase(base, element.getNamedChildValue("id"), element.getNamedChildValue("path"));
@@ -599,6 +639,10 @@ public class StructureDefinitionValidator extends BaseValidator {
return ok;
}
+ private boolean canSlice(ElementDefinition ted) {
+ return !("1".equals(ted.getMax())) || ted.getPath().contains("[x]");
+ }
+
private boolean checkTypeParameters(List errors, NodeStack stack, Element typeE, String tc, StructureDefinition tsd, String path, StructureDefinition sd) {
boolean ok = true;
if (tsd.hasExtension(ToolingExtensions.EXT_TYPE_PARAMETER)) {
diff --git a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/special/TxTester.java b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/special/TxTester.java
index 36708c84a..896fb2d75 100644
--- a/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/special/TxTester.java
+++ b/org.hl7.fhir.validation/src/main/java/org/hl7/fhir/validation/special/TxTester.java
@@ -4,7 +4,6 @@ import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.DataInputStream;
import java.io.File;
-import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
@@ -26,7 +25,6 @@ import java.util.TimeZone;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
-import org.hl7.fhir.convertors.conv40_50.VersionConvertor_40_50;
import org.hl7.fhir.convertors.factory.VersionConvertorFactory_40_50;
import org.hl7.fhir.convertors.txClient.TerminologyClientFactory;
import org.hl7.fhir.exceptions.DefinitionException;
@@ -41,19 +39,14 @@ import org.hl7.fhir.r5.model.ValueSet;
import org.hl7.fhir.r5.terminologies.client.ITerminologyClient;
import org.hl7.fhir.r5.test.utils.CompareUtilities;
import org.hl7.fhir.r5.utils.client.EFhirClientException;
-import org.hl7.fhir.utilities.CommaSeparatedStringBuilder;
-import org.hl7.fhir.utilities.FhirPublication;
-import org.hl7.fhir.utilities.TextFile;
-import org.hl7.fhir.utilities.Utilities;
-import org.hl7.fhir.utilities.VersionUtil;
+import org.hl7.fhir.utilities.*;
import org.hl7.fhir.utilities.filesystem.ManagedFileAccess;
+import org.hl7.fhir.utilities.http.HTTPHeader;
import org.hl7.fhir.utilities.json.JsonException;
import org.hl7.fhir.utilities.json.model.JsonArray;
import org.hl7.fhir.utilities.json.model.JsonObject;
import org.hl7.fhir.utilities.json.parser.JsonParser;
-import okhttp3.internal.http2.Header;
-
public class TxTester {
@@ -134,7 +127,7 @@ public class TxTester {
return true;
} else {
System.out.println(software+" did not pass all HL7 terminology service tests (modes "+m+", tests v"+loadVersion()+", runner v"+VersionUtil.getBaseVersion()+")");
- System.out.println("Failed Tests: "+CommaSeparatedStringBuilder.join(",", fails ));
+ System.out.println("Failed Tests: "+ CommaSeparatedStringBuilder.join(",", fails ));
return false;
}
} else {
@@ -250,12 +243,12 @@ public class TxTester {
outputT.add("name", test.asString("name"));
if (Utilities.noString(filter) || filter.equals("*") || test.asString("name").contains(filter)) {
System.out.print(" Test "+test.asString("name")+": ");
- Header header = null;
+ HTTPHeader header = null;
try {
if (test.has("header")) {
JsonObject hdr = test.getJsonObject("header");
if (hdr.has("mode") && modes.contains(hdr.asString("mode"))) {
- header = new Header(hdr.asString("name"), hdr.asString("value"));
+ header = new HTTPHeader(hdr.asString("name"), hdr.asString("value"));
tx.getClientHeaders().addHeader(header);
}
}
diff --git a/org.hl7.fhir.validation/src/test/java/ManagedWebAccessAuthTests.java b/org.hl7.fhir.validation/src/test/java/ManagedWebAccessAuthTests.java
new file mode 100644
index 000000000..ce57a67bc
--- /dev/null
+++ b/org.hl7.fhir.validation/src/test/java/ManagedWebAccessAuthTests.java
@@ -0,0 +1,9 @@
+import org.hl7.fhir.utilities.http.ManagedWebAccess;
+import org.junit.BeforeClass;
+
+public class ManagedWebAccessAuthTests {
+ @BeforeClass
+ public static void setUp() {
+ ManagedWebAccess.setUserAgent("hapi-fhir-testing-client");
+ }
+}
diff --git a/pom.xml b/pom.xml
index f6bd4d850..e408832b0 100644
--- a/pom.xml
+++ b/pom.xml
@@ -434,7 +434,7 @@
while if true it will use an executable. -->
true
512m
- 768m
+ 1000m
true
-J-XX:MaxRAMPercentage=50.0
diff --git a/setup-and-cache-job-template.yml b/setup-and-cache-job-template.yml
index a5100e4ed..3469e06e6 100644
--- a/setup-and-cache-job-template.yml
+++ b/setup-and-cache-job-template.yml
@@ -28,7 +28,7 @@ jobs:
options: '--settings $(Agent.TempDirectory)/settings.xml -Dmaven.repo.local=$(MAVEN_CACHE_FOLDER) -DskipTests -DdeployToSonatype -P CHECKSTYLE'
${{ else }}:
options: '-Dmaven.repo.local=$(MAVEN_CACHE_FOLDER) -DskipTests -P CHECKSTYLE'
- mavenOptions: '-Xmx768m'
+ mavenOptions: '-Xmx1G'
javaHomeOption: 'JDKVersion'
jdkVersionOption: '1.11'
jdkArchitectureOption: 'x64'