Add CapabilityStatementFilterFactory to register filters that can modify the capability statement (#5814)
* added new customizer filter factory. have not yet deleted old code it replaces. * added new customizer filter factory. have not yet deleted old code it replaces. * replaced websocket filter. works. still some cleanup to do * replaced websocket filter. works. still some cleanup to do * cosmetic change * add coverage and fix bugs it found * spotless * move capability statement classes * add changelog and rename new classes * review feedback
This commit is contained in:
parent
bdea4b6900
commit
3a5ff47c58
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
type: change
|
||||||
|
issue: 5814
|
||||||
|
title: "Extracted methods out of ResourceProviderFactory into ObservableSupplierSet so that functionality can be used by
|
||||||
|
other services. Unit tests revealed a cleanup bug in MdmProviderLoader that is fixed in this MR."
|
|
@ -32,6 +32,8 @@ import jakarta.annotation.PreDestroy;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.stereotype.Service;
|
import org.springframework.stereotype.Service;
|
||||||
|
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
@Service
|
@Service
|
||||||
public class MdmProviderLoader {
|
public class MdmProviderLoader {
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -58,26 +60,28 @@ public class MdmProviderLoader {
|
||||||
@Autowired
|
@Autowired
|
||||||
private IInterceptorBroadcaster myInterceptorBroadcaster;
|
private IInterceptorBroadcaster myInterceptorBroadcaster;
|
||||||
|
|
||||||
private BaseMdmProvider myMdmProvider;
|
private Supplier<Object> myMdmProviderSupplier;
|
||||||
private MdmLinkHistoryProviderDstu3Plus myMdmHistoryProvider;
|
private Supplier<Object> myMdmHistoryProviderSupplier;
|
||||||
|
|
||||||
public void loadProvider() {
|
public void loadProvider() {
|
||||||
switch (myFhirContext.getVersion().getVersion()) {
|
switch (myFhirContext.getVersion().getVersion()) {
|
||||||
case DSTU3:
|
case DSTU3:
|
||||||
case R4:
|
case R4:
|
||||||
case R5:
|
case R5:
|
||||||
myResourceProviderFactory.addSupplier(() -> new MdmProviderDstu3Plus(
|
// We store the supplier so that removeSupplier works properly
|
||||||
|
myMdmProviderSupplier = () -> new MdmProviderDstu3Plus(
|
||||||
myFhirContext,
|
myFhirContext,
|
||||||
myMdmControllerSvc,
|
myMdmControllerSvc,
|
||||||
myMdmControllerHelper,
|
myMdmControllerHelper,
|
||||||
myMdmSubmitSvc,
|
myMdmSubmitSvc,
|
||||||
myInterceptorBroadcaster,
|
myInterceptorBroadcaster,
|
||||||
myMdmSettings));
|
myMdmSettings);
|
||||||
|
// We store the supplier so that removeSupplier works properly
|
||||||
|
myResourceProviderFactory.addSupplier(myMdmProviderSupplier);
|
||||||
if (myStorageSettings.isNonResourceDbHistoryEnabled()) {
|
if (myStorageSettings.isNonResourceDbHistoryEnabled()) {
|
||||||
myResourceProviderFactory.addSupplier(() -> {
|
myMdmHistoryProviderSupplier = () -> new MdmLinkHistoryProviderDstu3Plus(
|
||||||
return new MdmLinkHistoryProviderDstu3Plus(
|
|
||||||
myFhirContext, myMdmControllerSvc, myInterceptorBroadcaster);
|
myFhirContext, myMdmControllerSvc, myInterceptorBroadcaster);
|
||||||
});
|
myResourceProviderFactory.addSupplier(myMdmHistoryProviderSupplier);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -88,11 +92,11 @@ public class MdmProviderLoader {
|
||||||
|
|
||||||
@PreDestroy
|
@PreDestroy
|
||||||
public void unloadProvider() {
|
public void unloadProvider() {
|
||||||
if (myMdmProvider != null) {
|
if (myMdmProviderSupplier != null) {
|
||||||
myResourceProviderFactory.removeSupplier(() -> myMdmProvider);
|
myResourceProviderFactory.removeSupplier(myMdmProviderSupplier);
|
||||||
}
|
}
|
||||||
if (myMdmHistoryProvider != null) {
|
if (myMdmHistoryProviderSupplier != null) {
|
||||||
myResourceProviderFactory.removeSupplier(() -> myMdmHistoryProvider);
|
myResourceProviderFactory.removeSupplier(myMdmHistoryProviderSupplier);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,14 @@
|
||||||
|
package ca.uhn.fhir.rest.server.provider;
|
||||||
|
|
||||||
|
import jakarta.annotation.Nonnull;
|
||||||
|
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* See {@link ObservableSupplierSet}
|
||||||
|
*/
|
||||||
|
public interface IObservableSupplierSetObserver {
|
||||||
|
void update(@Nonnull Supplier<Object> theSupplier);
|
||||||
|
|
||||||
|
void remove(@Nonnull Supplier<Object> theSupplier);
|
||||||
|
}
|
|
@ -19,12 +19,7 @@
|
||||||
*/
|
*/
|
||||||
package ca.uhn.fhir.rest.server.provider;
|
package ca.uhn.fhir.rest.server.provider;
|
||||||
|
|
||||||
import jakarta.annotation.Nonnull;
|
/**
|
||||||
|
* See {@link ObservableSupplierSet}
|
||||||
import java.util.function.Supplier;
|
*/
|
||||||
|
public interface IResourceProviderFactoryObserver extends IObservableSupplierSetObserver {}
|
||||||
public interface IResourceProviderFactoryObserver {
|
|
||||||
void update(@Nonnull Supplier<Object> theSupplier);
|
|
||||||
|
|
||||||
void remove(@Nonnull Supplier<Object> theSupplier);
|
|
||||||
}
|
|
||||||
|
|
|
@ -0,0 +1,104 @@
|
||||||
|
package ca.uhn.fhir.rest.server.provider;
|
||||||
|
|
||||||
|
import jakarta.annotation.Nonnull;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.LinkedHashSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This is a generic implementation of the <a href="https://refactoring.guru/design-patterns/observer">Observer Design Pattern</a>.
|
||||||
|
* We use this to pass sets of beans from exporting Spring application contexts to importing Spring application contexts. We defer
|
||||||
|
* resolving the observed beans via a Supplier to give the exporting context a chance to initialize the beans before they are used.
|
||||||
|
* @param <T> the class of the Observer
|
||||||
|
* <p>
|
||||||
|
* A typical usage pattern would be:
|
||||||
|
* <ol>
|
||||||
|
* <li>Create {@link ObservableSupplierSet} in exporter context.</li>
|
||||||
|
* <li>Add all the suppliers in the exporter context.</li>
|
||||||
|
* <li>Attach the importer to the {@link ObservableSupplierSet}</li>
|
||||||
|
* <li>Importer calls {@link ObservableSupplierSet#getSupplierResults} and processes all the beans</li>
|
||||||
|
* <li>Some other service beans may add more suppliers later as a part of their initialization and the observer handlers will process them accordingly</li>
|
||||||
|
* <li>Those other service beans should call {@link ObservableSupplierSet#removeSupplier(Supplier)} in a @PreDestroy method so they are properly cleaned up if those services are shut down or restarted</li>
|
||||||
|
* </ol>
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class ObservableSupplierSet<T extends IObservableSupplierSetObserver> {
|
||||||
|
private static final Logger ourLog = LoggerFactory.getLogger(ObservableSupplierSet.class);
|
||||||
|
|
||||||
|
private final Set<T> myObservers = Collections.synchronizedSet(new HashSet<>());
|
||||||
|
|
||||||
|
private final Set<Supplier<Object>> mySuppliers = new LinkedHashSet<>();
|
||||||
|
|
||||||
|
/** Add a supplier and notify all observers
|
||||||
|
*
|
||||||
|
* @param theSupplier supplies the object to be observed
|
||||||
|
*/
|
||||||
|
public void addSupplier(@Nonnull Supplier<Object> theSupplier) {
|
||||||
|
if (mySuppliers.add(theSupplier)) {
|
||||||
|
myObservers.forEach(observer -> observer.update(theSupplier));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Remove a supplier and notify all observers. CAUTION, you might think that this code would work, but it does not:
|
||||||
|
* <code>
|
||||||
|
* observableSupplierSet.addSupplier(() -> myBean);
|
||||||
|
* ...
|
||||||
|
* observableSupplierSet.removeSupplier(() -> myBean);
|
||||||
|
* </code>
|
||||||
|
* the removeSupplier in this example would fail because it is a different lambda instance from the first. Instead,
|
||||||
|
* you need to store the supplier between the add and remove:
|
||||||
|
* <code>
|
||||||
|
* mySupplier = () -> myBean;
|
||||||
|
* observableSupplierSet.addSupplier(mySupplier);
|
||||||
|
* ...
|
||||||
|
* observableSupplierSet.removeSupplier(mySupplier);
|
||||||
|
* </code>
|
||||||
|
*
|
||||||
|
* @param theSupplier the supplier to be removed
|
||||||
|
*/
|
||||||
|
public void removeSupplier(@Nonnull Supplier<Object> theSupplier) {
|
||||||
|
if (mySuppliers.remove(theSupplier)) {
|
||||||
|
myObservers.forEach(observer -> observer.remove(theSupplier));
|
||||||
|
} else {
|
||||||
|
ourLog.warn("Failed to remove supplier", new RuntimeException());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attach an observer to this observableSupplierSet. This observer will be notified every time a supplier is added or removed.
|
||||||
|
* @param theObserver the observer to be notified
|
||||||
|
*/
|
||||||
|
public void attach(T theObserver) {
|
||||||
|
myObservers.add(theObserver);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detach an observer from this observableSupplierSet, so it is no longer notified when suppliers are added and removed.
|
||||||
|
* @param theObserver the observer to be removed
|
||||||
|
*/
|
||||||
|
public void detach(T theObserver) {
|
||||||
|
myObservers.remove(theObserver);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @return a list of get() being called on all suppliers.
|
||||||
|
*/
|
||||||
|
protected List<Object> getSupplierResults() {
|
||||||
|
List<Object> retVal = new ArrayList<>();
|
||||||
|
for (Supplier<Object> next : mySuppliers) {
|
||||||
|
Object nextRp = next.get();
|
||||||
|
if (nextRp != null) {
|
||||||
|
retVal.add(nextRp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,45 +19,16 @@
|
||||||
*/
|
*/
|
||||||
package ca.uhn.fhir.rest.server.provider;
|
package ca.uhn.fhir.rest.server.provider;
|
||||||
|
|
||||||
import jakarta.annotation.Nonnull;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Set;
|
|
||||||
import java.util.function.Supplier;
|
|
||||||
|
|
||||||
public class ResourceProviderFactory {
|
/**
|
||||||
private Set<IResourceProviderFactoryObserver> myObservers = Collections.synchronizedSet(new HashSet<>());
|
* This Factory stores FHIR Resource Provider instance suppliers that will be registered on a FHIR Endpoint later.
|
||||||
private List<Supplier<Object>> mySuppliers = new ArrayList<>();
|
* See {@link ObservableSupplierSet}
|
||||||
|
*/
|
||||||
public void addSupplier(@Nonnull Supplier<Object> theSupplier) {
|
public class ResourceProviderFactory extends ObservableSupplierSet<IResourceProviderFactoryObserver> {
|
||||||
mySuppliers.add(theSupplier);
|
public ResourceProviderFactory() {}
|
||||||
myObservers.forEach(observer -> observer.update(theSupplier));
|
|
||||||
}
|
|
||||||
|
|
||||||
public void removeSupplier(@Nonnull Supplier<Object> theSupplier) {
|
|
||||||
mySuppliers.remove(theSupplier);
|
|
||||||
myObservers.forEach(observer -> observer.remove(theSupplier));
|
|
||||||
}
|
|
||||||
|
|
||||||
public List<Object> createProviders() {
|
public List<Object> createProviders() {
|
||||||
List<Object> retVal = new ArrayList<>();
|
return super.getSupplierResults();
|
||||||
for (Supplier<Object> next : mySuppliers) {
|
|
||||||
Object nextRp = next.get();
|
|
||||||
if (nextRp != null) {
|
|
||||||
retVal.add(nextRp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return retVal;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void attach(IResourceProviderFactoryObserver theObserver) {
|
|
||||||
myObservers.add(theObserver);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void detach(IResourceProviderFactoryObserver theObserver) {
|
|
||||||
myObservers.remove(theObserver);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,102 @@
|
||||||
|
package ca.uhn.fhir.rest.server.provider;
|
||||||
|
|
||||||
|
import ca.uhn.test.util.LogbackCaptureTestExtension;
|
||||||
|
import ch.qos.logback.classic.Level;
|
||||||
|
import ch.qos.logback.classic.spi.ILoggingEvent;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.junit.jupiter.api.extension.RegisterExtension;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
import java.util.function.Supplier;
|
||||||
|
|
||||||
|
import static org.hamcrest.MatcherAssert.assertThat;
|
||||||
|
import static org.hamcrest.Matchers.hasSize;
|
||||||
|
import static org.junit.jupiter.api.Assertions.*;
|
||||||
|
|
||||||
|
class ObservableSupplierSetTest {
|
||||||
|
|
||||||
|
private final ObservableSupplierSet<TestObserver> myObservableSupplierSet = new ObservableSupplierSet<>();
|
||||||
|
private final TestObserver myObserver = new TestObserver();
|
||||||
|
private final AtomicInteger myCounter = new AtomicInteger();
|
||||||
|
|
||||||
|
@RegisterExtension
|
||||||
|
final LogbackCaptureTestExtension myLogger = new LogbackCaptureTestExtension((ch.qos.logback.classic.Logger) LoggerFactory.getLogger(ObservableSupplierSet.class));
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
public void before() {
|
||||||
|
myObservableSupplierSet.attach(myObserver);
|
||||||
|
myObserver.assertCalls(0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void observersNotifiedByAddRemove() {
|
||||||
|
Supplier<Object> supplier = myCounter::incrementAndGet;
|
||||||
|
myObservableSupplierSet.addSupplier(supplier);
|
||||||
|
myObserver.assertCalls(1, 0);
|
||||||
|
assertEquals(0, myCounter.get());
|
||||||
|
assertThat(myObservableSupplierSet.getSupplierResults(), hasSize(1));
|
||||||
|
assertEquals(1, myCounter.get());
|
||||||
|
myObservableSupplierSet.removeSupplier(supplier);
|
||||||
|
myObserver.assertCalls(1, 1);
|
||||||
|
assertThat(myObservableSupplierSet.getSupplierResults(), hasSize(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRemoveWrongSupplier() {
|
||||||
|
myObservableSupplierSet.addSupplier(myCounter::incrementAndGet);
|
||||||
|
myObserver.assertCalls(1, 0);
|
||||||
|
assertEquals(0, myCounter.get());
|
||||||
|
assertThat(myObservableSupplierSet.getSupplierResults(), hasSize(1));
|
||||||
|
assertEquals(1, myCounter.get());
|
||||||
|
|
||||||
|
// You might expect this to remove our supplier, but in fact it is a different lambda, so it fails and logs a stack trace
|
||||||
|
myObservableSupplierSet.removeSupplier(myCounter::incrementAndGet);
|
||||||
|
myObserver.assertCalls(1, 0);
|
||||||
|
assertThat(myObservableSupplierSet.getSupplierResults(), hasSize(1));
|
||||||
|
List<ILoggingEvent> events = myLogger.filterLoggingEventsWithMessageContaining("Failed to remove supplier");
|
||||||
|
assertThat(events, hasSize(1));
|
||||||
|
assertEquals(Level.WARN, events.get(0).getLevel());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDetach() {
|
||||||
|
myObservableSupplierSet.addSupplier(myCounter::incrementAndGet);
|
||||||
|
myObserver.assertCalls(1, 0);
|
||||||
|
assertThat(myObservableSupplierSet.getSupplierResults(), hasSize(1));
|
||||||
|
|
||||||
|
myObservableSupplierSet.addSupplier(myCounter::incrementAndGet);
|
||||||
|
myObserver.assertCalls(2, 0);
|
||||||
|
assertThat(myObservableSupplierSet.getSupplierResults(), hasSize(2));
|
||||||
|
|
||||||
|
myObservableSupplierSet.detach(myObserver);
|
||||||
|
|
||||||
|
// We now have a third supplier but the observer has been detached, so it was not notified of the third supplier
|
||||||
|
myObservableSupplierSet.addSupplier(myCounter::incrementAndGet);
|
||||||
|
myObserver.assertCalls(2, 0);
|
||||||
|
assertThat(myObservableSupplierSet.getSupplierResults(), hasSize(3));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class TestObserver implements IObservableSupplierSetObserver {
|
||||||
|
int updated = 0;
|
||||||
|
int removed = 0;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void update(@NotNull Supplier<Object> theSupplier) {
|
||||||
|
++updated;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void remove(@NotNull Supplier<Object> theSupplier) {
|
||||||
|
++removed;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void assertCalls(int theExpectedUpdated, int theExpectedRemoved) {
|
||||||
|
assertEquals(theExpectedUpdated, updated);
|
||||||
|
assertEquals(theExpectedRemoved, removed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue