Advanced FHIR endpoint parameters

Some more FHIR endpoint parameters are not required in all scenarios.

All parameters are usually set on the FhirTransactionConfiguration (i.e. per FHIR component type), but you may wish to overwrite these settings in your endpoint URI.

Parameters

Parameter name Type Default value Short description
fhirContext FhirContext STU3 or R4 Reference to a global HAPI FHIR Context. Otherwise a default FHIR contxt is used.
resourceProvider FhirProvider n/a Reference to a custom resource provider, configurable per endpoint.
clientRequestFactory ClientRequestFactory n/a reference to a custom ClientRequestFactory
hapiClientInterceptorFactories HapiClientInterceptorFactory n/a reference to custom client interceptor factories
hapiServerInterceptorFactories HapiServerInterceptorFactory n/a reference to custom server interceptor factories
consumerSelector Predicate () -> true reference to a Predicate that selects a FhirConsumer
sort Boolean false whether the endpoint shall try to sort a search result if requested by the client

fhirContext

The most commonly used of these is fhirContext (e.g. ?fhirContext=#fhirContext), which allows to reference a global FHIR Context instance like the one provided by the IPF Spring Boot FHIR starter. This is useful if you expose several FHIR endpoints that shall all use same configured context.

resourceProvider

This is to determine what the HAPI Resource Provider shall be used. Usually, every component statically defines its own resource provider, but for custom searches etc. you may override the default resource provider with this option.

clientRequestFactory

This is to determine what kind of ClientRequestFactory the Producer endpoint shall instantiate for constructing FHIR requests.

consumerSelector

The consumerSelector parameter can be used to determine whether the given endpoint is selected for a FHIR payload or not. This is primarily useful for FHIR batch or transaction requests. These requests are directed at the root URL of the FHIR servlet, but if you need to support more than one type of batch requests via different endpoints you can choose what type shall be handled by this endpoint.

Example:

MHD ITI-65 is a transaction request containing DocumentManifest (MHD 3.2), List, DocumentReference and Binary resources. The request is defined with a dedicated Bundle.meta.profile value (depending on the MHD version), so the transaction definition of ITI-65 statically configures org.openehealth.ipf.commons.ihe.fhir.support.BundleProfileSelector so that ITI-65 consumer endpoints receive these requests:

	public Iti65TransactionConfiguration() {
        super("mhd-iti65",
                "Provide Document Bundle",
                false,
                new Iti65ClientAuditStrategy(),
                new Iti65ServerAuditStrategy(),
                FhirVersionEnum.R4,
                BatchTransactionResourceProvider.getInstance(),      // Consumer side. accept registrations
                BatchTransactionClientRequestFactory.getInstance(),  // Formulate requests
                new Iti65Validator());
    setStaticConsumerSelector(new BundleProfileSelector(
            ITI65_LEGACY_METADATA_PROFILE,
            ITI65_COMPREHENSIVE_METADATA_PROFILE,
            ITI65_MINIMAL_METADATA_PROFILE,
            ITI65_COMPREHENSIVE_BUNDLE_PROFILE,
            ITI65_MINIMAL_BUNDLE_PROFILE,
            ITI65_UNCONTAINED_COMPREHENSIVE_BUNDLE_PROFILE));

If you need to alter this behavior (using a different or no profile for your ITI-65 endpoint), you can do so by using consumerSelector=#mySelector, where mySelector is the name of a bean of type Predicate<Object>.

For demonstration purposes, here is the implementation of BundleProfileSelector:

public class BundleProfileSelector implements Predicate<Object> {

    private final Set<String> profileUris;

    /**
     * @param profileUris Profile URIs expected in the Bundle's meta element
     */
    public BundleProfileSelector(String... profileUris) {
        this.profileUris = new HashSet<>(Arrays.asList(profileUris));
    }

    /**
     * @param object bundle
     * @return true if one of the {@link #profileUris} are present in the Bundle's meta.profile
     */
    @Override
    public boolean test(Object object) {
        Bundle bundle = (Bundle) object;
        return bundle.getMeta().getProfile().stream()
                .map(UriType::getValueAsString)
                .anyMatch(profileUris::contains);
    }
}

sort

This parameter can be set to true if the FHIR consumer endpoint shall attempt server-side orting if the client requests to do so. Sorting must be explicitly supported by the FHIR endpoint’s FhirSearchParameters implementation - it must extend FhirSearchAndSortParameters and implement the comparatorFor(String) method that returns a Comparator object for the provided sort parameter.

For demonstration purposes, here is the implementation of Iti78SearchParameters that supports selected sorting parameters:


public class Iti78SearchParameters extends FhirSearchAndSortParameters<PdqPatient> {

....

    @Override
    public Optional<Comparator<PdqPatient>> comparatorFor(String paramName) {
        return switch (paramName) {
            case PdqPatient.SP_BIRTHDATE -> Optional.of(CP_DATE);
            case PdqPatient.SP_FAMILY -> Optional.of(CP_FAMILY);
            case PdqPatient.SP_GIVEN -> Optional.of(CP_GIVEN);
            default -> Optional.empty();
        };
    }

    private static final Comparator<PdqPatient> CP_DATE = nullsLast(comparing(PdqPatient::getBirthDate));
    private static final Comparator<PdqPatient> CP_FAMILY = nullsLast(comparing(patient -> patient.getNameFirstRep().getFamily()));
    private static final Comparator<PdqPatient> CP_GIVEN = nullsLast(comparing(patient -> patient.getNameFirstRep().getGivenAsSingleString()));
    
}

Any “unknown” sorting parameters are ignored. The returned Comparators must be capable of handling null values - in this case they are ordered after all non-null values.