Messages submitted to the subscription pipeline are lost when submission fails. (#4733)
* One more fix for #4467
* Enabling massIngestionMode causes incomplete resource deletion (#4476)
* Adding initial test.
* Adding fix and subsequent test.
* Adding changelog.
---------
Co-authored-by: peartree <etienne.poirier@smilecdr.com>
* Provide the capability to request that the name of the subscription matching channel be unqualified (#4464)
* Adding initial test.
* Adding initial solution implementation.
* Adding change log and code clean up.
* addressing comments from 1st code review.
---------
Co-authored-by: peartree <etienne.poirier@smilecdr.com>
* Change visibility of migration method (#4471)
* change migration visibility
* add empty migration method for 640
---------
Co-authored-by: nathaniel.doef <nathaniel.doef@smilecdr.com>
* Fix subscription validation not to validate partition ID when invoked from an update pointcut (#4484)
* First commit: Make SubscriptionValidatingInterceptor aware of which Pointcut is being called. In validatePermissions(), skip determinePartition() if the Pointcut is STORAGE_PRESTORAGE_RESOURCE_UPDATED. Fix resulting compile errors in various unit tests.
* Fix/enhance unit tests. Mark methods as deprecated instead of deleting them. Add proper error code. Complete changelog.
* Remove erroneous TODOs and tweak the validation logic.
* Enhance unit tests and fix changelog.
* Reindex batch job fails when processing deleted resources. (#4482)
* adding changelog.
* Providing solution and adding changelog.
* Adding new test.
---------
Co-authored-by: peartree <etienne.poirier@smilecdr.com>
* cleaning up checkstyle files (#4470)
* cleaning up checkstyle files
* One more fix for #4467 (#4469)
* added exlusions for files at base project level so checkstyle doesn't error out
* duplicate error code from merge
* changing lifecycle goal for all module checkstyle check
* moving checkstyle to base pom file, changing exectution phase on base check, cleaning dependency, resolving duplicate error code
* wip
* trying to figure out why pipeline cannot copy files
* removing modules that don't actually need to be built.
* I messed up the version
---------
Co-authored-by: James Agnew <jamesagnew@gmail.com>
* Bump core to 5.6.881 (#4496)
* Bump core to 5.6.881-SNAPSHOT
* Work on fixing tests
* Work on fixing tests 2
* Bump to core release
---------
Co-authored-by: dotasek <david.otasek@smilecdr.com>
* Issue 4486 mdm inconsistent possible match score values (#4487)
* Extract method for readability
* Save always normalized score values in POSSIBLE_MATCH links.
* Avoid setting properties to null values. Adjust test.
* Simplify fix
* Fix test. Add RangeTestHelper.
---------
Co-authored-by: juan.marchionatto <juan.marchionatto@smilecdr.com>
* Revert "cleaning up checkstyle files (#4470)"
This reverts commit efae3b5d5f
.
* core version fix
* Loosen rules for id helper
* initial test implementation/changes and solution wiring.
* adding support for jpa/mongo abstraction.
* refact completed. need to start implementing scheduling of subscription re-submission with the scheduler service.
* post release, merging in master.
* wip: commit before merging in branch master.
* wip: resuming work on the issue.
* wip: before merging in master.
* wip: partial implementation of mongo support.
* wip: more work to support partial implementation on mongo.
* Test implementation
* providing method getMessageKeyOrDefault();
* providing changelog
* preping codereview.
* wip: passing tests
* wip: finally solving the Meta issue.
* wip: adjusting interfaces
* Adding IT and
* IT test;
fixing rollback issue.
* removing non required logging
* fixing merge conflict
* Preparing for code review.
* Preparing for code review.
* Compiling code after merging in master.
* Fixing checkstyle failure.
* Modifications towards getting all tests to pass.
* addressing comments from code review.
* compiling code after switching target merge branch to rel_pub_2023_05
* compiling code after switching target merge branch to rel_6_6
* Addressing comments made in design meeting. wip.
* Addressing comments made in design meeting. finally completed.
and last 2 comments addressed.
* removing non required override
* reverting changes as they are not required.
* Modified test and implementation.
* implementation of a SynchronousSubscriptionMatcherInterceptor to submit modifiedResources to the subscription pipeline for testing only.
* fix tests by modifying the matchingInterceptor to call pointcut when processing
* fixing more tests.
* WIP to pass all tests in jpaserer-4504-a-resource-submitted-to-the-subscription-pipeline-is-lost-when-the-pipeline-encounters-an-issue
* fixing circular dependency
* Adding import dependency to class TestSubscriptionSubmitterConfig to allow submission of messages with the SynchronousSubscriptionMatcherInterceptor.
* bumping version 6.7.7-SNAPSHOT
* bumping HAPI version to 6.7.7-SNAPSHOT
* fixing tests failure caused by asynch deliver of resourceModifiedMsg to the processing pipeline.
* passing all tests.
* unbumping to 6.7.7
* compiling after merging in master.
* creating a custom hapi version
* providing an synchronous subscription matcher interceptor for testing.
* updating pom version to generate artifacts.
* fixing tests, WIP/0
* fixing tests, WIP/1
* fixing tests, WIP/2
* Matching entity annotation with migration task declarations.
* Moving AsynchResourceModified Scheduler and Submitter to hapi-fhir-storage.
* Moving AsynchResourceModified Scheduler and Submitter to hapi-fhir-storage.
* Reverting move of AsyncResourceModified[ProcessingSchedulerSvc|SubmitterSvc].
* Reverting move of AsyncResourceModified[ProcessingSchedulerSvc|SubmitterSvc] again.
* Removing creation of new transaction to inflate PersistedRresourceModifiedMessages since it is already executing in a transaction.
* bumping to version 6.7.14-ASYNC-SNAPSHOT for pipeline.
* bumping hapi version to 6.7.14-ASYNC-SNAPSHOT for pipeline.
* Formatter changes.
* add generic
* add generic
* Addressing code review comments.
* Adding file asynch_subscription_610.md to docs/upgrades
* changing poms to 6.9.2-SNAPSHOT before merging in master.
* compiling after merging in master.
* Lowering logging level to debug.
* Bumping to 6.9.3-SNAPSHOT
* Bumping hapi version to 6.9.4-SNAPSHOT
* Adding update documentation.
* Modifying update file following review.
---------
Co-authored-by: James Agnew <jamesagnew@gmail.com>
Co-authored-by: peartree <etienne.poirier@smilecdr.com>
Co-authored-by: Nathan Doef <n.doef@protonmail.com>
Co-authored-by: nathaniel.doef <nathaniel.doef@smilecdr.com>
Co-authored-by: Luke deGruchy <luke.degruchy@smilecdr.com>
Co-authored-by: Mark Iantorno <markiantorno@gmail.com>
Co-authored-by: dotasek <dotasek.dev@gmail.com>
Co-authored-by: dotasek <david.otasek@smilecdr.com>
Co-authored-by: jmarchionatto <60409882+jmarchionatto@users.noreply.github.com>
Co-authored-by: juan.marchionatto <juan.marchionatto@smilecdr.com>
Co-authored-by: Tadgh <garygrantgraham@gmail.com>
Co-authored-by: Ken Stevens <ken@smilecdr.com>
This commit is contained in:
parent
6a7340a879
commit
fb3e141c4d
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir</artifactId>
|
<artifactId>hapi-fhir</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<modelVersion>4.0.0</modelVersion>
|
<modelVersion>4.0.0</modelVersion>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir-bom</artifactId>
|
<artifactId>hapi-fhir-bom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<packaging>pom</packaging>
|
<packaging>pom</packaging>
|
||||||
<name>HAPI FHIR BOM</name>
|
<name>HAPI FHIR BOM</name>
|
||||||
|
@ -12,7 +12,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir</artifactId>
|
<artifactId>hapi-fhir</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir-cli</artifactId>
|
<artifactId>hapi-fhir-cli</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir</artifactId>
|
<artifactId>hapi-fhir</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir</artifactId>
|
<artifactId>hapi-fhir</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
This release introduces significant a change to the mechanism performing submission of resource modification events
|
||||||
|
to the message broker. Previously, an event would be submitted as part of the synchronous transaction
|
||||||
|
modifying a resource. Synchronous submission yielded responsive publishing with the caveat that events would be dropped
|
||||||
|
upon submission failure.
|
||||||
|
|
||||||
|
We have replaced the synchronous mechanism with a two stage process. Events are initially stored in
|
||||||
|
database upon completion of the transaction and subsequently submitted to the broker by a scheduled task.
|
||||||
|
This new asynchronous submission mechanism will introduce a slight delay in event publishing. It is our view that such
|
||||||
|
delay is largely compensated by the capability to retry submission upon failure which will eliminate event losses.
|
|
@ -11,7 +11,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -57,6 +57,7 @@ import ca.uhn.fhir.jpa.dao.MatchResourceUrlService;
|
||||||
import ca.uhn.fhir.jpa.dao.ObservationLastNIndexPersistSvc;
|
import ca.uhn.fhir.jpa.dao.ObservationLastNIndexPersistSvc;
|
||||||
import ca.uhn.fhir.jpa.dao.SearchBuilderFactory;
|
import ca.uhn.fhir.jpa.dao.SearchBuilderFactory;
|
||||||
import ca.uhn.fhir.jpa.dao.TransactionProcessor;
|
import ca.uhn.fhir.jpa.dao.TransactionProcessor;
|
||||||
|
import ca.uhn.fhir.jpa.dao.data.IResourceModifiedDao;
|
||||||
import ca.uhn.fhir.jpa.dao.data.IResourceSearchUrlDao;
|
import ca.uhn.fhir.jpa.dao.data.IResourceSearchUrlDao;
|
||||||
import ca.uhn.fhir.jpa.dao.expunge.ExpungeEverythingService;
|
import ca.uhn.fhir.jpa.dao.expunge.ExpungeEverythingService;
|
||||||
import ca.uhn.fhir.jpa.dao.expunge.ExpungeOperation;
|
import ca.uhn.fhir.jpa.dao.expunge.ExpungeOperation;
|
||||||
|
@ -155,6 +156,7 @@ import ca.uhn.fhir.jpa.searchparam.extractor.IResourceLinkResolver;
|
||||||
import ca.uhn.fhir.jpa.searchparam.registry.ISearchParamProvider;
|
import ca.uhn.fhir.jpa.searchparam.registry.ISearchParamProvider;
|
||||||
import ca.uhn.fhir.jpa.sp.ISearchParamPresenceSvc;
|
import ca.uhn.fhir.jpa.sp.ISearchParamPresenceSvc;
|
||||||
import ca.uhn.fhir.jpa.sp.SearchParamPresenceSvcImpl;
|
import ca.uhn.fhir.jpa.sp.SearchParamPresenceSvcImpl;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.ResourceModifiedMessagePersistenceSvcImpl;
|
||||||
import ca.uhn.fhir.jpa.term.TermCodeSystemStorageSvcImpl;
|
import ca.uhn.fhir.jpa.term.TermCodeSystemStorageSvcImpl;
|
||||||
import ca.uhn.fhir.jpa.term.TermConceptMappingSvcImpl;
|
import ca.uhn.fhir.jpa.term.TermConceptMappingSvcImpl;
|
||||||
import ca.uhn.fhir.jpa.term.TermReadSvcImpl;
|
import ca.uhn.fhir.jpa.term.TermReadSvcImpl;
|
||||||
|
@ -181,6 +183,7 @@ import ca.uhn.fhir.rest.server.interceptor.ResponseTerminologyTranslationSvc;
|
||||||
import ca.uhn.fhir.rest.server.interceptor.consent.IConsentContextServices;
|
import ca.uhn.fhir.rest.server.interceptor.consent.IConsentContextServices;
|
||||||
import ca.uhn.fhir.rest.server.interceptor.partition.RequestTenantPartitionInterceptor;
|
import ca.uhn.fhir.rest.server.interceptor.partition.RequestTenantPartitionInterceptor;
|
||||||
import ca.uhn.fhir.rest.server.util.ISearchParamRegistry;
|
import ca.uhn.fhir.rest.server.util.ISearchParamRegistry;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import ca.uhn.hapi.converters.canonical.VersionCanonicalizer;
|
import ca.uhn.hapi.converters.canonical.VersionCanonicalizer;
|
||||||
import org.hl7.fhir.common.hapi.validation.support.UnknownCodeSystemWarningValidationSupport;
|
import org.hl7.fhir.common.hapi.validation.support.UnknownCodeSystemWarningValidationSupport;
|
||||||
import org.hl7.fhir.utilities.graphql.IGraphQLStorageServices;
|
import org.hl7.fhir.utilities.graphql.IGraphQLStorageServices;
|
||||||
|
@ -891,4 +894,14 @@ public class JpaConfig {
|
||||||
public IMdmClearHelperSvc<JpaPid> helperSvc(IDeleteExpungeSvc<JpaPid> theDeleteExpungeSvc) {
|
public IMdmClearHelperSvc<JpaPid> helperSvc(IDeleteExpungeSvc<JpaPid> theDeleteExpungeSvc) {
|
||||||
return new MdmClearHelperSvcImpl(theDeleteExpungeSvc);
|
return new MdmClearHelperSvcImpl(theDeleteExpungeSvc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public IResourceModifiedMessagePersistenceSvc subscriptionMessagePersistence(
|
||||||
|
FhirContext theFhirContext,
|
||||||
|
IResourceModifiedDao theIResourceModifiedDao,
|
||||||
|
DaoRegistry theDaoRegistry,
|
||||||
|
HapiTransactionService theHapiTransactionService) {
|
||||||
|
return new ResourceModifiedMessagePersistenceSvcImpl(
|
||||||
|
theFhirContext, theIResourceModifiedDao, theDaoRegistry, theHapiTransactionService);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
package ca.uhn.fhir.jpa.dao.data;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Server
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.PersistedResourceModifiedMessageEntityPK;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.ResourceModifiedEntity;
|
||||||
|
import org.springframework.data.jpa.repository.JpaRepository;
|
||||||
|
import org.springframework.data.jpa.repository.Modifying;
|
||||||
|
import org.springframework.data.jpa.repository.Query;
|
||||||
|
import org.springframework.data.repository.query.Param;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public interface IResourceModifiedDao
|
||||||
|
extends JpaRepository<ResourceModifiedEntity, PersistedResourceModifiedMessageEntityPK>,
|
||||||
|
IHapiFhirJpaRepository {
|
||||||
|
@Query("SELECT r FROM ResourceModifiedEntity r ORDER BY r.myCreatedTime ASC")
|
||||||
|
List<IPersistedResourceModifiedMessage> findAllOrderedByCreatedTime();
|
||||||
|
|
||||||
|
@Modifying
|
||||||
|
@Query("delete from ResourceModifiedEntity r where r.myResourceModifiedEntityPK =:pk")
|
||||||
|
int removeById(@Param("pk") PersistedResourceModifiedMessageEntityPK thePK);
|
||||||
|
}
|
|
@ -437,6 +437,16 @@ public class HapiFhirJpaMigrationTasks extends BaseMigrationTasks<VersionEnum> {
|
||||||
.references(enversRevisionTable, revColumnName);
|
.references(enversRevisionTable, revColumnName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
Builder.BuilderAddTableByColumns resourceModifiedTable =
|
||||||
|
version.addTableByColumns("20230315.1", "HFJ_RESOURCE_MODIFIED", "RES_ID", "RES_VER");
|
||||||
|
resourceModifiedTable.addColumn("RES_ID").nonNullable().type(ColumnTypeEnum.STRING, 256);
|
||||||
|
resourceModifiedTable.addColumn("RES_VER").nonNullable().type(ColumnTypeEnum.STRING, 8);
|
||||||
|
resourceModifiedTable.addColumn("CREATED_TIME").nonNullable().type(ColumnTypeEnum.DATE_TIMESTAMP);
|
||||||
|
resourceModifiedTable.addColumn("SUMMARY_MESSAGE").nonNullable().type(ColumnTypeEnum.STRING, 4000);
|
||||||
|
resourceModifiedTable.addColumn("RESOURCE_TYPE").nonNullable().type(ColumnTypeEnum.STRING, 40);
|
||||||
|
}
|
||||||
|
|
||||||
{
|
{
|
||||||
// The pre-release already contains the long version of this column
|
// The pre-release already contains the long version of this column
|
||||||
// We do this becausea doing a modifyColumn on Postgres (and possibly other RDBMS's) will fail with a nasty
|
// We do this becausea doing a modifyColumn on Postgres (and possibly other RDBMS's) will fail with a nasty
|
||||||
|
|
|
@ -0,0 +1,181 @@
|
||||||
|
package ca.uhn.fhir.jpa.subscription;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Server
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ca.uhn.fhir.context.ConfigurationException;
|
||||||
|
import ca.uhn.fhir.context.FhirContext;
|
||||||
|
import ca.uhn.fhir.i18n.Msg;
|
||||||
|
import ca.uhn.fhir.jpa.api.dao.DaoRegistry;
|
||||||
|
import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao;
|
||||||
|
import ca.uhn.fhir.jpa.dao.data.IResourceModifiedDao;
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.HapiTransactionService;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessagePK;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.PersistedResourceModifiedMessageEntityPK;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.ResourceModifiedEntity;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.async.AsyncResourceModifiedSubmitterSvc;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.model.primitive.IdDt;
|
||||||
|
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import org.hl7.fhir.instance.model.api.IBaseResource;
|
||||||
|
import org.hl7.fhir.instance.model.api.IIdType;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import static ca.uhn.fhir.jpa.model.entity.PersistedResourceModifiedMessageEntityPK.with;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This implementer provides the capability to persist subscription messages for asynchronous submission
|
||||||
|
* to the subscription processing pipeline with the purpose of offering a retry mechanism
|
||||||
|
* upon submission failure (see @link {@link AsyncResourceModifiedSubmitterSvc}).
|
||||||
|
*/
|
||||||
|
public class ResourceModifiedMessagePersistenceSvcImpl implements IResourceModifiedMessagePersistenceSvc {
|
||||||
|
|
||||||
|
private final FhirContext myFhirContext;
|
||||||
|
|
||||||
|
private final IResourceModifiedDao myResourceModifiedDao;
|
||||||
|
|
||||||
|
private final DaoRegistry myDaoRegistry;
|
||||||
|
|
||||||
|
private final ObjectMapper myObjectMapper;
|
||||||
|
|
||||||
|
private final HapiTransactionService myHapiTransactionService;
|
||||||
|
|
||||||
|
private static final Logger ourLog = LoggerFactory.getLogger(ResourceModifiedMessagePersistenceSvcImpl.class);
|
||||||
|
|
||||||
|
public ResourceModifiedMessagePersistenceSvcImpl(
|
||||||
|
FhirContext theFhirContext,
|
||||||
|
IResourceModifiedDao theResourceModifiedDao,
|
||||||
|
DaoRegistry theDaoRegistry,
|
||||||
|
HapiTransactionService theHapiTransactionService) {
|
||||||
|
myFhirContext = theFhirContext;
|
||||||
|
myResourceModifiedDao = theResourceModifiedDao;
|
||||||
|
myDaoRegistry = theDaoRegistry;
|
||||||
|
myHapiTransactionService = theHapiTransactionService;
|
||||||
|
myObjectMapper = new ObjectMapper();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<IPersistedResourceModifiedMessage> findAllOrderedByCreatedTime() {
|
||||||
|
return myHapiTransactionService.withSystemRequest().execute(myResourceModifiedDao::findAllOrderedByCreatedTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public IPersistedResourceModifiedMessage persist(ResourceModifiedMessage theMsg) {
|
||||||
|
ResourceModifiedEntity resourceModifiedEntity = createEntityFrom(theMsg);
|
||||||
|
return myResourceModifiedDao.save(resourceModifiedEntity);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ResourceModifiedMessage inflatePersistedResourceModifiedMessage(
|
||||||
|
IPersistedResourceModifiedMessage thePersistedResourceModifiedMessage) {
|
||||||
|
|
||||||
|
return inflateResourceModifiedMessageFromEntity((ResourceModifiedEntity) thePersistedResourceModifiedMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public long getMessagePersistedCount() {
|
||||||
|
return myResourceModifiedDao.count();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean deleteByPK(IPersistedResourceModifiedMessagePK theResourceModifiedPK) {
|
||||||
|
int removedCount =
|
||||||
|
myResourceModifiedDao.removeById((PersistedResourceModifiedMessageEntityPK) theResourceModifiedPK);
|
||||||
|
|
||||||
|
return removedCount == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected ResourceModifiedMessage inflateResourceModifiedMessageFromEntity(
|
||||||
|
ResourceModifiedEntity theResourceModifiedEntity) {
|
||||||
|
String resourcePid =
|
||||||
|
theResourceModifiedEntity.getResourceModifiedEntityPK().getResourcePid();
|
||||||
|
String resourceVersion =
|
||||||
|
theResourceModifiedEntity.getResourceModifiedEntityPK().getResourceVersion();
|
||||||
|
String resourceType = theResourceModifiedEntity.getResourceType();
|
||||||
|
ResourceModifiedMessage retVal =
|
||||||
|
getPayloadLessMessageFromString(theResourceModifiedEntity.getSummaryResourceModifiedMessage());
|
||||||
|
SystemRequestDetails systemRequestDetails =
|
||||||
|
new SystemRequestDetails().setRequestPartitionId(retVal.getPartitionId());
|
||||||
|
|
||||||
|
IdDt resourceIdDt = new IdDt(resourceType, resourcePid, resourceVersion);
|
||||||
|
IFhirResourceDao dao = myDaoRegistry.getResourceDao(resourceType);
|
||||||
|
|
||||||
|
IBaseResource iBaseResource = dao.read(resourceIdDt, systemRequestDetails, true);
|
||||||
|
|
||||||
|
retVal.setNewPayload(myFhirContext, iBaseResource);
|
||||||
|
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
|
ResourceModifiedEntity createEntityFrom(ResourceModifiedMessage theMsg) {
|
||||||
|
IIdType theMsgId = theMsg.getPayloadId(myFhirContext);
|
||||||
|
|
||||||
|
ResourceModifiedEntity resourceModifiedEntity = new ResourceModifiedEntity();
|
||||||
|
resourceModifiedEntity.setResourceModifiedEntityPK(with(theMsgId.getIdPart(), theMsgId.getVersionIdPart()));
|
||||||
|
|
||||||
|
String partialModifiedMessage = getPayloadLessMessageAsString(theMsg);
|
||||||
|
resourceModifiedEntity.setSummaryResourceModifiedMessage(partialModifiedMessage);
|
||||||
|
resourceModifiedEntity.setResourceType(theMsgId.getResourceType());
|
||||||
|
resourceModifiedEntity.setCreatedTime(new Date());
|
||||||
|
|
||||||
|
return resourceModifiedEntity;
|
||||||
|
}
|
||||||
|
|
||||||
|
private ResourceModifiedMessage getPayloadLessMessageFromString(String thePayloadLessMessage) {
|
||||||
|
try {
|
||||||
|
return myObjectMapper.readValue(thePayloadLessMessage, ResourceModifiedMessage.class);
|
||||||
|
} catch (JsonProcessingException e) {
|
||||||
|
throw new ConfigurationException(Msg.code(2334) + "Failed to json deserialize payloadless message", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getPayloadLessMessageAsString(ResourceModifiedMessage theMsg) {
|
||||||
|
ResourceModifiedMessage tempMessage = new PayloadLessResourceModifiedMessage(theMsg);
|
||||||
|
|
||||||
|
try {
|
||||||
|
return myObjectMapper.writeValueAsString(tempMessage);
|
||||||
|
} catch (JsonProcessingException e) {
|
||||||
|
throw new ConfigurationException(Msg.code(2335) + "Failed to serialize empty ResourceModifiedMessage", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class PayloadLessResourceModifiedMessage extends ResourceModifiedMessage {
|
||||||
|
|
||||||
|
public PayloadLessResourceModifiedMessage(ResourceModifiedMessage theMsg) {
|
||||||
|
this.myPayloadId = theMsg.getPayloadId();
|
||||||
|
this.myPayloadVersion = theMsg.getPayloadVersion();
|
||||||
|
setSubscriptionId(theMsg.getSubscriptionId());
|
||||||
|
setMediaType(theMsg.getMediaType());
|
||||||
|
setOperationType(theMsg.getOperationType());
|
||||||
|
setPartitionId(theMsg.getPartitionId());
|
||||||
|
setTransactionId(theMsg.getTransactionId());
|
||||||
|
setMessageKey(theMsg.getMessageKeyOrNull());
|
||||||
|
copyAdditionalPropertiesFrom(theMsg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -9,6 +9,7 @@ import ca.uhn.fhir.jpa.ips.strategy.DefaultIpsGenerationStrategy;
|
||||||
import ca.uhn.fhir.model.api.ResourceMetadataKeyEnum;
|
import ca.uhn.fhir.model.api.ResourceMetadataKeyEnum;
|
||||||
import ca.uhn.fhir.model.valueset.BundleEntrySearchModeEnum;
|
import ca.uhn.fhir.model.valueset.BundleEntrySearchModeEnum;
|
||||||
import ca.uhn.fhir.rest.api.server.IBundleProvider;
|
import ca.uhn.fhir.rest.api.server.IBundleProvider;
|
||||||
|
import ca.uhn.fhir.rest.api.server.RequestDetails;
|
||||||
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
||||||
import ca.uhn.fhir.rest.param.TokenParam;
|
import ca.uhn.fhir.rest.param.TokenParam;
|
||||||
import ca.uhn.fhir.rest.server.SimpleBundleProvider;
|
import ca.uhn.fhir.rest.server.SimpleBundleProvider;
|
||||||
|
@ -499,7 +500,7 @@ public class IpsGeneratorSvcImplTest {
|
||||||
IFhirResourceDao<Patient> patientDao = registerResourceDaoWithNoData(Patient.class);
|
IFhirResourceDao<Patient> patientDao = registerResourceDaoWithNoData(Patient.class);
|
||||||
Patient patient = new Patient();
|
Patient patient = new Patient();
|
||||||
patient.setId(PATIENT_ID);
|
patient.setId(PATIENT_ID);
|
||||||
when(patientDao.read(any(), any())).thenReturn(patient);
|
when(patientDao.read(any(), any(RequestDetails.class))).thenReturn(patient);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void registerRemainingResourceDaos() {
|
private void registerRemainingResourceDaos() {
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -74,7 +74,7 @@ class MdmSubscriptionLoaderTest {
|
||||||
Subscription subscription = new Subscription();
|
Subscription subscription = new Subscription();
|
||||||
IdType id = new IdType("2401");
|
IdType id = new IdType("2401");
|
||||||
subscription.setIdElement(id);
|
subscription.setIdElement(id);
|
||||||
when(mySubscriptionDao.read(eq(id), any())).thenThrow(new ResourceGoneException(""));
|
when(mySubscriptionDao.read(eq(id), any(RequestDetails.class))).thenThrow(new ResourceGoneException(""));
|
||||||
mySvc.updateIfNotPresent(subscription);
|
mySvc.updateIfNotPresent(subscription);
|
||||||
verify(mySubscriptionDao).update(eq(subscription), any(RequestDetails.class));
|
verify(mySubscriptionDao).update(eq(subscription), any(RequestDetails.class));
|
||||||
}
|
}
|
||||||
|
@ -84,7 +84,7 @@ class MdmSubscriptionLoaderTest {
|
||||||
Subscription subscription = new Subscription();
|
Subscription subscription = new Subscription();
|
||||||
IdType id = new IdType("2401");
|
IdType id = new IdType("2401");
|
||||||
subscription.setIdElement(id);
|
subscription.setIdElement(id);
|
||||||
when(mySubscriptionDao.read(eq(id), any())).thenThrow(new ResourceNotFoundException(""));
|
when(mySubscriptionDao.read(eq(id), any(RequestDetails.class))).thenThrow(new ResourceNotFoundException(""));
|
||||||
mySvc.updateIfNotPresent(subscription);
|
mySvc.updateIfNotPresent(subscription);
|
||||||
verify(mySubscriptionDao).update(eq(subscription), any(RequestDetails.class));
|
verify(mySubscriptionDao).update(eq(subscription), any(RequestDetails.class));
|
||||||
}
|
}
|
||||||
|
@ -94,7 +94,7 @@ class MdmSubscriptionLoaderTest {
|
||||||
Subscription subscription = new Subscription();
|
Subscription subscription = new Subscription();
|
||||||
IdType id = new IdType("2401");
|
IdType id = new IdType("2401");
|
||||||
subscription.setIdElement(id);
|
subscription.setIdElement(id);
|
||||||
when(mySubscriptionDao.read(eq(id), any())).thenReturn(subscription);
|
when(mySubscriptionDao.read(eq(id), any(RequestDetails.class))).thenReturn(subscription);
|
||||||
mySvc.updateIfNotPresent(subscription);
|
mySvc.updateIfNotPresent(subscription);
|
||||||
verify(mySubscriptionDao, never()).update(any(), any(RequestDetails.class));
|
verify(mySubscriptionDao, never()).update(any(), any(RequestDetails.class));
|
||||||
}
|
}
|
||||||
|
@ -106,7 +106,7 @@ class MdmSubscriptionLoaderTest {
|
||||||
when(myMdmSettings.getMdmRules()).thenReturn(mdmRulesJson);
|
when(myMdmSettings.getMdmRules()).thenReturn(mdmRulesJson);
|
||||||
when(myChannelNamer.getChannelName(any(), any())).thenReturn("Test");
|
when(myChannelNamer.getChannelName(any(), any())).thenReturn("Test");
|
||||||
when(myDaoRegistry.getResourceDao(eq("Subscription"))).thenReturn(mySubscriptionDao);
|
when(myDaoRegistry.getResourceDao(eq("Subscription"))).thenReturn(mySubscriptionDao);
|
||||||
when(mySubscriptionDao.read(any(), any())).thenThrow(new ResourceGoneException(""));
|
when(mySubscriptionDao.read(any(), any(RequestDetails.class))).thenThrow(new ResourceGoneException(""));
|
||||||
|
|
||||||
mySvc.daoUpdateMdmSubscriptions();
|
mySvc.daoUpdateMdmSubscriptions();
|
||||||
|
|
||||||
|
|
|
@ -4,12 +4,13 @@ import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
import ca.uhn.fhir.jpa.mdm.helper.MdmHelperR4;
|
import ca.uhn.fhir.jpa.mdm.helper.MdmHelperR4;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.config.SubscriptionChannelConfig;
|
import ca.uhn.fhir.jpa.subscription.channel.config.SubscriptionChannelConfig;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.config.SubscriptionSubmitterConfig;
|
import ca.uhn.fhir.jpa.subscription.submit.config.SubscriptionSubmitterConfig;
|
||||||
|
import ca.uhn.fhir.jpa.test.config.TestSubscriptionMatcherInterceptorConfig;
|
||||||
import org.hl7.fhir.dstu2.model.Subscription;
|
import org.hl7.fhir.dstu2.model.Subscription;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Import;
|
import org.springframework.context.annotation.Import;
|
||||||
import org.springframework.context.annotation.Primary;
|
import org.springframework.context.annotation.Primary;
|
||||||
|
|
||||||
@Import({SubscriptionSubmitterConfig.class, SubscriptionChannelConfig.class})
|
@Import({TestSubscriptionMatcherInterceptorConfig.class, SubscriptionSubmitterConfig.class, SubscriptionChannelConfig.class})
|
||||||
public class TestMdmConfigR4 extends BaseTestMdmConfig {
|
public class TestMdmConfigR4 extends BaseTestMdmConfig {
|
||||||
@Bean
|
@Bean
|
||||||
MdmHelperR4 mdmHelperR4() {
|
MdmHelperR4 mdmHelperR4() {
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -0,0 +1,27 @@
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Model
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
package ca.uhn.fhir.jpa.model.entity;
|
||||||
|
|
||||||
|
public interface IPersistedResourceModifiedMessage {
|
||||||
|
|
||||||
|
IPersistedResourceModifiedMessagePK getPersistedResourceModifiedMessagePk();
|
||||||
|
|
||||||
|
String getResourceType();
|
||||||
|
}
|
|
@ -0,0 +1,27 @@
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Model
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
package ca.uhn.fhir.jpa.model.entity;
|
||||||
|
|
||||||
|
public interface IPersistedResourceModifiedMessagePK {
|
||||||
|
|
||||||
|
String getResourcePid();
|
||||||
|
|
||||||
|
String getResourceVersion();
|
||||||
|
}
|
|
@ -0,0 +1,78 @@
|
||||||
|
package ca.uhn.fhir.jpa.model.entity;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Model
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.Objects;
|
||||||
|
import javax.persistence.Column;
|
||||||
|
import javax.persistence.Embeddable;
|
||||||
|
|
||||||
|
@Embeddable
|
||||||
|
public class PersistedResourceModifiedMessageEntityPK implements IPersistedResourceModifiedMessagePK, Serializable {
|
||||||
|
|
||||||
|
@Column(name = "RES_ID", length = 256, nullable = false)
|
||||||
|
private String myResourcePid;
|
||||||
|
|
||||||
|
@Column(name = "RES_VER", length = 8, nullable = false)
|
||||||
|
private String myResourceVersion;
|
||||||
|
|
||||||
|
public String getResourcePid() {
|
||||||
|
return myResourcePid;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PersistedResourceModifiedMessageEntityPK setResourcePid(String theResourcePid) {
|
||||||
|
myResourcePid = theResourcePid;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getResourceVersion() {
|
||||||
|
return myResourceVersion;
|
||||||
|
}
|
||||||
|
|
||||||
|
public PersistedResourceModifiedMessageEntityPK setResourceVersion(String theResourceVersion) {
|
||||||
|
myResourceVersion = theResourceVersion;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static PersistedResourceModifiedMessageEntityPK with(String theResourcePid, String theResourceVersion) {
|
||||||
|
return new PersistedResourceModifiedMessageEntityPK()
|
||||||
|
.setResourcePid(theResourcePid)
|
||||||
|
.setResourceVersion(theResourceVersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object theO) {
|
||||||
|
if (this == theO) return true;
|
||||||
|
if (theO == null || getClass() != theO.getClass()) return false;
|
||||||
|
PersistedResourceModifiedMessageEntityPK that = (PersistedResourceModifiedMessageEntityPK) theO;
|
||||||
|
return myResourcePid.equals(that.myResourcePid) && myResourceVersion.equals(that.myResourceVersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int hashCode() {
|
||||||
|
return Objects.hash(myResourcePid, myResourceVersion);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return myResourcePid + "/" + myResourceVersion;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,99 @@
|
||||||
|
package ca.uhn.fhir.jpa.model.entity;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Model
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.Date;
|
||||||
|
import javax.persistence.Column;
|
||||||
|
import javax.persistence.EmbeddedId;
|
||||||
|
import javax.persistence.Entity;
|
||||||
|
import javax.persistence.Table;
|
||||||
|
import javax.persistence.Temporal;
|
||||||
|
import javax.persistence.TemporalType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This class describes how a resourceModifiedMessage is stored for later processing in the event where
|
||||||
|
* submission to the subscription processing pipeline would fail. The persisted message does not include a
|
||||||
|
* payload (resource) as an in-memory version of the same message would. Instead, it points to a payload
|
||||||
|
* through the entity primary key {@link PersistedResourceModifiedMessageEntityPK} which is composed
|
||||||
|
* of the resource Pid and current version.
|
||||||
|
*/
|
||||||
|
@Entity
|
||||||
|
@Table(name = "HFJ_RESOURCE_MODIFIED")
|
||||||
|
public class ResourceModifiedEntity implements IPersistedResourceModifiedMessage, Serializable {
|
||||||
|
|
||||||
|
public static final int MESSAGE_LENGTH = 4000;
|
||||||
|
|
||||||
|
@EmbeddedId
|
||||||
|
private PersistedResourceModifiedMessageEntityPK myResourceModifiedEntityPK;
|
||||||
|
|
||||||
|
@Column(name = "SUMMARY_MESSAGE", length = MESSAGE_LENGTH, nullable = false)
|
||||||
|
private String mySummaryResourceModifiedMessage;
|
||||||
|
|
||||||
|
@Column(name = "CREATED_TIME", nullable = false)
|
||||||
|
@Temporal(TemporalType.TIMESTAMP)
|
||||||
|
private Date myCreatedTime;
|
||||||
|
|
||||||
|
@Column(name = "RESOURCE_TYPE", length = ResourceTable.RESTYPE_LEN, nullable = false)
|
||||||
|
private String myResourceType;
|
||||||
|
|
||||||
|
public PersistedResourceModifiedMessageEntityPK getResourceModifiedEntityPK() {
|
||||||
|
return myResourceModifiedEntityPK;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ResourceModifiedEntity setResourceModifiedEntityPK(
|
||||||
|
PersistedResourceModifiedMessageEntityPK theResourceModifiedEntityPK) {
|
||||||
|
myResourceModifiedEntityPK = theResourceModifiedEntityPK;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getResourceType() {
|
||||||
|
return myResourceType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ResourceModifiedEntity setResourceType(String theResourceType) {
|
||||||
|
myResourceType = theResourceType;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date getCreatedTime() {
|
||||||
|
return myCreatedTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setCreatedTime(Date theCreatedTime) {
|
||||||
|
myCreatedTime = theCreatedTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSummaryResourceModifiedMessage() {
|
||||||
|
return mySummaryResourceModifiedMessage;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ResourceModifiedEntity setSummaryResourceModifiedMessage(String theSummaryResourceModifiedMessage) {
|
||||||
|
mySummaryResourceModifiedMessage = theSummaryResourceModifiedMessage;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public IPersistedResourceModifiedMessagePK getPersistedResourceModifiedMessagePk() {
|
||||||
|
return myResourceModifiedEntityPK;
|
||||||
|
}
|
||||||
|
}
|
|
@ -26,6 +26,7 @@ import ca.uhn.fhir.model.api.TemporalPrecisionEnum;
|
||||||
import ca.uhn.fhir.rest.server.interceptor.ResponseTerminologyTranslationSvc;
|
import ca.uhn.fhir.rest.server.interceptor.ResponseTerminologyTranslationSvc;
|
||||||
import ca.uhn.fhir.util.HapiExtensions;
|
import ca.uhn.fhir.util.HapiExtensions;
|
||||||
import com.google.common.annotations.VisibleForTesting;
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import org.apache.commons.collections4.CollectionUtils;
|
||||||
import org.apache.commons.lang3.Validate;
|
import org.apache.commons.lang3.Validate;
|
||||||
import org.hl7.fhir.dstu2.model.Subscription;
|
import org.hl7.fhir.dstu2.model.Subscription;
|
||||||
import org.hl7.fhir.instance.model.api.IPrimitiveType;
|
import org.hl7.fhir.instance.model.api.IPrimitiveType;
|
||||||
|
@ -781,6 +782,15 @@ public class StorageSettings {
|
||||||
return Collections.unmodifiableSet(mySupportedSubscriptionTypes);
|
return Collections.unmodifiableSet(mySupportedSubscriptionTypes);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Indicate whether a subscription channel type is supported by this server.
|
||||||
|
*
|
||||||
|
* @return true if at least one subscription channel type is supported by this server false otherwise.
|
||||||
|
*/
|
||||||
|
public boolean hasSupportedSubscriptionTypes() {
|
||||||
|
return CollectionUtils.isNotEmpty(mySupportedSubscriptionTypes);
|
||||||
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
@VisibleForTesting
|
||||||
public void clearSupportedSubscriptionTypesForUnitTest() {
|
public void clearSupportedSubscriptionTypesForUnitTest() {
|
||||||
mySupportedSubscriptionTypes.clear();
|
mySupportedSubscriptionTypes.clear();
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,66 @@
|
||||||
|
package ca.uhn.fhir.jpa.subscription.async;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR Subscription Server
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.model.sched.HapiJob;
|
||||||
|
import ca.uhn.fhir.jpa.model.sched.IHasScheduledJobs;
|
||||||
|
import ca.uhn.fhir.jpa.model.sched.ISchedulerService;
|
||||||
|
import ca.uhn.fhir.jpa.model.sched.ScheduledJobDefinition;
|
||||||
|
import org.quartz.JobExecutionContext;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This service is responsible for scheduling a job that will submit messages
|
||||||
|
* to the subscription processing pipeline at a given interval.
|
||||||
|
*/
|
||||||
|
public class AsyncResourceModifiedProcessingSchedulerSvc implements IHasScheduledJobs {
|
||||||
|
|
||||||
|
public static final long DEFAULT_SUBMISSION_INTERVAL_IN_MS = 5000;
|
||||||
|
|
||||||
|
public long mySubmissionIntervalInMilliSeconds;
|
||||||
|
|
||||||
|
public AsyncResourceModifiedProcessingSchedulerSvc() {
|
||||||
|
this(DEFAULT_SUBMISSION_INTERVAL_IN_MS);
|
||||||
|
}
|
||||||
|
|
||||||
|
public AsyncResourceModifiedProcessingSchedulerSvc(long theSubmissionIntervalInMilliSeconds) {
|
||||||
|
mySubmissionIntervalInMilliSeconds = theSubmissionIntervalInMilliSeconds;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void scheduleJobs(ISchedulerService theSchedulerService) {
|
||||||
|
ScheduledJobDefinition jobDetail = new ScheduledJobDefinition();
|
||||||
|
jobDetail.setId(getClass().getName());
|
||||||
|
jobDetail.setJobClass(AsyncResourceModifiedProcessingSchedulerSvc.Job.class);
|
||||||
|
|
||||||
|
theSchedulerService.scheduleClusteredJob(mySubmissionIntervalInMilliSeconds, jobDetail);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class Job implements HapiJob {
|
||||||
|
@Autowired
|
||||||
|
private AsyncResourceModifiedSubmitterSvc myAsyncResourceModifiedSubmitterSvc;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void execute(JobExecutionContext theContext) {
|
||||||
|
myAsyncResourceModifiedSubmitterSvc.runDeliveryPass();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,67 @@
|
||||||
|
package ca.uhn.fhir.jpa.subscription.async;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR Subscription Server
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedConsumerWithRetries;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The purpose of this service is to submit messages to the processing pipeline for which previous attempts at
|
||||||
|
* submission has failed. See also {@link AsyncResourceModifiedProcessingSchedulerSvc} and {@link IResourceModifiedMessagePersistenceSvc}.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class AsyncResourceModifiedSubmitterSvc {
|
||||||
|
private static final Logger ourLog = LoggerFactory.getLogger(AsyncResourceModifiedSubmitterSvc.class);
|
||||||
|
|
||||||
|
private final IResourceModifiedMessagePersistenceSvc myResourceModifiedMessagePersistenceSvc;
|
||||||
|
private final IResourceModifiedConsumerWithRetries myResourceModifiedConsumer;
|
||||||
|
|
||||||
|
public AsyncResourceModifiedSubmitterSvc(
|
||||||
|
IResourceModifiedMessagePersistenceSvc theResourceModifiedMessagePersistenceSvc,
|
||||||
|
IResourceModifiedConsumerWithRetries theResourceModifiedConsumer) {
|
||||||
|
myResourceModifiedMessagePersistenceSvc = theResourceModifiedMessagePersistenceSvc;
|
||||||
|
myResourceModifiedConsumer = theResourceModifiedConsumer;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void runDeliveryPass() {
|
||||||
|
|
||||||
|
List<IPersistedResourceModifiedMessage> allPersistedResourceModifiedMessages =
|
||||||
|
myResourceModifiedMessagePersistenceSvc.findAllOrderedByCreatedTime();
|
||||||
|
ourLog.debug(
|
||||||
|
"Attempting to submit {} resources to consumer channel.", allPersistedResourceModifiedMessages.size());
|
||||||
|
|
||||||
|
for (IPersistedResourceModifiedMessage persistedResourceModifiedMessage :
|
||||||
|
allPersistedResourceModifiedMessages) {
|
||||||
|
|
||||||
|
boolean wasProcessed =
|
||||||
|
myResourceModifiedConsumer.submitPersisedResourceModifiedMessage(persistedResourceModifiedMessage);
|
||||||
|
|
||||||
|
if (!wasProcessed) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR Subscription Server
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
package ca.uhn.fhir.jpa.subscription.submit.config;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SynchronousSubscriptionMatcherInterceptor;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.core.env.Environment;
|
||||||
|
|
||||||
|
import static ca.uhn.fhir.jpa.sched.BaseSchedulerServiceImpl.SCHEDULING_DISABLED;
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
public class SubscriptionMatcherInterceptorConfig {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private Environment myEnvironment;
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public SubscriptionMatcherInterceptor subscriptionMatcherInterceptor() {
|
||||||
|
if (isSchedulingDisabledForTests()) {
|
||||||
|
return new SynchronousSubscriptionMatcherInterceptor();
|
||||||
|
}
|
||||||
|
|
||||||
|
return new SubscriptionMatcherInterceptor();
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isSchedulingDisabledForTests() {
|
||||||
|
String schedulingDisabled = myEnvironment.getProperty(SCHEDULING_DISABLED);
|
||||||
|
return "true".equals(schedulingDisabled);
|
||||||
|
}
|
||||||
|
}
|
|
@ -20,14 +20,21 @@
|
||||||
package ca.uhn.fhir.jpa.subscription.submit.config;
|
package ca.uhn.fhir.jpa.subscription.submit.config;
|
||||||
|
|
||||||
import ca.uhn.fhir.jpa.api.dao.DaoRegistry;
|
import ca.uhn.fhir.jpa.api.dao.DaoRegistry;
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.async.AsyncResourceModifiedProcessingSchedulerSvc;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.async.AsyncResourceModifiedSubmitterSvc;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.SubscriptionStrategyEvaluator;
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.SubscriptionStrategyEvaluator;
|
||||||
import ca.uhn.fhir.jpa.subscription.model.config.SubscriptionModelConfig;
|
import ca.uhn.fhir.jpa.subscription.model.config.SubscriptionModelConfig;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionQueryValidator;
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionQueryValidator;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionSubmitInterceptorLoader;
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionSubmitInterceptorLoader;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionValidatingInterceptor;
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionValidatingInterceptor;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.subscription.triggering.ISubscriptionTriggeringSvc;
|
import ca.uhn.fhir.jpa.subscription.triggering.ISubscriptionTriggeringSvc;
|
||||||
import ca.uhn.fhir.jpa.subscription.triggering.SubscriptionTriggeringSvcImpl;
|
import ca.uhn.fhir.jpa.subscription.triggering.SubscriptionTriggeringSvcImpl;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedConsumerWithRetries;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Configuration;
|
import org.springframework.context.annotation.Configuration;
|
||||||
import org.springframework.context.annotation.Import;
|
import org.springframework.context.annotation.Import;
|
||||||
|
@ -38,14 +45,9 @@ import org.springframework.context.annotation.Lazy;
|
||||||
* matching queue for processing
|
* matching queue for processing
|
||||||
*/
|
*/
|
||||||
@Configuration
|
@Configuration
|
||||||
@Import(SubscriptionModelConfig.class)
|
@Import({SubscriptionModelConfig.class, SubscriptionMatcherInterceptorConfig.class})
|
||||||
public class SubscriptionSubmitterConfig {
|
public class SubscriptionSubmitterConfig {
|
||||||
|
|
||||||
@Bean
|
|
||||||
public SubscriptionMatcherInterceptor subscriptionMatcherInterceptor() {
|
|
||||||
return new SubscriptionMatcherInterceptor();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
public SubscriptionValidatingInterceptor subscriptionValidatingInterceptor() {
|
public SubscriptionValidatingInterceptor subscriptionValidatingInterceptor() {
|
||||||
return new SubscriptionValidatingInterceptor();
|
return new SubscriptionValidatingInterceptor();
|
||||||
|
@ -67,4 +69,31 @@ public class SubscriptionSubmitterConfig {
|
||||||
public ISubscriptionTriggeringSvc subscriptionTriggeringSvc() {
|
public ISubscriptionTriggeringSvc subscriptionTriggeringSvc() {
|
||||||
return new SubscriptionTriggeringSvcImpl();
|
return new SubscriptionTriggeringSvcImpl();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public ResourceModifiedSubmitterSvc resourceModifiedSvc(
|
||||||
|
IHapiTransactionService theHapiTransactionService,
|
||||||
|
IResourceModifiedMessagePersistenceSvc theResourceModifiedMessagePersistenceSvc,
|
||||||
|
SubscriptionChannelFactory theSubscriptionChannelFactory,
|
||||||
|
StorageSettings theStorageSettings) {
|
||||||
|
|
||||||
|
return new ResourceModifiedSubmitterSvc(
|
||||||
|
theStorageSettings,
|
||||||
|
theSubscriptionChannelFactory,
|
||||||
|
theResourceModifiedMessagePersistenceSvc,
|
||||||
|
theHapiTransactionService);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public AsyncResourceModifiedProcessingSchedulerSvc asyncResourceModifiedProcessingSchedulerSvc() {
|
||||||
|
return new AsyncResourceModifiedProcessingSchedulerSvc();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public AsyncResourceModifiedSubmitterSvc asyncResourceModifiedSubmitterSvc(
|
||||||
|
IResourceModifiedMessagePersistenceSvc theIResourceModifiedMessagePersistenceSvc,
|
||||||
|
IResourceModifiedConsumerWithRetries theResourceModifiedConsumer) {
|
||||||
|
return new AsyncResourceModifiedSubmitterSvc(
|
||||||
|
theIResourceModifiedMessagePersistenceSvc, theResourceModifiedConsumer);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,31 +28,26 @@ import ca.uhn.fhir.interceptor.api.Pointcut;
|
||||||
import ca.uhn.fhir.interceptor.model.RequestPartitionId;
|
import ca.uhn.fhir.interceptor.model.RequestPartitionId;
|
||||||
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc;
|
import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.api.ChannelProducerSettings;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.impl.LinkedBlockingChannel;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.match.matcher.subscriber.SubscriptionMatchingSubscriber;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedJsonMessage;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
import ca.uhn.fhir.rest.api.server.RequestDetails;
|
import ca.uhn.fhir.rest.api.server.RequestDetails;
|
||||||
|
import ca.uhn.fhir.rest.server.messaging.BaseResourceMessage;
|
||||||
import ca.uhn.fhir.rest.server.util.CompositeInterceptorBroadcaster;
|
import ca.uhn.fhir.rest.server.util.CompositeInterceptorBroadcaster;
|
||||||
import com.google.common.annotations.VisibleForTesting;
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import org.apache.commons.lang3.Validate;
|
|
||||||
import org.hl7.fhir.instance.model.api.IBaseResource;
|
import org.hl7.fhir.instance.model.api.IBaseResource;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
import org.springframework.context.event.ContextRefreshedEvent;
|
|
||||||
import org.springframework.context.event.EventListener;
|
|
||||||
import org.springframework.messaging.MessageChannel;
|
|
||||||
import org.springframework.transaction.support.TransactionSynchronizationAdapter;
|
|
||||||
import org.springframework.transaction.support.TransactionSynchronizationManager;
|
|
||||||
|
|
||||||
import static org.apache.commons.lang3.StringUtils.isNotBlank;
|
import static java.util.Objects.isNull;
|
||||||
|
import static org.apache.commons.lang3.StringUtils.isBlank;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* This interceptor is responsible for submitting operations on resources to the subscription pipeline.
|
||||||
|
*
|
||||||
|
*/
|
||||||
@Interceptor
|
@Interceptor
|
||||||
public class SubscriptionMatcherInterceptor implements IResourceModifiedConsumer {
|
public class SubscriptionMatcherInterceptor {
|
||||||
private static final Logger ourLog = LoggerFactory.getLogger(SubscriptionMatcherInterceptor.class);
|
private static final Logger ourLog = LoggerFactory.getLogger(SubscriptionMatcherInterceptor.class);
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -61,16 +56,14 @@ public class SubscriptionMatcherInterceptor implements IResourceModifiedConsumer
|
||||||
@Autowired
|
@Autowired
|
||||||
private IInterceptorBroadcaster myInterceptorBroadcaster;
|
private IInterceptorBroadcaster myInterceptorBroadcaster;
|
||||||
|
|
||||||
@Autowired
|
|
||||||
private SubscriptionChannelFactory mySubscriptionChannelFactory;
|
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private StorageSettings myStorageSettings;
|
private StorageSettings myStorageSettings;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private IRequestPartitionHelperSvc myRequestPartitionHelperSvc;
|
private IRequestPartitionHelperSvc myRequestPartitionHelperSvc;
|
||||||
|
|
||||||
private volatile MessageChannel myMatchingChannel;
|
@Autowired
|
||||||
|
private IResourceModifiedMessagePersistenceSvc myResourceModifiedMessagePersistenceSvc;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor
|
* Constructor
|
||||||
|
@ -79,122 +72,91 @@ public class SubscriptionMatcherInterceptor implements IResourceModifiedConsumer
|
||||||
super();
|
super();
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventListener(classes = {ContextRefreshedEvent.class})
|
|
||||||
public void startIfNeeded() {
|
|
||||||
if (myStorageSettings.getSupportedSubscriptionTypes().isEmpty()) {
|
|
||||||
ourLog.debug(
|
|
||||||
"Subscriptions are disabled on this server. Skipping {} channel creation.",
|
|
||||||
SubscriptionMatchingSubscriber.SUBSCRIPTION_MATCHING_CHANNEL_NAME);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (myMatchingChannel == null) {
|
|
||||||
myMatchingChannel = mySubscriptionChannelFactory.newMatchingSendingChannel(
|
|
||||||
SubscriptionMatchingSubscriber.SUBSCRIPTION_MATCHING_CHANNEL_NAME, getChannelProducerSettings());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Hook(Pointcut.STORAGE_PRECOMMIT_RESOURCE_CREATED)
|
@Hook(Pointcut.STORAGE_PRECOMMIT_RESOURCE_CREATED)
|
||||||
public void resourceCreated(IBaseResource theResource, RequestDetails theRequest) {
|
public void resourceCreated(IBaseResource theResource, RequestDetails theRequest) {
|
||||||
startIfNeeded();
|
|
||||||
submitResourceModified(theResource, ResourceModifiedMessage.OperationTypeEnum.CREATE, theRequest);
|
processResourceModifiedEvent(theResource, ResourceModifiedMessage.OperationTypeEnum.CREATE, theRequest);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Hook(Pointcut.STORAGE_PRECOMMIT_RESOURCE_DELETED)
|
@Hook(Pointcut.STORAGE_PRECOMMIT_RESOURCE_DELETED)
|
||||||
public void resourceDeleted(IBaseResource theResource, RequestDetails theRequest) {
|
public void resourceDeleted(IBaseResource theResource, RequestDetails theRequest) {
|
||||||
startIfNeeded();
|
|
||||||
submitResourceModified(theResource, ResourceModifiedMessage.OperationTypeEnum.DELETE, theRequest);
|
processResourceModifiedEvent(theResource, ResourceModifiedMessage.OperationTypeEnum.DELETE, theRequest);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Hook(Pointcut.STORAGE_PRECOMMIT_RESOURCE_UPDATED)
|
@Hook(Pointcut.STORAGE_PRECOMMIT_RESOURCE_UPDATED)
|
||||||
public void resourceUpdated(IBaseResource theOldResource, IBaseResource theNewResource, RequestDetails theRequest) {
|
public void resourceUpdated(IBaseResource theOldResource, IBaseResource theNewResource, RequestDetails theRequest) {
|
||||||
startIfNeeded();
|
boolean dontTriggerSubscriptionWhenVersionsAreTheSame =
|
||||||
if (!myStorageSettings.isTriggerSubscriptionsForNonVersioningChanges()) {
|
!myStorageSettings.isTriggerSubscriptionsForNonVersioningChanges();
|
||||||
if (theOldResource != null && theNewResource != null) {
|
boolean resourceVersionsAreTheSame = isSameResourceVersion(theOldResource, theNewResource);
|
||||||
String oldVersion = theOldResource.getIdElement().getVersionIdPart();
|
|
||||||
String newVersion = theNewResource.getIdElement().getVersionIdPart();
|
if (dontTriggerSubscriptionWhenVersionsAreTheSame && resourceVersionsAreTheSame) {
|
||||||
if (isNotBlank(oldVersion) && isNotBlank(newVersion) && oldVersion.equals(newVersion)) {
|
return;
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
submitResourceModified(theNewResource, ResourceModifiedMessage.OperationTypeEnum.UPDATE, theRequest);
|
processResourceModifiedEvent(theNewResource, ResourceModifiedMessage.OperationTypeEnum.UPDATE, theRequest);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is an internal API - Use with caution!
|
* This is an internal API - Use with caution!
|
||||||
|
*
|
||||||
|
* This method will create a {@link ResourceModifiedMessage}, persist it and arrange for its delivery to the
|
||||||
|
* subscription pipeline after the resource was committed. The message is persisted to provide asynchronous submission
|
||||||
|
* in the event where submission would fail.
|
||||||
*/
|
*/
|
||||||
@Override
|
protected void processResourceModifiedEvent(
|
||||||
public void submitResourceModified(
|
|
||||||
IBaseResource theNewResource,
|
IBaseResource theNewResource,
|
||||||
ResourceModifiedMessage.OperationTypeEnum theOperationType,
|
ResourceModifiedMessage.OperationTypeEnum theOperationType,
|
||||||
RequestDetails theRequest) {
|
RequestDetails theRequest) {
|
||||||
// Even though the resource is being written, the subscription will be interacting with it by effectively
|
|
||||||
// "reading" it so we set the RequestPartitionId as a read request
|
ResourceModifiedMessage msg = createResourceModifiedMessage(theNewResource, theOperationType, theRequest);
|
||||||
RequestPartitionId requestPartitionId = myRequestPartitionHelperSvc.determineReadPartitionForRequestForRead(
|
|
||||||
theRequest, theNewResource.getIdElement().getResourceType(), theNewResource.getIdElement());
|
|
||||||
ResourceModifiedMessage msg = new ResourceModifiedMessage(
|
|
||||||
myFhirContext, theNewResource, theOperationType, theRequest, requestPartitionId);
|
|
||||||
|
|
||||||
// Interceptor call: SUBSCRIPTION_RESOURCE_MODIFIED
|
// Interceptor call: SUBSCRIPTION_RESOURCE_MODIFIED
|
||||||
HookParams params = new HookParams().add(ResourceModifiedMessage.class, msg);
|
HookParams params = new HookParams().add(ResourceModifiedMessage.class, msg);
|
||||||
boolean outcome = CompositeInterceptorBroadcaster.doCallHooks(
|
boolean outcome = CompositeInterceptorBroadcaster.doCallHooks(
|
||||||
myInterceptorBroadcaster, theRequest, Pointcut.SUBSCRIPTION_RESOURCE_MODIFIED, params);
|
myInterceptorBroadcaster, theRequest, Pointcut.SUBSCRIPTION_RESOURCE_MODIFIED, params);
|
||||||
|
|
||||||
if (!outcome) {
|
if (!outcome) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
submitResourceModified(msg);
|
processResourceModifiedMessage(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
protected void processResourceModifiedMessage(ResourceModifiedMessage theResourceModifiedMessage) {
|
||||||
* This is an internal API - Use with caution!
|
// persist the message for async submission to the processing pipeline. see {@link
|
||||||
*/
|
// AsyncResourceModifiedProcessingSchedulerSvc}
|
||||||
@Override
|
myResourceModifiedMessagePersistenceSvc.persist(theResourceModifiedMessage);
|
||||||
public void submitResourceModified(final ResourceModifiedMessage theMsg) {
|
}
|
||||||
/*
|
|
||||||
* We only want to submit the message to the processing queue once the
|
|
||||||
* transaction is committed. We do this in order to make sure that the
|
|
||||||
* data is actually in the DB, in case it's the database matcher.
|
|
||||||
*/
|
|
||||||
if (TransactionSynchronizationManager.isSynchronizationActive()) {
|
|
||||||
TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronizationAdapter() {
|
|
||||||
@Override
|
|
||||||
public int getOrder() {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
protected ResourceModifiedMessage createResourceModifiedMessage(
|
||||||
public void afterCommit() {
|
IBaseResource theNewResource,
|
||||||
sendToProcessingChannel(theMsg);
|
BaseResourceMessage.OperationTypeEnum theOperationType,
|
||||||
}
|
RequestDetails theRequest) {
|
||||||
});
|
// Even though the resource is being written, the subscription will be interacting with it by effectively
|
||||||
} else {
|
// "reading" it so we set the RequestPartitionId as a read request
|
||||||
sendToProcessingChannel(theMsg);
|
RequestPartitionId requestPartitionId = myRequestPartitionHelperSvc.determineReadPartitionForRequestForRead(
|
||||||
|
theRequest, theNewResource.getIdElement().getResourceType(), theNewResource.getIdElement());
|
||||||
|
return new ResourceModifiedMessage(
|
||||||
|
myFhirContext, theNewResource, theOperationType, theRequest, requestPartitionId);
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isSameResourceVersion(IBaseResource theOldResource, IBaseResource theNewResource) {
|
||||||
|
if (isNull(theOldResource) || isNull(theNewResource)) {
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
protected void sendToProcessingChannel(final ResourceModifiedMessage theMessage) {
|
String oldVersion = theOldResource.getIdElement().getVersionIdPart();
|
||||||
ourLog.trace("Sending resource modified message to processing channel");
|
String newVersion = theNewResource.getIdElement().getVersionIdPart();
|
||||||
Validate.notNull(
|
|
||||||
myMatchingChannel,
|
|
||||||
"A SubscriptionMatcherInterceptor has been registered without calling start() on it.");
|
|
||||||
myMatchingChannel.send(new ResourceModifiedJsonMessage(theMessage));
|
|
||||||
}
|
|
||||||
|
|
||||||
private ChannelProducerSettings getChannelProducerSettings() {
|
if (isBlank(oldVersion) || isBlank(newVersion)) {
|
||||||
ChannelProducerSettings channelProducerSettings = new ChannelProducerSettings();
|
return false;
|
||||||
channelProducerSettings.setQualifyChannelName(myStorageSettings.isQualifySubscriptionMatchingChannelName());
|
}
|
||||||
return channelProducerSettings;
|
|
||||||
|
return oldVersion.equals(newVersion);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setFhirContext(FhirContext theCtx) {
|
public void setFhirContext(FhirContext theCtx) {
|
||||||
myFhirContext = theCtx;
|
myFhirContext = theCtx;
|
||||||
}
|
}
|
||||||
|
|
||||||
@VisibleForTesting
|
|
||||||
public LinkedBlockingChannel getProcessingChannelForUnitTest() {
|
|
||||||
startIfNeeded();
|
|
||||||
return (LinkedBlockingChannel) myMatchingChannel;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,59 @@
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR Subscription Server
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
package ca.uhn.fhir.jpa.subscription.submit.interceptor;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.subscription.async.AsyncResourceModifiedProcessingSchedulerSvc;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.transaction.support.TransactionSynchronizationAdapter;
|
||||||
|
import org.springframework.transaction.support.TransactionSynchronizationManager;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The purpose of this interceptor is to synchronously submit ResourceModifiedMessage to the
|
||||||
|
* subscription processing pipeline, ie, as part of processing the operation on a resource.
|
||||||
|
* It is meant to replace the SubscriptionMatcherInterceptor in integrated tests where
|
||||||
|
* scheduling is disabled. See {@link AsyncResourceModifiedProcessingSchedulerSvc}
|
||||||
|
* for further details on asynchronous submissions.
|
||||||
|
*/
|
||||||
|
public class SynchronousSubscriptionMatcherInterceptor extends SubscriptionMatcherInterceptor {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private IResourceModifiedConsumer myResourceModifiedConsumer;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void processResourceModifiedMessage(ResourceModifiedMessage theResourceModifiedMessage) {
|
||||||
|
if (TransactionSynchronizationManager.isSynchronizationActive()) {
|
||||||
|
TransactionSynchronizationManager.registerSynchronization(new TransactionSynchronizationAdapter() {
|
||||||
|
@Override
|
||||||
|
public int getOrder() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void afterCommit() {
|
||||||
|
myResourceModifiedConsumer.submitResourceModified(theResourceModifiedMessage);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
myResourceModifiedConsumer.submitResourceModified(theResourceModifiedMessage);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,230 @@
|
||||||
|
package ca.uhn.fhir.jpa.subscription.submit.svc;
|
||||||
|
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR Subscription Server
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessagePK;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.api.ChannelProducerSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelProducer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedJsonMessage;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
||||||
|
import ca.uhn.fhir.rest.server.exceptions.ResourceNotFoundException;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedConsumerWithRetries;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
|
import org.apache.commons.lang3.Validate;
|
||||||
|
import org.hl7.fhir.r5.model.IdType;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.context.event.ContextRefreshedEvent;
|
||||||
|
import org.springframework.context.event.EventListener;
|
||||||
|
import org.springframework.messaging.MessageChannel;
|
||||||
|
import org.springframework.messaging.MessageDeliveryException;
|
||||||
|
import org.springframework.transaction.annotation.Propagation;
|
||||||
|
import org.springframework.transaction.support.TransactionCallback;
|
||||||
|
|
||||||
|
import java.util.Optional;
|
||||||
|
|
||||||
|
import static ca.uhn.fhir.jpa.subscription.match.matcher.subscriber.SubscriptionMatchingSubscriber.SUBSCRIPTION_MATCHING_CHANNEL_NAME;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This service provides two distinct contexts in which it submits messages to the subscription pipeline.
|
||||||
|
*
|
||||||
|
* It implements {@link IResourceModifiedConsumer} for synchronous submissions where retry upon failures is not required.
|
||||||
|
*
|
||||||
|
* It implements {@link IResourceModifiedConsumerWithRetries} for synchronous submissions performed as part of processing
|
||||||
|
* an operation on a resource (see {@link SubscriptionMatcherInterceptor}). Submissions in such context require retries
|
||||||
|
* upon submission failure.
|
||||||
|
*
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class ResourceModifiedSubmitterSvc implements IResourceModifiedConsumer, IResourceModifiedConsumerWithRetries {
|
||||||
|
|
||||||
|
private static final Logger ourLog = LoggerFactory.getLogger(ResourceModifiedSubmitterSvc.class);
|
||||||
|
private volatile MessageChannel myMatchingChannel;
|
||||||
|
|
||||||
|
private final StorageSettings myStorageSettings;
|
||||||
|
private final SubscriptionChannelFactory mySubscriptionChannelFactory;
|
||||||
|
private final IResourceModifiedMessagePersistenceSvc myResourceModifiedMessagePersistenceSvc;
|
||||||
|
private final IHapiTransactionService myHapiTransactionService;
|
||||||
|
|
||||||
|
@EventListener(classes = {ContextRefreshedEvent.class})
|
||||||
|
public void startIfNeeded() {
|
||||||
|
if (!myStorageSettings.hasSupportedSubscriptionTypes()) {
|
||||||
|
ourLog.debug(
|
||||||
|
"Subscriptions are disabled on this server. Skipping {} channel creation.",
|
||||||
|
SUBSCRIPTION_MATCHING_CHANNEL_NAME);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (myMatchingChannel == null) {
|
||||||
|
myMatchingChannel = mySubscriptionChannelFactory.newMatchingSendingChannel(
|
||||||
|
SUBSCRIPTION_MATCHING_CHANNEL_NAME, getChannelProducerSettings());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public ResourceModifiedSubmitterSvc(
|
||||||
|
StorageSettings theStorageSettings,
|
||||||
|
SubscriptionChannelFactory theSubscriptionChannelFactory,
|
||||||
|
IResourceModifiedMessagePersistenceSvc resourceModifiedMessagePersistenceSvc,
|
||||||
|
IHapiTransactionService theHapiTransactionService) {
|
||||||
|
myStorageSettings = theStorageSettings;
|
||||||
|
mySubscriptionChannelFactory = theSubscriptionChannelFactory;
|
||||||
|
myResourceModifiedMessagePersistenceSvc = resourceModifiedMessagePersistenceSvc;
|
||||||
|
myHapiTransactionService = theHapiTransactionService;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @inheritDoc
|
||||||
|
* Submit a message to the broker without retries.
|
||||||
|
*
|
||||||
|
* Implementation of the {@link IResourceModifiedConsumer}
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void submitResourceModified(ResourceModifiedMessage theMsg) {
|
||||||
|
startIfNeeded();
|
||||||
|
|
||||||
|
ourLog.trace("Sending resource modified message to processing channel");
|
||||||
|
Validate.notNull(
|
||||||
|
myMatchingChannel,
|
||||||
|
"A SubscriptionMatcherInterceptor has been registered without calling start() on it.");
|
||||||
|
myMatchingChannel.send(new ResourceModifiedJsonMessage(theMsg));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method will inflate the ResourceModifiedMessage represented by the IPersistedResourceModifiedMessage and attempts
|
||||||
|
* to submit it to the subscription processing pipeline.
|
||||||
|
*
|
||||||
|
* If submission succeeds, the IPersistedResourceModifiedMessage is deleted and true is returned. In the event where submission
|
||||||
|
* fails, we return false and the IPersistedResourceModifiedMessage is rollback for later re-submission.
|
||||||
|
*
|
||||||
|
* @param thePersistedResourceModifiedMessage A ResourceModifiedMessage in it's IPersistedResourceModifiedMessage that requires submission.
|
||||||
|
* @return Whether the message was successfully submitted to the broker.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public boolean submitPersisedResourceModifiedMessage(
|
||||||
|
IPersistedResourceModifiedMessage thePersistedResourceModifiedMessage) {
|
||||||
|
return myHapiTransactionService
|
||||||
|
.withSystemRequest()
|
||||||
|
.withPropagation(Propagation.REQUIRES_NEW)
|
||||||
|
.execute(doProcessResourceModifiedInTransaction(thePersistedResourceModifiedMessage));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method is the cornerstone in the submit and retry upon failure mechanism for messages needing submission to the subscription processing pipeline.
|
||||||
|
* It requires execution in a transaction for rollback of deleting the persistedResourceModifiedMessage pointed to by <code>thePersistedResourceModifiedMessage<code/>
|
||||||
|
* in the event where submission would fail.
|
||||||
|
*
|
||||||
|
* @param thePersistedResourceModifiedMessage the primary key pointing to the persisted version (IPersistedResourceModifiedMessage) of a ResourceModifiedMessage needing submission
|
||||||
|
* @return true upon successful submission, false otherwise.
|
||||||
|
*/
|
||||||
|
protected TransactionCallback<Boolean> doProcessResourceModifiedInTransaction(
|
||||||
|
IPersistedResourceModifiedMessage thePersistedResourceModifiedMessage) {
|
||||||
|
return theStatus -> {
|
||||||
|
boolean processed = true;
|
||||||
|
ResourceModifiedMessage resourceModifiedMessage = null;
|
||||||
|
try {
|
||||||
|
|
||||||
|
// delete the entry to lock the row to ensure unique processing
|
||||||
|
boolean wasDeleted = deletePersistedResourceModifiedMessage(
|
||||||
|
thePersistedResourceModifiedMessage.getPersistedResourceModifiedMessagePk());
|
||||||
|
|
||||||
|
Optional<ResourceModifiedMessage> optionalResourceModifiedMessage =
|
||||||
|
inflatePersistedResourceMessage(thePersistedResourceModifiedMessage);
|
||||||
|
|
||||||
|
if (wasDeleted && optionalResourceModifiedMessage.isPresent()) {
|
||||||
|
// the PK did exist and we were able to deleted it, ie, we are the only one processing the message
|
||||||
|
resourceModifiedMessage = optionalResourceModifiedMessage.get();
|
||||||
|
submitResourceModified(resourceModifiedMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (MessageDeliveryException exception) {
|
||||||
|
// we encountered an issue when trying to send the message so mark the transaction for rollback
|
||||||
|
ourLog.error(
|
||||||
|
"Channel submission failed for resource with id {} matching subscription with id {}. Further attempts will be performed at later time.",
|
||||||
|
resourceModifiedMessage.getPayloadId(),
|
||||||
|
resourceModifiedMessage.getSubscriptionId());
|
||||||
|
processed = false;
|
||||||
|
theStatus.setRollbackOnly();
|
||||||
|
}
|
||||||
|
|
||||||
|
return processed;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private Optional<ResourceModifiedMessage> inflatePersistedResourceMessage(
|
||||||
|
IPersistedResourceModifiedMessage thePersistedResourceModifiedMessage) {
|
||||||
|
ResourceModifiedMessage resourceModifiedMessage = null;
|
||||||
|
|
||||||
|
try {
|
||||||
|
|
||||||
|
resourceModifiedMessage = myResourceModifiedMessagePersistenceSvc.inflatePersistedResourceModifiedMessage(
|
||||||
|
thePersistedResourceModifiedMessage);
|
||||||
|
|
||||||
|
} catch (ResourceNotFoundException e) {
|
||||||
|
IPersistedResourceModifiedMessagePK persistedResourceModifiedMessagePk =
|
||||||
|
thePersistedResourceModifiedMessage.getPersistedResourceModifiedMessagePk();
|
||||||
|
|
||||||
|
IdType idType = new IdType(
|
||||||
|
thePersistedResourceModifiedMessage.getResourceType(),
|
||||||
|
persistedResourceModifiedMessagePk.getResourcePid(),
|
||||||
|
persistedResourceModifiedMessagePk.getResourceVersion());
|
||||||
|
|
||||||
|
ourLog.warn(
|
||||||
|
"Scheduled submission will be ignored since resource {} cannot be found", idType.asStringValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
return Optional.ofNullable(resourceModifiedMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean deletePersistedResourceModifiedMessage(IPersistedResourceModifiedMessagePK theResourceModifiedPK) {
|
||||||
|
|
||||||
|
try {
|
||||||
|
// delete the entry to lock the row to ensure unique processing
|
||||||
|
return myResourceModifiedMessagePersistenceSvc.deleteByPK(theResourceModifiedPK);
|
||||||
|
} catch (ResourceNotFoundException exception) {
|
||||||
|
ourLog.warn(
|
||||||
|
"thePersistedResourceModifiedMessage with {} and version {} could not be deleted as it may have already been deleted.",
|
||||||
|
theResourceModifiedPK.getResourcePid(),
|
||||||
|
theResourceModifiedPK.getResourceVersion());
|
||||||
|
// we were not able to delete the pk. this implies that someone else did read/delete the PK and processed
|
||||||
|
// the message
|
||||||
|
// successfully before we did.
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private ChannelProducerSettings getChannelProducerSettings() {
|
||||||
|
ChannelProducerSettings channelProducerSettings = new ChannelProducerSettings();
|
||||||
|
channelProducerSettings.setQualifyChannelName(myStorageSettings.isQualifySubscriptionMatchingChannelName());
|
||||||
|
return channelProducerSettings;
|
||||||
|
}
|
||||||
|
|
||||||
|
public IChannelProducer getProcessingChannelForUnitTest() {
|
||||||
|
startIfNeeded();
|
||||||
|
return (IChannelProducer) myMatchingChannel;
|
||||||
|
}
|
||||||
|
}
|
|
@ -39,7 +39,7 @@ import java.util.function.Function;
|
||||||
* This interceptor can be used for troubleshooting subscription processing. It provides very
|
* This interceptor can be used for troubleshooting subscription processing. It provides very
|
||||||
* detailed logging about the subscription processing pipeline.
|
* detailed logging about the subscription processing pipeline.
|
||||||
* <p>
|
* <p>
|
||||||
* This interceptor loges each step in the processing pipeline with a
|
* This interceptor logs each step in the processing pipeline with a
|
||||||
* different event code, using the event codes itemized in
|
* different event code, using the event codes itemized in
|
||||||
* {@link EventCodeEnum}. By default these are each placed in a logger with
|
* {@link EventCodeEnum}. By default these are each placed in a logger with
|
||||||
* a different name (e.g. <code>ca.uhn.fhir.jpa.subscription.util.SubscriptionDebugLogInterceptor.SUBS20</code>
|
* a different name (e.g. <code>ca.uhn.fhir.jpa.subscription.util.SubscriptionDebugLogInterceptor.SUBS20</code>
|
||||||
|
@ -91,7 +91,7 @@ public class SubscriptionDebugLogInterceptor {
|
||||||
}
|
}
|
||||||
log(
|
log(
|
||||||
EventCodeEnum.SUBS1,
|
EventCodeEnum.SUBS1,
|
||||||
"Resource {} was submitted to the processing pipeline (op={})",
|
"Resource {} is starting the processing pipeline (op={})",
|
||||||
resourceId,
|
resourceId,
|
||||||
theMessage.getOperationType());
|
theMessage.getOperationType());
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,60 +0,0 @@
|
||||||
package ca.uhn.fhir.jpa.subscription.submit.interceptor;
|
|
||||||
|
|
||||||
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.api.ChannelProducerSettings;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
|
||||||
import org.junit.jupiter.api.extension.ExtendWith;
|
|
||||||
import org.junit.jupiter.params.ParameterizedTest;
|
|
||||||
import org.junit.jupiter.params.provider.ValueSource;
|
|
||||||
import org.mockito.ArgumentCaptor;
|
|
||||||
import org.mockito.Captor;
|
|
||||||
import org.mockito.InjectMocks;
|
|
||||||
import org.mockito.Mock;
|
|
||||||
import org.mockito.junit.jupiter.MockitoExtension;
|
|
||||||
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import static org.hamcrest.MatcherAssert.assertThat;
|
|
||||||
import static org.hamcrest.Matchers.is;
|
|
||||||
import static org.hl7.fhir.dstu2.model.Subscription.SubscriptionChannelType.RESTHOOK;
|
|
||||||
import static org.mockito.ArgumentMatchers.anyString;
|
|
||||||
import static org.mockito.Mockito.verify;
|
|
||||||
import static org.mockito.Mockito.when;
|
|
||||||
|
|
||||||
|
|
||||||
@ExtendWith(MockitoExtension.class)
|
|
||||||
public class SubscriptionMatcherInterceptorTest {
|
|
||||||
|
|
||||||
@Mock
|
|
||||||
StorageSettings myStorageSettings;
|
|
||||||
@Mock
|
|
||||||
SubscriptionChannelFactory mySubscriptionChannelFactory;
|
|
||||||
@InjectMocks
|
|
||||||
SubscriptionMatcherInterceptor myUnitUnderTest;
|
|
||||||
@Captor
|
|
||||||
ArgumentCaptor<ChannelProducerSettings> myArgumentCaptor;
|
|
||||||
|
|
||||||
@ParameterizedTest
|
|
||||||
@ValueSource(booleans = {false, true})
|
|
||||||
public void testMethodStartIfNeeded_withQualifySubscriptionMatchingChannelNameProperty_mayQualifyChannelName(boolean theIsQualifySubMatchingChannelName){
|
|
||||||
// given
|
|
||||||
boolean expectedResult = theIsQualifySubMatchingChannelName;
|
|
||||||
when(myStorageSettings.isQualifySubscriptionMatchingChannelName()).thenReturn(theIsQualifySubMatchingChannelName);
|
|
||||||
when(myStorageSettings.getSupportedSubscriptionTypes()).thenReturn(Set.of(RESTHOOK));
|
|
||||||
|
|
||||||
// when
|
|
||||||
myUnitUnderTest.startIfNeeded();
|
|
||||||
|
|
||||||
// then
|
|
||||||
ChannelProducerSettings capturedChannelProducerSettings = getCapturedChannelProducerSettings();
|
|
||||||
assertThat(capturedChannelProducerSettings.isQualifyChannelName(), is(expectedResult));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private ChannelProducerSettings getCapturedChannelProducerSettings(){
|
|
||||||
verify(mySubscriptionChannelFactory).newMatchingSendingChannel(anyString(), myArgumentCaptor.capture());
|
|
||||||
return myArgumentCaptor.getValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
|
@ -6,12 +6,14 @@ import ca.uhn.fhir.interceptor.api.IInterceptorService;
|
||||||
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
import ca.uhn.fhir.jpa.api.dao.DaoRegistry;
|
import ca.uhn.fhir.jpa.api.dao.DaoRegistry;
|
||||||
import ca.uhn.fhir.jpa.cache.IResourceVersionSvc;
|
import ca.uhn.fhir.jpa.cache.IResourceVersionSvc;
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
import ca.uhn.fhir.jpa.model.config.PartitionSettings;
|
import ca.uhn.fhir.jpa.model.config.PartitionSettings;
|
||||||
import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc;
|
import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc;
|
||||||
import ca.uhn.fhir.jpa.searchparam.config.SearchParamConfig;
|
import ca.uhn.fhir.jpa.searchparam.config.SearchParamConfig;
|
||||||
import ca.uhn.fhir.jpa.searchparam.registry.ISearchParamProvider;
|
import ca.uhn.fhir.jpa.searchparam.registry.ISearchParamProvider;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.config.SubscriptionSubmitterConfig;
|
import ca.uhn.fhir.jpa.subscription.submit.config.SubscriptionSubmitterConfig;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import org.hl7.fhir.dstu2.model.Subscription;
|
import org.hl7.fhir.dstu2.model.Subscription;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.api.extension.ExtendWith;
|
import org.junit.jupiter.api.extension.ExtendWith;
|
||||||
|
@ -21,6 +23,7 @@ import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Configuration;
|
import org.springframework.context.annotation.Configuration;
|
||||||
import org.springframework.test.context.ContextConfiguration;
|
import org.springframework.test.context.ContextConfiguration;
|
||||||
import org.springframework.test.context.junit.jupiter.SpringExtension;
|
import org.springframework.test.context.junit.jupiter.SpringExtension;
|
||||||
|
import org.springframework.transaction.PlatformTransactionManager;
|
||||||
|
|
||||||
import static org.mockito.ArgumentMatchers.eq;
|
import static org.mockito.ArgumentMatchers.eq;
|
||||||
import static org.mockito.Mockito.times;
|
import static org.mockito.Mockito.times;
|
||||||
|
@ -34,24 +37,12 @@ import static org.mockito.Mockito.verify;
|
||||||
})
|
})
|
||||||
public class SubscriptionSubmitInterceptorLoaderTest {
|
public class SubscriptionSubmitInterceptorLoaderTest {
|
||||||
|
|
||||||
@MockBean
|
|
||||||
private ISearchParamProvider mySearchParamProvider;
|
|
||||||
@MockBean
|
|
||||||
private IInterceptorService myInterceptorService;
|
|
||||||
@MockBean
|
|
||||||
private IValidationSupport myValidationSupport;
|
|
||||||
@MockBean
|
|
||||||
private SubscriptionChannelFactory mySubscriptionChannelFactory;
|
|
||||||
@MockBean
|
|
||||||
private DaoRegistry myDaoRegistry;
|
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionSubmitInterceptorLoader mySubscriptionSubmitInterceptorLoader;
|
private SubscriptionSubmitInterceptorLoader mySubscriptionSubmitInterceptorLoader;
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
private SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
||||||
@MockBean
|
@MockBean
|
||||||
private IResourceVersionSvc myResourceVersionSvc;
|
private IInterceptorService myInterceptorService;
|
||||||
@MockBean
|
|
||||||
private IRequestPartitionHelperSvc myRequestPartitionHelperSvc;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* It should be possible to run only the {@link SubscriptionSubmitterConfig} without the
|
* It should be possible to run only the {@link SubscriptionSubmitterConfig} without the
|
||||||
|
@ -82,6 +73,25 @@ public class SubscriptionSubmitInterceptorLoaderTest {
|
||||||
return storageSettings;
|
return storageSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@MockBean
|
||||||
|
private ISearchParamProvider mySearchParamProvider;
|
||||||
|
@MockBean
|
||||||
|
private IValidationSupport myValidationSupport;
|
||||||
|
@MockBean
|
||||||
|
private SubscriptionChannelFactory mySubscriptionChannelFactory;
|
||||||
|
@MockBean
|
||||||
|
private DaoRegistry myDaoRegistry;
|
||||||
|
@MockBean
|
||||||
|
private IResourceVersionSvc myResourceVersionSvc;
|
||||||
|
@MockBean
|
||||||
|
private IRequestPartitionHelperSvc myRequestPartitionHelperSvc;
|
||||||
|
@MockBean
|
||||||
|
private PlatformTransactionManager myPlatformTransactionManager;
|
||||||
|
@MockBean
|
||||||
|
private IResourceModifiedMessagePersistenceSvc myResourceModifiedMessagePersistenceSvc;
|
||||||
|
@MockBean
|
||||||
|
private IHapiTransactionService myHapiTransactionService;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -6,6 +6,7 @@ import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao;
|
||||||
import ca.uhn.fhir.jpa.searchparam.matcher.InMemoryMatchResult;
|
import ca.uhn.fhir.jpa.searchparam.matcher.InMemoryMatchResult;
|
||||||
import ca.uhn.fhir.jpa.searchparam.matcher.SearchParamMatcher;
|
import ca.uhn.fhir.jpa.searchparam.matcher.SearchParamMatcher;
|
||||||
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.rest.api.server.RequestDetails;
|
||||||
import org.hl7.fhir.r5.model.Encounter;
|
import org.hl7.fhir.r5.model.Encounter;
|
||||||
import org.hl7.fhir.r5.model.IdType;
|
import org.hl7.fhir.r5.model.IdType;
|
||||||
import org.hl7.fhir.r5.model.SubscriptionTopic;
|
import org.hl7.fhir.r5.model.SubscriptionTopic;
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -9,6 +9,7 @@ import ca.uhn.fhir.jpa.dao.SearchBuilderFactory;
|
||||||
import ca.uhn.fhir.jpa.dao.tx.HapiTransactionService;
|
import ca.uhn.fhir.jpa.dao.tx.HapiTransactionService;
|
||||||
import ca.uhn.fhir.jpa.model.dao.JpaPid;
|
import ca.uhn.fhir.jpa.model.dao.JpaPid;
|
||||||
import ca.uhn.fhir.jpa.search.builder.SearchBuilder;
|
import ca.uhn.fhir.jpa.search.builder.SearchBuilder;
|
||||||
|
import ca.uhn.fhir.jpa.svc.MockHapiTransactionService;
|
||||||
import ca.uhn.fhir.jpa.util.BaseIterator;
|
import ca.uhn.fhir.jpa.util.BaseIterator;
|
||||||
import ca.uhn.fhir.model.dstu2.resource.Patient;
|
import ca.uhn.fhir.model.dstu2.resource.Patient;
|
||||||
import ca.uhn.fhir.rest.api.server.storage.IResourcePersistentId;
|
import ca.uhn.fhir.rest.api.server.storage.IResourcePersistentId;
|
||||||
|
|
|
@ -210,7 +210,6 @@ public class RestHookTestWithInterceptorRegisteredToStorageSettingsDstu2Test ext
|
||||||
Subscription subscription1 = createSubscription(criteria1, payload, ourListenerServerBase);
|
Subscription subscription1 = createSubscription(criteria1, payload, ourListenerServerBase);
|
||||||
Subscription subscription2 = createSubscription(criteria2, payload, ourListenerServerBase);
|
Subscription subscription2 = createSubscription(criteria2, payload, ourListenerServerBase);
|
||||||
|
|
||||||
|
|
||||||
runInTransaction(() -> {
|
runInTransaction(() -> {
|
||||||
ourLog.info("All token indexes:\n * {}", myResourceIndexedSearchParamTokenDao.findAll().stream().map(t -> t.toString()).collect(Collectors.joining("\n * ")));
|
ourLog.info("All token indexes:\n * {}", myResourceIndexedSearchParamTokenDao.findAll().stream().map(t -> t.toString()).collect(Collectors.joining("\n * ")));
|
||||||
});
|
});
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -16,8 +16,8 @@ import ca.uhn.fhir.jpa.model.dao.JpaPid;
|
||||||
import ca.uhn.fhir.jpa.model.entity.ForcedId;
|
import ca.uhn.fhir.jpa.model.entity.ForcedId;
|
||||||
import ca.uhn.fhir.jpa.model.entity.ResourceTable;
|
import ca.uhn.fhir.jpa.model.entity.ResourceTable;
|
||||||
import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc;
|
import ca.uhn.fhir.jpa.partition.IRequestPartitionHelperSvc;
|
||||||
import ca.uhn.fhir.jpa.search.MockHapiTransactionService;
|
|
||||||
import ca.uhn.fhir.jpa.searchparam.SearchParameterMap;
|
import ca.uhn.fhir.jpa.searchparam.SearchParameterMap;
|
||||||
|
import ca.uhn.fhir.jpa.svc.MockHapiTransactionService;
|
||||||
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
||||||
import ca.uhn.fhir.rest.api.server.RequestDetails;
|
import ca.uhn.fhir.rest.api.server.RequestDetails;
|
||||||
import ca.uhn.fhir.rest.api.server.storage.TransactionDetails;
|
import ca.uhn.fhir.rest.api.server.storage.TransactionDetails;
|
||||||
|
|
|
@ -25,7 +25,7 @@ import ca.uhn.fhir.jpa.model.util.JpaConstants;
|
||||||
import ca.uhn.fhir.jpa.provider.BaseResourceProviderR4Test;
|
import ca.uhn.fhir.jpa.provider.BaseResourceProviderR4Test;
|
||||||
import ca.uhn.fhir.jpa.search.PersistedJpaSearchFirstPageBundleProvider;
|
import ca.uhn.fhir.jpa.search.PersistedJpaSearchFirstPageBundleProvider;
|
||||||
import ca.uhn.fhir.jpa.searchparam.SearchParameterMap;
|
import ca.uhn.fhir.jpa.searchparam.SearchParameterMap;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.subscription.triggering.ISubscriptionTriggeringSvc;
|
import ca.uhn.fhir.jpa.subscription.triggering.ISubscriptionTriggeringSvc;
|
||||||
import ca.uhn.fhir.jpa.term.TermReadSvcImpl;
|
import ca.uhn.fhir.jpa.term.TermReadSvcImpl;
|
||||||
import ca.uhn.fhir.jpa.util.SqlQuery;
|
import ca.uhn.fhir.jpa.util.SqlQuery;
|
||||||
|
@ -126,6 +126,7 @@ import static org.mockito.Mockito.when;
|
||||||
@SuppressWarnings("JavadocBlankLines")
|
@SuppressWarnings("JavadocBlankLines")
|
||||||
@TestMethodOrder(MethodOrderer.MethodName.class)
|
@TestMethodOrder(MethodOrderer.MethodName.class)
|
||||||
public class FhirResourceDaoR4QueryCountTest extends BaseResourceProviderR4Test {
|
public class FhirResourceDaoR4QueryCountTest extends BaseResourceProviderR4Test {
|
||||||
|
|
||||||
@RegisterExtension
|
@RegisterExtension
|
||||||
@Order(0)
|
@Order(0)
|
||||||
public static final RestfulServerExtension ourServer = new RestfulServerExtension(FhirContext.forR4Cached())
|
public static final RestfulServerExtension ourServer = new RestfulServerExtension(FhirContext.forR4Cached())
|
||||||
|
@ -139,7 +140,7 @@ public class FhirResourceDaoR4QueryCountTest extends BaseResourceProviderR4Test
|
||||||
@Autowired
|
@Autowired
|
||||||
private ISubscriptionTriggeringSvc mySubscriptionTriggeringSvc;
|
private ISubscriptionTriggeringSvc mySubscriptionTriggeringSvc;
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
private ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;;
|
||||||
@Autowired
|
@Autowired
|
||||||
private ReindexStep myReindexStep;
|
private ReindexStep myReindexStep;
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -3090,7 +3091,7 @@ public class FhirResourceDaoR4QueryCountTest extends BaseResourceProviderR4Test
|
||||||
// Setup
|
// Setup
|
||||||
|
|
||||||
myStorageSettings.addSupportedSubscriptionType(org.hl7.fhir.dstu2.model.Subscription.SubscriptionChannelType.RESTHOOK);
|
myStorageSettings.addSupportedSubscriptionType(org.hl7.fhir.dstu2.model.Subscription.SubscriptionChannelType.RESTHOOK);
|
||||||
mySubscriptionMatcherInterceptor.startIfNeeded();
|
myResourceModifiedSubmitterSvc.startIfNeeded();
|
||||||
|
|
||||||
for (int i = 0; i < 10; i++) {
|
for (int i = 0; i < 10; i++) {
|
||||||
createPatient(withActiveTrue());
|
createPatient(withActiveTrue());
|
||||||
|
|
|
@ -2,12 +2,14 @@ package ca.uhn.fhir.jpa.subscription;
|
||||||
|
|
||||||
import ca.uhn.fhir.context.FhirContext;
|
import ca.uhn.fhir.context.FhirContext;
|
||||||
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.dao.data.IResourceModifiedDao;
|
||||||
import ca.uhn.fhir.jpa.provider.BaseResourceProviderR4Test;
|
import ca.uhn.fhir.jpa.provider.BaseResourceProviderR4Test;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.impl.LinkedBlockingChannel;
|
import ca.uhn.fhir.jpa.subscription.channel.impl.LinkedBlockingChannel;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
||||||
import ca.uhn.fhir.rest.api.MethodOutcome;
|
import ca.uhn.fhir.rest.api.MethodOutcome;
|
||||||
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import ca.uhn.fhir.test.utilities.server.HashMapResourceProviderExtension;
|
import ca.uhn.fhir.test.utilities.server.HashMapResourceProviderExtension;
|
||||||
import ca.uhn.fhir.test.utilities.server.RestfulServerExtension;
|
import ca.uhn.fhir.test.utilities.server.RestfulServerExtension;
|
||||||
import ca.uhn.fhir.test.utilities.server.TransactionCapturingProviderExtension;
|
import ca.uhn.fhir.test.utilities.server.TransactionCapturingProviderExtension;
|
||||||
|
@ -61,7 +63,11 @@ public abstract class BaseSubscriptionsR4Test extends BaseResourceProviderR4Test
|
||||||
@Autowired
|
@Autowired
|
||||||
protected SubscriptionTestUtil mySubscriptionTestUtil;
|
protected SubscriptionTestUtil mySubscriptionTestUtil;
|
||||||
@Autowired
|
@Autowired
|
||||||
protected SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
protected ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
|
@Autowired
|
||||||
|
protected IResourceModifiedMessagePersistenceSvc myResourceModifiedMessagePersistenceSvc;
|
||||||
|
@Autowired
|
||||||
|
protected IResourceModifiedDao myResourceModifiedDao;
|
||||||
protected CountingInterceptor myCountingInterceptor;
|
protected CountingInterceptor myCountingInterceptor;
|
||||||
protected List<IIdType> mySubscriptionIds = Collections.synchronizedList(new ArrayList<>());
|
protected List<IIdType> mySubscriptionIds = Collections.synchronizedList(new ArrayList<>());
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -84,6 +90,7 @@ public abstract class BaseSubscriptionsR4Test extends BaseResourceProviderR4Test
|
||||||
myStorageSettings.setAllowMultipleDelete(new JpaStorageSettings().isAllowMultipleDelete());
|
myStorageSettings.setAllowMultipleDelete(new JpaStorageSettings().isAllowMultipleDelete());
|
||||||
|
|
||||||
mySubscriptionTestUtil.unregisterSubscriptionInterceptor();
|
mySubscriptionTestUtil.unregisterSubscriptionInterceptor();
|
||||||
|
myResourceModifiedDao.deleteAll();
|
||||||
}
|
}
|
||||||
|
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
|
@ -102,7 +109,7 @@ public abstract class BaseSubscriptionsR4Test extends BaseResourceProviderR4Test
|
||||||
waitForActivatedSubscriptionCount(0);
|
waitForActivatedSubscriptionCount(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
LinkedBlockingChannel processingChannel = mySubscriptionMatcherInterceptor.getProcessingChannelForUnitTest();
|
LinkedBlockingChannel processingChannel = (LinkedBlockingChannel) myResourceModifiedSubmitterSvc.getProcessingChannelForUnitTest();
|
||||||
if (processingChannel != null) {
|
if (processingChannel != null) {
|
||||||
processingChannel.clearInterceptorsForUnitTest();
|
processingChannel.clearInterceptorsForUnitTest();
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,158 @@
|
||||||
|
package ca.uhn.fhir.jpa.subscription.async;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.BaseSubscriptionsR4Test;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SynchronousSubscriptionMatcherInterceptor;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.api.ChannelConsumerSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelReceiver;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.message.TestQueueConsumerHandler;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedJsonMessage;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
||||||
|
import ca.uhn.fhir.jpa.test.util.StoppableSubscriptionDeliveringRestHookSubscriber;
|
||||||
|
import org.hl7.fhir.instance.model.api.IBaseResource;
|
||||||
|
import org.hl7.fhir.r4.model.Coding;
|
||||||
|
import org.hl7.fhir.r4.model.Observation;
|
||||||
|
import org.hl7.fhir.r4.model.Subscription;
|
||||||
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.boot.test.mock.mockito.SpyBean;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.context.annotation.Primary;
|
||||||
|
import org.springframework.test.context.ContextConfiguration;
|
||||||
|
|
||||||
|
import static org.hamcrest.MatcherAssert.assertThat;
|
||||||
|
import static org.hamcrest.Matchers.equalTo;
|
||||||
|
import static org.hamcrest.Matchers.hasSize;
|
||||||
|
import static org.hamcrest.Matchers.is;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertFalse;
|
||||||
|
|
||||||
|
@ContextConfiguration(classes = {AsyncSubscriptionMessageSubmissionIT.SpringConfig.class})
|
||||||
|
public class AsyncSubscriptionMessageSubmissionIT extends BaseSubscriptionsR4Test {
|
||||||
|
|
||||||
|
@SpyBean
|
||||||
|
IResourceModifiedConsumer myResourceModifiedConsumer;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
AsyncResourceModifiedSubmitterSvc myAsyncResourceModifiedSubmitterSvc;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private SubscriptionChannelFactory myChannelFactory;
|
||||||
|
|
||||||
|
@Autowired SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
StoppableSubscriptionDeliveringRestHookSubscriber myStoppableSubscriptionDeliveringRestHookSubscriber;
|
||||||
|
private TestQueueConsumerHandler<ResourceModifiedJsonMessage> myQueueConsumerHandler;
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
|
public void cleanupStoppableSubscriptionDeliveringRestHookSubscriber() {
|
||||||
|
myStoppableSubscriptionDeliveringRestHookSubscriber.setCountDownLatch(null);
|
||||||
|
myStoppableSubscriptionDeliveringRestHookSubscriber.unPause();
|
||||||
|
myStorageSettings.setTriggerSubscriptionsForNonVersioningChanges(new JpaStorageSettings().isTriggerSubscriptionsForNonVersioningChanges());
|
||||||
|
myStorageSettings.setTagStorageMode(new JpaStorageSettings().getTagStorageMode());
|
||||||
|
}
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
public void beforeRegisterRestHookListenerAndSchedulePoisonPillInterceptor() {
|
||||||
|
mySubscriptionTestUtil.registerMessageInterceptor();
|
||||||
|
|
||||||
|
IChannelReceiver receiver = myChannelFactory.newMatchingReceivingChannel("my-queue-name", new ChannelConsumerSettings());
|
||||||
|
myQueueConsumerHandler = new TestQueueConsumerHandler();
|
||||||
|
receiver.subscribe(myQueueConsumerHandler);
|
||||||
|
|
||||||
|
myStorageSettings.setTagStorageMode(JpaStorageSettings.TagStorageModeEnum.NON_VERSIONED);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSpringInjects_BeanOfTypeSubscriptionMatchingInterceptor_whenBeanDeclarationIsOverwrittenLocally(){
|
||||||
|
assertFalse(mySubscriptionMatcherInterceptor instanceof SynchronousSubscriptionMatcherInterceptor);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
// the purpose of this test is to assert that a resource matching a given subscription is
|
||||||
|
// delivered asynchronously to the subscription processing pipeline.
|
||||||
|
public void testAsynchronousDeliveryOfResourceMatchingASubscription_willSucceed() throws Exception {
|
||||||
|
String aCode = "zoop";
|
||||||
|
String aSystem = "SNOMED-CT";
|
||||||
|
// given
|
||||||
|
createAndSubmitSubscriptionWithCriteria("[Observation]");
|
||||||
|
waitForActivatedSubscriptionCount(1);
|
||||||
|
|
||||||
|
// when
|
||||||
|
Observation obs = sendObservation(aCode, aSystem);
|
||||||
|
|
||||||
|
assertCountOfResourcesNeedingSubmission(2); // the subscription and the observation
|
||||||
|
assertCountOfResourcesReceivedAtSubscriptionTerminalEndpoint(0);
|
||||||
|
|
||||||
|
// since scheduled tasks are disabled during tests, let's trigger a submission
|
||||||
|
// just like the AsyncResourceModifiedProcessingSchedulerSvc would.
|
||||||
|
myAsyncResourceModifiedSubmitterSvc.runDeliveryPass();
|
||||||
|
|
||||||
|
//then
|
||||||
|
waitForQueueToDrain();
|
||||||
|
assertCountOfResourcesNeedingSubmission(0);
|
||||||
|
assertCountOfResourcesReceivedAtSubscriptionTerminalEndpoint(1);
|
||||||
|
|
||||||
|
Observation observation = (Observation) fetchSingleResourceFromSubscriptionTerminalEndpoint();
|
||||||
|
Coding coding = observation.getCode().getCodingFirstRep();
|
||||||
|
|
||||||
|
assertThat(coding.getCode(), equalTo(aCode));
|
||||||
|
assertThat(coding.getSystem(), equalTo(aSystem));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertCountOfResourcesNeedingSubmission(int theExpectedCount) {
|
||||||
|
assertThat(myResourceModifiedMessagePersistenceSvc.findAllOrderedByCreatedTime(), hasSize(theExpectedCount));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Subscription createAndSubmitSubscriptionWithCriteria(String theCriteria) {
|
||||||
|
Subscription subscription = new Subscription();
|
||||||
|
subscription.setReason("Monitor new neonatal function (note, age will be determined by the monitor)");
|
||||||
|
subscription.setStatus(Subscription.SubscriptionStatus.REQUESTED);
|
||||||
|
subscription.setCriteria(theCriteria);
|
||||||
|
|
||||||
|
Subscription.SubscriptionChannelComponent channel = subscription.getChannel();
|
||||||
|
channel.setType(Subscription.SubscriptionChannelType.MESSAGE);
|
||||||
|
channel.setPayload("application/fhir+json");
|
||||||
|
channel.setEndpoint("channel:my-queue-name");
|
||||||
|
|
||||||
|
subscription.setChannel(channel);
|
||||||
|
postOrPutSubscription(subscription);
|
||||||
|
|
||||||
|
myAsyncResourceModifiedSubmitterSvc.runDeliveryPass();
|
||||||
|
|
||||||
|
return subscription;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private IBaseResource fetchSingleResourceFromSubscriptionTerminalEndpoint() {
|
||||||
|
assertThat(myQueueConsumerHandler.getMessages().size(), is(equalTo(1)));
|
||||||
|
ResourceModifiedJsonMessage resourceModifiedJsonMessage = myQueueConsumerHandler.getMessages().get(0);
|
||||||
|
ResourceModifiedMessage payload = resourceModifiedJsonMessage.getPayload();
|
||||||
|
String payloadString = payload.getPayloadString();
|
||||||
|
IBaseResource resource = myFhirContext.newJsonParser().parseResource(payloadString);
|
||||||
|
myQueueConsumerHandler.clearMessages();
|
||||||
|
return resource;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertCountOfResourcesReceivedAtSubscriptionTerminalEndpoint(int expectedCount) {
|
||||||
|
assertThat(myQueueConsumerHandler.getMessages(), hasSize(expectedCount));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
public static class SpringConfig {
|
||||||
|
|
||||||
|
@Primary
|
||||||
|
@Bean
|
||||||
|
public SubscriptionMatcherInterceptor subscriptionMatcherInterceptor() {
|
||||||
|
return new SubscriptionMatcherInterceptor();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,6 +1,11 @@
|
||||||
package ca.uhn.fhir.jpa.subscription.message;
|
package ca.uhn.fhir.jpa.subscription.message;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.interceptor.model.RequestPartitionId;
|
||||||
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.dao.data.IResourceModifiedDao;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.IPersistedResourceModifiedMessagePK;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.PersistedResourceModifiedMessageEntityPK;
|
||||||
import ca.uhn.fhir.jpa.subscription.BaseSubscriptionsR4Test;
|
import ca.uhn.fhir.jpa.subscription.BaseSubscriptionsR4Test;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.api.ChannelConsumerSettings;
|
import ca.uhn.fhir.jpa.subscription.channel.api.ChannelConsumerSettings;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelReceiver;
|
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelReceiver;
|
||||||
|
@ -11,20 +16,28 @@ import ca.uhn.fhir.jpa.test.util.StoppableSubscriptionDeliveringRestHookSubscrib
|
||||||
import ca.uhn.fhir.rest.client.api.Header;
|
import ca.uhn.fhir.rest.client.api.Header;
|
||||||
import ca.uhn.fhir.rest.client.api.IGenericClient;
|
import ca.uhn.fhir.rest.client.api.IGenericClient;
|
||||||
import ca.uhn.fhir.rest.client.interceptor.AdditionalRequestHeadersInterceptor;
|
import ca.uhn.fhir.rest.client.interceptor.AdditionalRequestHeadersInterceptor;
|
||||||
|
import ca.uhn.fhir.rest.server.messaging.BaseResourceMessage;
|
||||||
|
import com.fasterxml.jackson.core.JsonProcessingException;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
import org.hl7.fhir.instance.model.api.IBaseResource;
|
import org.hl7.fhir.instance.model.api.IBaseResource;
|
||||||
import org.hl7.fhir.instance.model.api.IIdType;
|
import org.hl7.fhir.instance.model.api.IIdType;
|
||||||
import org.hl7.fhir.r4.model.Coding;
|
import org.hl7.fhir.r4.model.Coding;
|
||||||
import org.hl7.fhir.r4.model.Observation;
|
import org.hl7.fhir.r4.model.Observation;
|
||||||
|
import org.hl7.fhir.r4.model.Organization;
|
||||||
import org.hl7.fhir.r4.model.Patient;
|
import org.hl7.fhir.r4.model.Patient;
|
||||||
import org.hl7.fhir.r4.model.Subscription;
|
import org.hl7.fhir.r4.model.Subscription;
|
||||||
import org.junit.jupiter.api.AfterEach;
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.Assertions;
|
||||||
import org.junit.jupiter.api.BeforeEach;
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.params.ParameterizedTest;
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
import org.junit.jupiter.params.provider.Arguments;
|
import org.junit.jupiter.params.provider.Arguments;
|
||||||
import org.junit.jupiter.params.provider.MethodSource;
|
import org.junit.jupiter.params.provider.MethodSource;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.transaction.PlatformTransactionManager;
|
||||||
|
import org.springframework.transaction.support.TransactionTemplate;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
@ -49,6 +62,12 @@ public class MessageSubscriptionR4Test extends BaseSubscriptionsR4Test {
|
||||||
private static final Logger ourLog = LoggerFactory.getLogger(MessageSubscriptionR4Test.class);
|
private static final Logger ourLog = LoggerFactory.getLogger(MessageSubscriptionR4Test.class);
|
||||||
private TestQueueConsumerHandler<ResourceModifiedJsonMessage> handler;
|
private TestQueueConsumerHandler<ResourceModifiedJsonMessage> handler;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
IResourceModifiedDao myResourceModifiedDao;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private PlatformTransactionManager myTxManager;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
StoppableSubscriptionDeliveringRestHookSubscriber myStoppableSubscriptionDeliveringRestHookSubscriber;
|
StoppableSubscriptionDeliveringRestHookSubscriber myStoppableSubscriptionDeliveringRestHookSubscriber;
|
||||||
|
|
||||||
|
@ -176,6 +195,109 @@ public class MessageSubscriptionR4Test extends BaseSubscriptionsR4Test {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMethodFindAllOrdered_willReturnAllPersistedResourceModifiedMessagesOrderedByCreatedTime(){
|
||||||
|
mySubscriptionTestUtil.unregisterSubscriptionInterceptor();
|
||||||
|
|
||||||
|
// given
|
||||||
|
Patient patient = sendPatient();
|
||||||
|
Organization organization = sendOrganization();
|
||||||
|
|
||||||
|
ResourceModifiedMessage patientResourceModifiedMessage = new ResourceModifiedMessage(myFhirContext, patient, BaseResourceMessage.OperationTypeEnum.CREATE);
|
||||||
|
ResourceModifiedMessage organizationResourceModifiedMessage = new ResourceModifiedMessage(myFhirContext, organization, BaseResourceMessage.OperationTypeEnum.CREATE);
|
||||||
|
|
||||||
|
IPersistedResourceModifiedMessage patientPersistedMessage = myResourceModifiedMessagePersistenceSvc.persist(patientResourceModifiedMessage);
|
||||||
|
IPersistedResourceModifiedMessage organizationPersistedMessage = myResourceModifiedMessagePersistenceSvc.persist(organizationResourceModifiedMessage);
|
||||||
|
|
||||||
|
// when
|
||||||
|
List<IPersistedResourceModifiedMessage> allPersisted = myResourceModifiedMessagePersistenceSvc.findAllOrderedByCreatedTime();
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertOnPksAndOrder(allPersisted, List.of(patientPersistedMessage, organizationPersistedMessage));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMethodDeleteByPK_whenEntityExists_willDeleteTheEntityAndReturnTrue(){
|
||||||
|
mySubscriptionTestUtil.unregisterSubscriptionInterceptor();
|
||||||
|
|
||||||
|
// given
|
||||||
|
TransactionTemplate transactionTemplate = new TransactionTemplate(myTxManager);
|
||||||
|
Patient patient = sendPatient();
|
||||||
|
|
||||||
|
ResourceModifiedMessage patientResourceModifiedMessage = new ResourceModifiedMessage(myFhirContext, patient, BaseResourceMessage.OperationTypeEnum.CREATE);
|
||||||
|
IPersistedResourceModifiedMessage persistedResourceModifiedMessage = myResourceModifiedMessagePersistenceSvc.persist(patientResourceModifiedMessage);
|
||||||
|
|
||||||
|
// when
|
||||||
|
boolean wasDeleted = transactionTemplate.execute(tx -> myResourceModifiedMessagePersistenceSvc.deleteByPK(persistedResourceModifiedMessage.getPersistedResourceModifiedMessagePk()));
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertThat(wasDeleted, is(Boolean.TRUE));
|
||||||
|
assertThat(myResourceModifiedMessagePersistenceSvc.findAllOrderedByCreatedTime(), hasSize(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testMethodDeleteByPK_whenEntityDoesNotExist_willReturnFalse(){
|
||||||
|
mySubscriptionTestUtil.unregisterSubscriptionInterceptor();
|
||||||
|
|
||||||
|
// given
|
||||||
|
TransactionTemplate transactionTemplate = new TransactionTemplate(myTxManager);
|
||||||
|
IPersistedResourceModifiedMessagePK nonExistentResourceWithPk = PersistedResourceModifiedMessageEntityPK.with("one", "one");
|
||||||
|
|
||||||
|
// when
|
||||||
|
boolean wasDeleted = transactionTemplate.execute(tx -> myResourceModifiedMessagePersistenceSvc.deleteByPK(nonExistentResourceWithPk));
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertThat(wasDeleted, is(Boolean.FALSE));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPersistedResourceModifiedMessage_whenFetchFromDb_willEqualOriginalMessage() throws JsonProcessingException {
|
||||||
|
mySubscriptionTestUtil.unregisterSubscriptionInterceptor();
|
||||||
|
// given
|
||||||
|
TransactionTemplate transactionTemplate = new TransactionTemplate(myTxManager);
|
||||||
|
Observation obs = sendObservation("zoop", "SNOMED-CT", "theExplicitSource", "theRequestId");
|
||||||
|
|
||||||
|
ResourceModifiedMessage originalResourceModifiedMessage = createResourceModifiedMessage(obs);
|
||||||
|
|
||||||
|
transactionTemplate.execute(tx -> {
|
||||||
|
|
||||||
|
IPersistedResourceModifiedMessage persistedResourceModifiedMessage = myResourceModifiedMessagePersistenceSvc.persist(originalResourceModifiedMessage);
|
||||||
|
|
||||||
|
// when
|
||||||
|
ResourceModifiedMessage restoredResourceModifiedMessage = myResourceModifiedMessagePersistenceSvc.inflatePersistedResourceModifiedMessage(persistedResourceModifiedMessage);
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertEquals(toJson(originalResourceModifiedMessage), toJson(restoredResourceModifiedMessage));
|
||||||
|
assertEquals(originalResourceModifiedMessage, restoredResourceModifiedMessage);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private ResourceModifiedMessage createResourceModifiedMessage(Observation theObservation){
|
||||||
|
ResourceModifiedMessage retVal = new ResourceModifiedMessage(myFhirContext, theObservation, BaseResourceMessage.OperationTypeEnum.CREATE);
|
||||||
|
retVal.setSubscriptionId("subId");
|
||||||
|
retVal.setTransactionId("txId");
|
||||||
|
retVal.setMessageKey("messageKey");
|
||||||
|
retVal.setMediaType("json");
|
||||||
|
retVal.setAttribute("attKey", "attValue");
|
||||||
|
retVal.setPartitionId(RequestPartitionId.allPartitions());
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void assertEquals(ResourceModifiedMessage theMsg, ResourceModifiedMessage theComparedTo){
|
||||||
|
assertThat(theMsg.getPayloadId(), equalTo(theComparedTo.getPayloadId()));
|
||||||
|
assertThat(theMsg.getOperationType(), equalTo(theComparedTo.getOperationType()));
|
||||||
|
assertThat(theMsg.getPayloadString(), equalTo(theComparedTo.getPayloadString()));
|
||||||
|
assertThat(theMsg.getSubscriptionId(), equalTo(theComparedTo.getSubscriptionId()));
|
||||||
|
assertThat(theMsg.getMediaType(), equalTo(theComparedTo.getMediaType()));
|
||||||
|
assertThat(theMsg.getMessageKeyOrNull(), equalTo(theComparedTo.getMessageKeyOrNull()));
|
||||||
|
assertThat(theMsg.getTransactionId(), equalTo(theComparedTo.getTransactionId()));
|
||||||
|
assertThat(theMsg.getAttributes(), equalTo(theComparedTo.getAttributes()));
|
||||||
|
}
|
||||||
|
|
||||||
private void maybeAddHeaderInterceptor(IGenericClient theClient, List<Header> theHeaders) {
|
private void maybeAddHeaderInterceptor(IGenericClient theClient, List<Header> theHeaders) {
|
||||||
if(theHeaders.isEmpty()){
|
if(theHeaders.isEmpty()){
|
||||||
return;
|
return;
|
||||||
|
@ -215,4 +337,32 @@ public class MessageSubscriptionR4Test extends BaseSubscriptionsR4Test {
|
||||||
return (T) resource;
|
return (T) resource;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void assertEquals(String theMsg, String theComparedTo){
|
||||||
|
assertThat(theMsg, equalTo(theComparedTo));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String toJson(Object theRequest) {
|
||||||
|
try {
|
||||||
|
return new ObjectMapper().writer().writeValueAsString(theRequest);
|
||||||
|
} catch (JsonProcessingException theE) {
|
||||||
|
throw new AssertionError("Failure during serialization: " + theE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void assertOnPksAndOrder(List<IPersistedResourceModifiedMessage> theFetchedResourceModifiedMessageList, List<IPersistedResourceModifiedMessage> theCompareToList ){
|
||||||
|
assertThat(theFetchedResourceModifiedMessageList, hasSize(theCompareToList.size()));
|
||||||
|
|
||||||
|
List<IPersistedResourceModifiedMessagePK> fetchedPks = theFetchedResourceModifiedMessageList
|
||||||
|
.stream()
|
||||||
|
.map(IPersistedResourceModifiedMessage::getPersistedResourceModifiedMessagePk)
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
|
||||||
|
List<IPersistedResourceModifiedMessagePK> compareToPks = theCompareToList
|
||||||
|
.stream()
|
||||||
|
.map(IPersistedResourceModifiedMessage::getPersistedResourceModifiedMessagePk)
|
||||||
|
.collect(Collectors.toList());
|
||||||
|
|
||||||
|
Assertions.assertEquals(fetchedPks, compareToPks);
|
||||||
|
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ package ca.uhn.fhir.jpa.subscription.resthook;
|
||||||
|
|
||||||
import ca.uhn.fhir.context.FhirContext;
|
import ca.uhn.fhir.context.FhirContext;
|
||||||
import ca.uhn.fhir.jpa.provider.BaseResourceProviderR4Test;
|
import ca.uhn.fhir.jpa.provider.BaseResourceProviderR4Test;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
||||||
import ca.uhn.fhir.rest.annotation.ResourceParam;
|
import ca.uhn.fhir.rest.annotation.ResourceParam;
|
||||||
import ca.uhn.fhir.rest.annotation.Update;
|
import ca.uhn.fhir.rest.annotation.Update;
|
||||||
|
@ -52,7 +52,7 @@ public class RestHookActivatesPreExistingSubscriptionsR4Test extends BaseResourc
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionTestUtil mySubscriptionTestUtil;
|
private SubscriptionTestUtil mySubscriptionTestUtil;
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
private ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
|
|
||||||
@AfterEach
|
@AfterEach
|
||||||
public void afterUnregisterRestHookListener() {
|
public void afterUnregisterRestHookListener() {
|
||||||
|
@ -63,7 +63,7 @@ public class RestHookActivatesPreExistingSubscriptionsR4Test extends BaseResourc
|
||||||
@BeforeEach
|
@BeforeEach
|
||||||
public void beforeSetSubscriptionActivatingInterceptor() {
|
public void beforeSetSubscriptionActivatingInterceptor() {
|
||||||
myStorageSettings.addSupportedSubscriptionType(org.hl7.fhir.dstu2.model.Subscription.SubscriptionChannelType.RESTHOOK);
|
myStorageSettings.addSupportedSubscriptionType(org.hl7.fhir.dstu2.model.Subscription.SubscriptionChannelType.RESTHOOK);
|
||||||
mySubscriptionMatcherInterceptor.startIfNeeded();
|
myResourceModifiedSubmitterSvc.startIfNeeded();
|
||||||
mySubscriptionLoader.doSyncSubscriptionsForUnitTest();
|
mySubscriptionLoader.doSyncSubscriptionsForUnitTest();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,7 @@ package ca.uhn.fhir.jpa.subscription.resthook;
|
||||||
import ca.uhn.fhir.i18n.Msg;
|
import ca.uhn.fhir.i18n.Msg;
|
||||||
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
import ca.uhn.fhir.jpa.subscription.BaseSubscriptionsR4Test;
|
import ca.uhn.fhir.jpa.subscription.BaseSubscriptionsR4Test;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.test.util.StoppableSubscriptionDeliveringRestHookSubscriber;
|
import ca.uhn.fhir.jpa.test.util.StoppableSubscriptionDeliveringRestHookSubscriber;
|
||||||
import ca.uhn.fhir.jpa.topic.SubscriptionTopicDispatcher;
|
import ca.uhn.fhir.jpa.topic.SubscriptionTopicDispatcher;
|
||||||
import ca.uhn.fhir.jpa.topic.SubscriptionTopicRegistry;
|
import ca.uhn.fhir.jpa.topic.SubscriptionTopicRegistry;
|
||||||
|
@ -31,6 +32,7 @@ import org.hl7.fhir.r4.model.SearchParameter;
|
||||||
import org.hl7.fhir.r4.model.StringType;
|
import org.hl7.fhir.r4.model.StringType;
|
||||||
import org.hl7.fhir.r4.model.Subscription;
|
import org.hl7.fhir.r4.model.Subscription;
|
||||||
import org.junit.jupiter.api.AfterEach;
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.params.ParameterizedTest;
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
import org.junit.jupiter.params.provider.ValueSource;
|
import org.junit.jupiter.params.provider.ValueSource;
|
||||||
|
@ -64,6 +66,9 @@ import static org.junit.jupiter.api.Assertions.fail;
|
||||||
public class RestHookTestR4Test extends BaseSubscriptionsR4Test {
|
public class RestHookTestR4Test extends BaseSubscriptionsR4Test {
|
||||||
private static final Logger ourLog = LoggerFactory.getLogger(RestHookTestR4Test.class);
|
private static final Logger ourLog = LoggerFactory.getLogger(RestHookTestR4Test.class);
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
StoppableSubscriptionDeliveringRestHookSubscriber myStoppableSubscriptionDeliveringRestHookSubscriber;
|
StoppableSubscriptionDeliveringRestHookSubscriber myStoppableSubscriptionDeliveringRestHookSubscriber;
|
||||||
@Autowired(required = false)
|
@Autowired(required = false)
|
||||||
|
@ -113,7 +118,6 @@ public class RestHookTestR4Test extends BaseSubscriptionsR4Test {
|
||||||
assertEquals("IN_MEMORY", subscription.getMeta().getTag().get(0).getCode());
|
assertEquals("IN_MEMORY", subscription.getMeta().getTag().get(0).getCode());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testRestHookSubscriptionApplicationFhirJson() throws Exception {
|
public void testRestHookSubscriptionApplicationFhirJson() throws Exception {
|
||||||
String payload = "application/fhir+json";
|
String payload = "application/fhir+json";
|
||||||
|
|
|
@ -0,0 +1,141 @@
|
||||||
|
package ca.uhn.fhir.jpa.subscription.svc;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.ResourceModifiedEntity;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.api.ChannelProducerSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.api.IChannelProducer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
|
import ca.uhn.fhir.jpa.svc.MockHapiTransactionService;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.junit.jupiter.api.extension.ExtendWith;
|
||||||
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
|
import org.junit.jupiter.params.provider.ValueSource;
|
||||||
|
import org.mockito.ArgumentCaptor;
|
||||||
|
import org.mockito.Captor;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.mockito.junit.jupiter.MockitoExtension;
|
||||||
|
import org.springframework.messaging.MessageDeliveryException;
|
||||||
|
import org.springframework.transaction.TransactionStatus;
|
||||||
|
import org.springframework.transaction.support.SimpleTransactionStatus;
|
||||||
|
|
||||||
|
import static org.hamcrest.MatcherAssert.assertThat;
|
||||||
|
import static org.hamcrest.Matchers.is;
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
|
import static org.mockito.Mockito.lenient;
|
||||||
|
import static org.mockito.Mockito.times;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
@ExtendWith(MockitoExtension.class)
|
||||||
|
public class ResourceModifiedSubmitterSvcTest {
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
StorageSettings myStorageSettings;
|
||||||
|
@Mock
|
||||||
|
SubscriptionChannelFactory mySubscriptionChannelFactory;
|
||||||
|
@Mock
|
||||||
|
IResourceModifiedMessagePersistenceSvc myResourceModifiedMessagePersistenceSvc;
|
||||||
|
@Captor
|
||||||
|
ArgumentCaptor<ChannelProducerSettings> myArgumentCaptor;
|
||||||
|
@Mock
|
||||||
|
IChannelProducer myChannelProducer;
|
||||||
|
|
||||||
|
ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
|
TransactionStatus myCapturingTransactionStatus;
|
||||||
|
|
||||||
|
@BeforeEach
|
||||||
|
public void beforeEach(){
|
||||||
|
myCapturingTransactionStatus = new SimpleTransactionStatus();
|
||||||
|
lenient().when(myStorageSettings.hasSupportedSubscriptionTypes()).thenReturn(true);
|
||||||
|
lenient().when(mySubscriptionChannelFactory.newMatchingSendingChannel(anyString(), any())).thenReturn(myChannelProducer);
|
||||||
|
|
||||||
|
IHapiTransactionService hapiTransactionService = new MockHapiTransactionService(myCapturingTransactionStatus);
|
||||||
|
myResourceModifiedSubmitterSvc = new ResourceModifiedSubmitterSvc(
|
||||||
|
myStorageSettings,
|
||||||
|
mySubscriptionChannelFactory,
|
||||||
|
myResourceModifiedMessagePersistenceSvc,
|
||||||
|
hapiTransactionService);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@ParameterizedTest
|
||||||
|
@ValueSource(booleans = {false, true})
|
||||||
|
public void testMethodStartIfNeeded_withQualifySubscriptionMatchingChannelNameProperty_mayQualifyChannelName(boolean theIsQualifySubMatchingChannelName){
|
||||||
|
// given
|
||||||
|
boolean expectedResult = theIsQualifySubMatchingChannelName;
|
||||||
|
when(myStorageSettings.isQualifySubscriptionMatchingChannelName()).thenReturn(theIsQualifySubMatchingChannelName);
|
||||||
|
|
||||||
|
// when
|
||||||
|
myResourceModifiedSubmitterSvc.startIfNeeded();
|
||||||
|
|
||||||
|
// then
|
||||||
|
ChannelProducerSettings capturedChannelProducerSettings = getCapturedChannelProducerSettings();
|
||||||
|
assertThat(capturedChannelProducerSettings.isQualifyChannelName(), is(expectedResult));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSubmitPersisedResourceModifiedMessage_withExistingPersistedResourceModifiedMessage_willSucceed(){
|
||||||
|
// given
|
||||||
|
// a successful deletion implies that the message did exist.
|
||||||
|
when(myResourceModifiedMessagePersistenceSvc.deleteByPK(any())).thenReturn(true);
|
||||||
|
when(myResourceModifiedMessagePersistenceSvc.inflatePersistedResourceModifiedMessage(any())).thenReturn(new ResourceModifiedMessage());
|
||||||
|
|
||||||
|
// when
|
||||||
|
boolean wasProcessed = myResourceModifiedSubmitterSvc.submitPersisedResourceModifiedMessage(new ResourceModifiedEntity());
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertThat(wasProcessed, is(Boolean.TRUE));
|
||||||
|
assertThat(myCapturingTransactionStatus.isRollbackOnly(), is(Boolean.FALSE));
|
||||||
|
verify(myChannelProducer, times(1)).send(any());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSubmitPersisedResourceModifiedMessage_whenMessageWasAlreadyProcess_willSucceed(){
|
||||||
|
// given
|
||||||
|
// deletion fails, someone else was faster and processed the message
|
||||||
|
when(myResourceModifiedMessagePersistenceSvc.deleteByPK(any())).thenReturn(false);
|
||||||
|
when(myResourceModifiedMessagePersistenceSvc.inflatePersistedResourceModifiedMessage(any())).thenReturn(new ResourceModifiedMessage());
|
||||||
|
|
||||||
|
// when
|
||||||
|
boolean wasProcessed = myResourceModifiedSubmitterSvc.submitPersisedResourceModifiedMessage(new ResourceModifiedEntity());
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertThat(wasProcessed, is(Boolean.TRUE));
|
||||||
|
assertThat(myCapturingTransactionStatus.isRollbackOnly(), is(Boolean.FALSE));
|
||||||
|
// we do not send a message which was already sent
|
||||||
|
verify(myChannelProducer, times(0)).send(any());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testSubmitPersisedResourceModifiedMessage_whitErrorOnSending_willRollbackDeletion(){
|
||||||
|
// given
|
||||||
|
when(myResourceModifiedMessagePersistenceSvc.deleteByPK(any())).thenReturn(true);
|
||||||
|
when(myResourceModifiedMessagePersistenceSvc.inflatePersistedResourceModifiedMessage(any())).thenReturn(new ResourceModifiedMessage());
|
||||||
|
|
||||||
|
// simulate failure writing to the channel
|
||||||
|
when(myChannelProducer.send(any())).thenThrow(new MessageDeliveryException("sendingError"));
|
||||||
|
|
||||||
|
// when
|
||||||
|
boolean wasProcessed = myResourceModifiedSubmitterSvc.submitPersisedResourceModifiedMessage(new ResourceModifiedEntity());
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertThat(wasProcessed, is(Boolean.FALSE));
|
||||||
|
assertThat(myCapturingTransactionStatus.isRollbackOnly(), is(Boolean.TRUE));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private ChannelProducerSettings getCapturedChannelProducerSettings(){
|
||||||
|
verify(mySubscriptionChannelFactory).newMatchingSendingChannel(anyString(), myArgumentCaptor.capture());
|
||||||
|
return myArgumentCaptor.getValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -4,7 +4,7 @@ import ca.uhn.fhir.context.FhirContext;
|
||||||
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
import ca.uhn.fhir.jpa.provider.r4b.BaseResourceProviderR4BTest;
|
import ca.uhn.fhir.jpa.provider.r4b.BaseResourceProviderR4BTest;
|
||||||
import ca.uhn.fhir.jpa.subscription.channel.impl.LinkedBlockingChannel;
|
import ca.uhn.fhir.jpa.subscription.channel.impl.LinkedBlockingChannel;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
||||||
import ca.uhn.fhir.rest.api.MethodOutcome;
|
import ca.uhn.fhir.rest.api.MethodOutcome;
|
||||||
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
|
||||||
|
@ -63,7 +63,7 @@ public abstract class BaseSubscriptionsR4BTest extends BaseResourceProviderR4BTe
|
||||||
@Autowired
|
@Autowired
|
||||||
protected SubscriptionTestUtil mySubscriptionTestUtil;
|
protected SubscriptionTestUtil mySubscriptionTestUtil;
|
||||||
@Autowired
|
@Autowired
|
||||||
protected SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
protected ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
protected CountingInterceptor myCountingInterceptor;
|
protected CountingInterceptor myCountingInterceptor;
|
||||||
protected List<IIdType> mySubscriptionIds = Collections.synchronizedList(new ArrayList<>());
|
protected List<IIdType> mySubscriptionIds = Collections.synchronizedList(new ArrayList<>());
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -104,12 +104,12 @@ public abstract class BaseSubscriptionsR4BTest extends BaseResourceProviderR4BTe
|
||||||
waitForActivatedSubscriptionCount(0);
|
waitForActivatedSubscriptionCount(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
LinkedBlockingChannel processingChannel = mySubscriptionMatcherInterceptor.getProcessingChannelForUnitTest();
|
myCountingInterceptor = new CountingInterceptor();
|
||||||
|
|
||||||
|
LinkedBlockingChannel processingChannel = (LinkedBlockingChannel) myResourceModifiedSubmitterSvc.getProcessingChannelForUnitTest();
|
||||||
|
|
||||||
if (processingChannel != null) {
|
if (processingChannel != null) {
|
||||||
processingChannel.clearInterceptorsForUnitTest();
|
processingChannel.clearInterceptorsForUnitTest();
|
||||||
}
|
|
||||||
myCountingInterceptor = new CountingInterceptor();
|
|
||||||
if (processingChannel != null) {
|
|
||||||
processingChannel.addInterceptor(myCountingInterceptor);
|
processingChannel.addInterceptor(myCountingInterceptor);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@ package ca.uhn.fhir.jpa.subscription;
|
||||||
|
|
||||||
import ca.uhn.fhir.i18n.Msg;
|
import ca.uhn.fhir.i18n.Msg;
|
||||||
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
import ca.uhn.fhir.jpa.api.config.JpaStorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.test.util.StoppableSubscriptionDeliveringRestHookSubscriber;
|
import ca.uhn.fhir.jpa.test.util.StoppableSubscriptionDeliveringRestHookSubscriber;
|
||||||
import ca.uhn.fhir.model.primitive.IdDt;
|
import ca.uhn.fhir.model.primitive.IdDt;
|
||||||
import ca.uhn.fhir.rest.api.CacheControlDirective;
|
import ca.uhn.fhir.rest.api.CacheControlDirective;
|
||||||
|
@ -25,6 +26,7 @@ import org.hl7.fhir.r4b.model.SearchParameter;
|
||||||
import org.hl7.fhir.r4b.model.StringType;
|
import org.hl7.fhir.r4b.model.StringType;
|
||||||
import org.hl7.fhir.r4b.model.Subscription;
|
import org.hl7.fhir.r4b.model.Subscription;
|
||||||
import org.junit.jupiter.api.AfterEach;
|
import org.junit.jupiter.api.AfterEach;
|
||||||
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
import org.junit.jupiter.api.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.params.ParameterizedTest;
|
import org.junit.jupiter.params.ParameterizedTest;
|
||||||
import org.junit.jupiter.params.provider.ValueSource;
|
import org.junit.jupiter.params.provider.ValueSource;
|
||||||
|
@ -56,6 +58,9 @@ import static org.junit.jupiter.api.Assertions.fail;
|
||||||
public class RestHookTestR4BTest extends BaseSubscriptionsR4BTest {
|
public class RestHookTestR4BTest extends BaseSubscriptionsR4BTest {
|
||||||
private static final Logger ourLog = LoggerFactory.getLogger(RestHookTestR4BTest.class);
|
private static final Logger ourLog = LoggerFactory.getLogger(RestHookTestR4BTest.class);
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
|
|
||||||
@Autowired
|
@Autowired
|
||||||
StoppableSubscriptionDeliveringRestHookSubscriber myStoppableSubscriptionDeliveringRestHookSubscriber;
|
StoppableSubscriptionDeliveringRestHookSubscriber myStoppableSubscriptionDeliveringRestHookSubscriber;
|
||||||
|
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -12,7 +12,7 @@ import ca.uhn.fhir.jpa.subscription.channel.impl.LinkedBlockingChannel;
|
||||||
import ca.uhn.fhir.jpa.subscription.model.CanonicalSubscriptionChannelType;
|
import ca.uhn.fhir.jpa.subscription.model.CanonicalSubscriptionChannelType;
|
||||||
import ca.uhn.fhir.jpa.subscription.model.CanonicalTopicSubscriptionFilter;
|
import ca.uhn.fhir.jpa.subscription.model.CanonicalTopicSubscriptionFilter;
|
||||||
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
import ca.uhn.fhir.jpa.subscription.model.ResourceModifiedMessage;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
import ca.uhn.fhir.jpa.test.util.SubscriptionTestUtil;
|
||||||
import ca.uhn.fhir.jpa.topic.SubscriptionTopicLoader;
|
import ca.uhn.fhir.jpa.topic.SubscriptionTopicLoader;
|
||||||
import ca.uhn.fhir.jpa.topic.SubscriptionTopicRegistry;
|
import ca.uhn.fhir.jpa.topic.SubscriptionTopicRegistry;
|
||||||
|
@ -73,7 +73,7 @@ public abstract class BaseSubscriptionsR5Test extends BaseResourceProviderR5Test
|
||||||
@Autowired
|
@Autowired
|
||||||
protected SubscriptionTestUtil mySubscriptionTestUtil;
|
protected SubscriptionTestUtil mySubscriptionTestUtil;
|
||||||
@Autowired
|
@Autowired
|
||||||
protected SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
protected ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
protected CountingInterceptor myCountingInterceptor;
|
protected CountingInterceptor myCountingInterceptor;
|
||||||
protected List<IIdType> mySubscriptionIds = Collections.synchronizedList(new ArrayList<>());
|
protected List<IIdType> mySubscriptionIds = Collections.synchronizedList(new ArrayList<>());
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -110,7 +110,7 @@ public abstract class BaseSubscriptionsR5Test extends BaseResourceProviderR5Test
|
||||||
waitForActivatedSubscriptionCount(0);
|
waitForActivatedSubscriptionCount(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
LinkedBlockingChannel processingChannel = mySubscriptionMatcherInterceptor.getProcessingChannelForUnitTest();
|
LinkedBlockingChannel processingChannel = (LinkedBlockingChannel) myResourceModifiedSubmitterSvc.getProcessingChannelForUnitTest();
|
||||||
if (processingChannel != null) {
|
if (processingChannel != null) {
|
||||||
processingChannel.clearInterceptorsForUnitTest();
|
processingChannel.clearInterceptorsForUnitTest();
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -17,9 +17,10 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
* #L%
|
* #L%
|
||||||
*/
|
*/
|
||||||
package ca.uhn.fhir.jpa.search;
|
package ca.uhn.fhir.jpa.svc;
|
||||||
|
|
||||||
import ca.uhn.fhir.jpa.dao.tx.HapiTransactionService;
|
import ca.uhn.fhir.jpa.dao.tx.HapiTransactionService;
|
||||||
|
import org.springframework.transaction.TransactionStatus;
|
||||||
import org.springframework.transaction.support.SimpleTransactionStatus;
|
import org.springframework.transaction.support.SimpleTransactionStatus;
|
||||||
import org.springframework.transaction.support.TransactionCallback;
|
import org.springframework.transaction.support.TransactionCallback;
|
||||||
|
|
||||||
|
@ -27,9 +28,19 @@ import javax.annotation.Nullable;
|
||||||
|
|
||||||
public class MockHapiTransactionService extends HapiTransactionService {
|
public class MockHapiTransactionService extends HapiTransactionService {
|
||||||
|
|
||||||
|
private TransactionStatus myTransactionStatus;
|
||||||
|
|
||||||
|
public MockHapiTransactionService() {
|
||||||
|
this(new SimpleTransactionStatus());
|
||||||
|
}
|
||||||
|
|
||||||
|
public MockHapiTransactionService(TransactionStatus theTransactionStatus) {
|
||||||
|
myTransactionStatus = theTransactionStatus;
|
||||||
|
}
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
@Override
|
@Override
|
||||||
protected <T> T doExecute(ExecutionBuilder theExecutionBuilder, TransactionCallback<T> theCallback) {
|
protected <T> T doExecute(ExecutionBuilder theExecutionBuilder, TransactionCallback<T> theCallback) {
|
||||||
return theCallback.doInTransaction(new SimpleTransactionStatus());
|
return theCallback.doInTransaction(myTransactionStatus);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -25,10 +25,16 @@ import ca.uhn.fhir.jpa.batch2.JpaBatch2Config;
|
||||||
import ca.uhn.fhir.jpa.config.HapiJpaConfig;
|
import ca.uhn.fhir.jpa.config.HapiJpaConfig;
|
||||||
import ca.uhn.fhir.jpa.config.JpaDstu2Config;
|
import ca.uhn.fhir.jpa.config.JpaDstu2Config;
|
||||||
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
||||||
import ca.uhn.fhir.validation.IInstanceValidatorModule;
|
import ca.uhn.fhir.validation.IInstanceValidatorModule;
|
||||||
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
||||||
|
@ -205,4 +211,5 @@ public class TestDstu2Config {
|
||||||
|
|
||||||
return requestValidator;
|
return requestValidator;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,15 +26,21 @@ import ca.uhn.fhir.jpa.config.HapiJpaConfig;
|
||||||
import ca.uhn.fhir.jpa.config.PackageLoaderConfig;
|
import ca.uhn.fhir.jpa.config.PackageLoaderConfig;
|
||||||
import ca.uhn.fhir.jpa.config.dstu3.JpaDstu3Config;
|
import ca.uhn.fhir.jpa.config.dstu3.JpaDstu3Config;
|
||||||
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
import ca.uhn.fhir.jpa.subscription.match.deliver.email.EmailSenderImpl;
|
import ca.uhn.fhir.jpa.subscription.match.deliver.email.EmailSenderImpl;
|
||||||
import ca.uhn.fhir.jpa.subscription.match.deliver.email.IEmailSender;
|
import ca.uhn.fhir.jpa.subscription.match.deliver.email.IEmailSender;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
||||||
import ca.uhn.fhir.rest.server.mail.IMailSvc;
|
import ca.uhn.fhir.rest.server.mail.IMailSvc;
|
||||||
import ca.uhn.fhir.rest.server.mail.MailConfig;
|
import ca.uhn.fhir.rest.server.mail.MailConfig;
|
||||||
import ca.uhn.fhir.rest.server.mail.MailSvc;
|
import ca.uhn.fhir.rest.server.mail.MailSvc;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
||||||
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
||||||
import net.ttddyy.dsproxy.support.ProxyDataSourceBuilder;
|
import net.ttddyy.dsproxy.support.ProxyDataSourceBuilder;
|
||||||
|
@ -220,5 +226,4 @@ public class TestDstu3Config {
|
||||||
return new PropertySourcesPlaceholderConfigurer();
|
return new PropertySourcesPlaceholderConfigurer();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,11 +27,17 @@ import ca.uhn.fhir.jpa.binstore.MemoryBinaryStorageSvcImpl;
|
||||||
import ca.uhn.fhir.jpa.config.HapiJpaConfig;
|
import ca.uhn.fhir.jpa.config.HapiJpaConfig;
|
||||||
import ca.uhn.fhir.jpa.config.r4b.JpaR4BConfig;
|
import ca.uhn.fhir.jpa.config.r4b.JpaR4BConfig;
|
||||||
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.topic.SubscriptionTopicConfig;
|
import ca.uhn.fhir.jpa.topic.SubscriptionTopicConfig;
|
||||||
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
||||||
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
||||||
import net.ttddyy.dsproxy.listener.SingleQueryCountHolder;
|
import net.ttddyy.dsproxy.listener.SingleQueryCountHolder;
|
||||||
|
|
|
@ -28,12 +28,18 @@ import ca.uhn.fhir.jpa.config.HapiJpaConfig;
|
||||||
import ca.uhn.fhir.jpa.config.PackageLoaderConfig;
|
import ca.uhn.fhir.jpa.config.PackageLoaderConfig;
|
||||||
import ca.uhn.fhir.jpa.config.r4.JpaR4Config;
|
import ca.uhn.fhir.jpa.config.r4.JpaR4Config;
|
||||||
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
import ca.uhn.fhir.jpa.config.util.HapiEntityManagerFactoryUtil;
|
||||||
|
import ca.uhn.fhir.jpa.dao.tx.IHapiTransactionService;
|
||||||
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
import ca.uhn.fhir.jpa.model.dialect.HapiFhirH2Dialect;
|
||||||
|
import ca.uhn.fhir.jpa.model.entity.StorageSettings;
|
||||||
import ca.uhn.fhir.jpa.searchparam.config.NicknameServiceConfig;
|
import ca.uhn.fhir.jpa.searchparam.config.NicknameServiceConfig;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.channel.subscription.SubscriptionChannelFactory;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.match.matcher.matching.IResourceModifiedConsumer;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CircularQueueCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
import ca.uhn.fhir.jpa.util.CurrentThreadCaptureQueriesListener;
|
||||||
import ca.uhn.fhir.rest.api.Constants;
|
import ca.uhn.fhir.rest.api.Constants;
|
||||||
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
import ca.uhn.fhir.rest.server.interceptor.RequestValidatingInterceptor;
|
||||||
|
import ca.uhn.fhir.subscription.api.IResourceModifiedMessagePersistenceSvc;
|
||||||
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
import ca.uhn.fhir.system.HapiTestSystemProperties;
|
||||||
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
import ca.uhn.fhir.validation.ResultSeverityEnum;
|
||||||
import net.ttddyy.dsproxy.listener.SingleQueryCountHolder;
|
import net.ttddyy.dsproxy.listener.SingleQueryCountHolder;
|
||||||
|
|
|
@ -0,0 +1,48 @@
|
||||||
|
/*-
|
||||||
|
* #%L
|
||||||
|
* HAPI FHIR JPA Server Test Utilities
|
||||||
|
* %%
|
||||||
|
* Copyright (C) 2014 - 2023 Smile CDR, Inc.
|
||||||
|
* %%
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
* #L%
|
||||||
|
*/
|
||||||
|
package ca.uhn.fhir.jpa.test.config;
|
||||||
|
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SynchronousSubscriptionMatcherInterceptor;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.context.annotation.Primary;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Production environments submit modified resources to the subscription processing pipeline asynchronously, ie, a
|
||||||
|
* modified resource is 'planned' for submission which is performed at a later time by a scheduled task.
|
||||||
|
*
|
||||||
|
* The purpose of this class is to provide submission of modified resources during tests since task scheduling required
|
||||||
|
* for asynchronous submission are either disabled or not present in testing context.
|
||||||
|
*
|
||||||
|
* Careful consideration is advised when configuring test context as the SubscriptionMatcherInterceptor Bean instantiated
|
||||||
|
* below will overwrite the Bean provided by class SubscriptionMatcherInterceptorConfig if both configuration classes
|
||||||
|
* are present in the context.
|
||||||
|
*/
|
||||||
|
@Configuration
|
||||||
|
public class TestSubscriptionMatcherInterceptorConfig {
|
||||||
|
|
||||||
|
@Primary
|
||||||
|
@Bean
|
||||||
|
public SubscriptionMatcherInterceptor subscriptionMatcherInterceptor() {
|
||||||
|
return new SynchronousSubscriptionMatcherInterceptor();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -29,8 +29,8 @@ import ca.uhn.fhir.jpa.subscription.match.deliver.email.EmailSenderImpl;
|
||||||
import ca.uhn.fhir.jpa.subscription.match.deliver.email.SubscriptionDeliveringEmailSubscriber;
|
import ca.uhn.fhir.jpa.subscription.match.deliver.email.SubscriptionDeliveringEmailSubscriber;
|
||||||
import ca.uhn.fhir.jpa.subscription.match.registry.ActiveSubscription;
|
import ca.uhn.fhir.jpa.subscription.match.registry.ActiveSubscription;
|
||||||
import ca.uhn.fhir.jpa.subscription.match.registry.SubscriptionRegistry;
|
import ca.uhn.fhir.jpa.subscription.match.registry.SubscriptionRegistry;
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionMatcherInterceptor;
|
|
||||||
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionSubmitInterceptorLoader;
|
import ca.uhn.fhir.jpa.subscription.submit.interceptor.SubscriptionSubmitInterceptorLoader;
|
||||||
|
import ca.uhn.fhir.jpa.subscription.submit.svc.ResourceModifiedSubmitterSvc;
|
||||||
import ca.uhn.fhir.jpa.subscription.util.SubscriptionDebugLogInterceptor;
|
import ca.uhn.fhir.jpa.subscription.util.SubscriptionDebugLogInterceptor;
|
||||||
import org.hl7.fhir.dstu2.model.Subscription;
|
import org.hl7.fhir.dstu2.model.Subscription;
|
||||||
import org.hl7.fhir.instance.model.api.IIdType;
|
import org.hl7.fhir.instance.model.api.IIdType;
|
||||||
|
@ -45,7 +45,7 @@ public class SubscriptionTestUtil {
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionSubmitInterceptorLoader mySubscriptionSubmitInterceptorLoader;
|
private SubscriptionSubmitInterceptorLoader mySubscriptionSubmitInterceptorLoader;
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionMatcherInterceptor mySubscriptionMatcherInterceptor;
|
private ResourceModifiedSubmitterSvc myResourceModifiedSubmitterSvc;
|
||||||
@Autowired
|
@Autowired
|
||||||
private SubscriptionRegistry mySubscriptionRegistry;
|
private SubscriptionRegistry mySubscriptionRegistry;
|
||||||
@Autowired
|
@Autowired
|
||||||
|
@ -56,7 +56,7 @@ public class SubscriptionTestUtil {
|
||||||
private IInterceptorService myInterceptorRegistry;
|
private IInterceptorService myInterceptorRegistry;
|
||||||
|
|
||||||
public int getExecutorQueueSize() {
|
public int getExecutorQueueSize() {
|
||||||
LinkedBlockingChannel channel = mySubscriptionMatcherInterceptor.getProcessingChannelForUnitTest();
|
LinkedBlockingChannel channel = (LinkedBlockingChannel) myResourceModifiedSubmitterSvc.getProcessingChannelForUnitTest();
|
||||||
return channel.getQueueSizeForUnitTest();
|
return channel.getQueueSizeForUnitTest();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir</artifactId>
|
<artifactId>hapi-fhir</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -23,8 +23,11 @@ import ca.uhn.fhir.i18n.Msg;
|
||||||
import ca.uhn.fhir.model.api.IModelJson;
|
import ca.uhn.fhir.model.api.IModelJson;
|
||||||
import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
|
import ca.uhn.fhir.rest.api.RestOperationTypeEnum;
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import com.google.common.annotations.VisibleForTesting;
|
||||||
|
import org.apache.commons.lang3.ObjectUtils;
|
||||||
import org.apache.commons.lang3.Validate;
|
import org.apache.commons.lang3.Validate;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
|
@ -229,4 +232,9 @@ public abstract class BaseResourceMessage implements IResourceMessage, IModelJso
|
||||||
return myRestOperationTypeEnum;
|
return myRestOperationTypeEnum;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@VisibleForTesting
|
||||||
|
public Map<String, String> getAttributes() {
|
||||||
|
return ObjectUtils.defaultIfNull(myAttributes, Collections.emptyMap());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -58,6 +58,9 @@ public abstract class BaseResourceModifiedMessage extends BaseResourceMessage im
|
||||||
@JsonIgnore
|
@JsonIgnore
|
||||||
protected transient String myPayloadType;
|
protected transient String myPayloadType;
|
||||||
|
|
||||||
|
@JsonIgnore
|
||||||
|
protected String myPayloadVersion;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor
|
* Constructor
|
||||||
*/
|
*/
|
||||||
|
@ -101,6 +104,10 @@ public abstract class BaseResourceModifiedMessage extends BaseResourceMessage im
|
||||||
return myPayloadId;
|
return myPayloadId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getPayloadVersion() {
|
||||||
|
return myPayloadVersion;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @since 5.6.0
|
* @since 5.6.0
|
||||||
*/
|
*/
|
||||||
|
@ -108,6 +115,7 @@ public abstract class BaseResourceModifiedMessage extends BaseResourceMessage im
|
||||||
myPayloadId = null;
|
myPayloadId = null;
|
||||||
if (thePayloadId != null) {
|
if (thePayloadId != null) {
|
||||||
myPayloadId = thePayloadId.toUnqualifiedVersionless().getValue();
|
myPayloadId = thePayloadId.toUnqualifiedVersionless().getValue();
|
||||||
|
myPayloadVersion = thePayloadId.getVersionIdPart();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -138,9 +146,11 @@ public abstract class BaseResourceModifiedMessage extends BaseResourceMessage im
|
||||||
*/
|
*/
|
||||||
public IIdType getPayloadId(FhirContext theCtx) {
|
public IIdType getPayloadId(FhirContext theCtx) {
|
||||||
IIdType retVal = null;
|
IIdType retVal = null;
|
||||||
|
|
||||||
if (myPayloadId != null) {
|
if (myPayloadId != null) {
|
||||||
retVal = theCtx.getVersion().newIdType().setValue(myPayloadId);
|
retVal = theCtx.getVersion().newIdType().setValue(myPayloadId).withVersion(myPayloadVersion);
|
||||||
}
|
}
|
||||||
|
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -172,7 +182,7 @@ public abstract class BaseResourceModifiedMessage extends BaseResourceMessage im
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void setNewPayload(FhirContext theCtx, IBaseResource thePayload) {
|
public void setNewPayload(FhirContext theCtx, IBaseResource thePayload) {
|
||||||
/*
|
/*
|
||||||
* References with placeholders would be invalid by the time we get here, and
|
* References with placeholders would be invalid by the time we get here, and
|
||||||
* would be caught before we even get here. This check is basically a last-ditch
|
* would be caught before we even get here. This check is basically a last-ditch
|
||||||
|
@ -246,7 +256,7 @@ public abstract class BaseResourceModifiedMessage extends BaseResourceMessage im
|
||||||
@Nullable
|
@Nullable
|
||||||
@Override
|
@Override
|
||||||
public String getMessageKeyOrDefault() {
|
public String getMessageKeyOrDefault() {
|
||||||
return StringUtils.defaultString(super.getMessageKey(), myPayloadId);
|
return StringUtils.defaultString(super.getMessageKeyOrNull(), myPayloadId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean hasPayloadType(FhirContext theFhirContext, @Nonnull String theResourceName) {
|
public boolean hasPayloadType(FhirContext theFhirContext, @Nonnull String theResourceName) {
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<artifactId>hapi-fhir-serviceloaders</artifactId>
|
<artifactId>hapi-fhir-serviceloaders</artifactId>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<artifactId>hapi-fhir-serviceloaders</artifactId>
|
<artifactId>hapi-fhir-serviceloaders</artifactId>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
@ -21,7 +21,7 @@
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir-caching-api</artifactId>
|
<artifactId>hapi-fhir-caching-api</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<artifactId>hapi-fhir-serviceloaders</artifactId>
|
<artifactId>hapi-fhir-serviceloaders</artifactId>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<artifactId>hapi-fhir</artifactId>
|
<artifactId>hapi-fhir</artifactId>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../../pom.xml</relativePath>
|
<relativePath>../../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir-spring-boot-samples</artifactId>
|
<artifactId>hapi-fhir-spring-boot-samples</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<artifactId>hapi-fhir-spring-boot-sample-client-apache</artifactId>
|
<artifactId>hapi-fhir-spring-boot-sample-client-apache</artifactId>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir-spring-boot-samples</artifactId>
|
<artifactId>hapi-fhir-spring-boot-samples</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir-spring-boot-samples</artifactId>
|
<artifactId>hapi-fhir-spring-boot-samples</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir-spring-boot</artifactId>
|
<artifactId>hapi-fhir-spring-boot</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-fhir</artifactId>
|
<artifactId>hapi-fhir</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../pom.xml</relativePath>
|
<relativePath>../pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>ca.uhn.hapi.fhir</groupId>
|
<groupId>ca.uhn.hapi.fhir</groupId>
|
||||||
<artifactId>hapi-deployable-pom</artifactId>
|
<artifactId>hapi-deployable-pom</artifactId>
|
||||||
<version>6.9.3-SNAPSHOT</version>
|
<version>6.9.4-SNAPSHOT</version>
|
||||||
|
|
||||||
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
<relativePath>../hapi-deployable-pom/pom.xml</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue