All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ca.uhn.fhir.jpa.mdm.config.MdmSubscriptionLoader Maven / Gradle / Ivy

The newest version!
/*-
 * #%L
 * HAPI FHIR JPA Server - Master Data Management
 * %%
 * Copyright (C) 2014 - 2024 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.mdm.config;

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.subscription.channel.api.ChannelProducerSettings;
import ca.uhn.fhir.jpa.subscription.channel.subscription.IChannelNamer;
import ca.uhn.fhir.jpa.subscription.match.registry.SubscriptionLoader;
import ca.uhn.fhir.jpa.subscription.model.CanonicalSubscriptionChannelType;
import ca.uhn.fhir.jpa.topic.SubscriptionTopicLoader;
import ca.uhn.fhir.mdm.api.IMdmSettings;
import ca.uhn.fhir.mdm.api.MdmConstants;
import ca.uhn.fhir.mdm.log.Logs;
import ca.uhn.fhir.rest.api.Constants;
import ca.uhn.fhir.rest.api.server.SystemRequestDetails;
import ca.uhn.fhir.rest.server.exceptions.ResourceGoneException;
import ca.uhn.fhir.rest.server.exceptions.ResourceNotFoundException;
import ca.uhn.fhir.util.HapiExtensions;
import org.hl7.fhir.instance.model.api.IBaseResource;
import org.hl7.fhir.r4.model.BooleanType;
import org.hl7.fhir.r4.model.Subscription;
import org.hl7.fhir.r5.model.Coding;
import org.hl7.fhir.r5.model.Enumerations;
import org.hl7.fhir.r5.model.SubscriptionTopic;
import org.slf4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

import java.util.Collections;
import java.util.List;
import java.util.stream.Collectors;

@Service
public class MdmSubscriptionLoader {

	public static final String MDM_SUBSCRIPTION_ID_PREFIX = "mdm-";
	private static final Logger ourLog = Logs.getMdmTroubleshootingLog();

	@Autowired
	public FhirContext myFhirContext;

	@Autowired
	public DaoRegistry myDaoRegistry;

	@Autowired
	IChannelNamer myChannelNamer;

	@Autowired
	private SubscriptionLoader mySubscriptionLoader;

	@Autowired
	private IMdmSettings myMdmSettings;

	@Autowired(required = false)
	private SubscriptionTopicLoader mySubscriptionTopicLoader;

	private IFhirResourceDao mySubscriptionDao;
	private IFhirResourceDao mySubscriptionTopicDao;

	public synchronized void daoUpdateMdmSubscriptions() {
		List subscriptions;
		List mdmResourceTypes = myMdmSettings.getMdmRules().getMdmTypes();
		switch (myFhirContext.getVersion().getVersion()) {
			case DSTU3:
				subscriptions = mdmResourceTypes.stream()
						.map(resourceType -> buildMdmSubscriptionDstu3(
								MDM_SUBSCRIPTION_ID_PREFIX + resourceType, resourceType + "?"))
						.collect(Collectors.toList());
				break;
			case R4:
				subscriptions = mdmResourceTypes.stream()
						.map(resourceType ->
								buildMdmSubscriptionR4(MDM_SUBSCRIPTION_ID_PREFIX + resourceType, resourceType + "?"))
						.collect(Collectors.toList());
				break;
			case R5:
				SubscriptionTopic subscriptionTopic = buildMdmSubscriptionTopicR5(mdmResourceTypes);
				updateSubscriptionTopic(subscriptionTopic);
				// After loading subscriptionTopic, sync subscriptionTopic to the registry.
				mySubscriptionTopicLoader.syncDatabaseToCache();

				subscriptions = buildMdmSubscriptionR5(subscriptionTopic);
				break;
			default:
				throw new ConfigurationException(Msg.code(736) + "MDM not supported for FHIR version "
						+ myFhirContext.getVersion().getVersion());
		}

		mySubscriptionDao = myDaoRegistry.getResourceDao("Subscription");
		for (IBaseResource subscription : subscriptions) {
			updateIfNotPresent(subscription);
		}
		// After loading all the subscriptions, sync the subscriptions to the registry.
		if (subscriptions != null && subscriptions.size() > 0) {
			mySubscriptionLoader.syncDatabaseToCache();
		}
	}

	synchronized void updateIfNotPresent(IBaseResource theSubscription) {
		try {
			mySubscriptionDao.read(theSubscription.getIdElement(), SystemRequestDetails.forAllPartitions());
		} catch (ResourceNotFoundException | ResourceGoneException e) {
			ourLog.info("Creating subscription " + theSubscription.getIdElement());
			mySubscriptionDao.update(theSubscription, SystemRequestDetails.forAllPartitions());
		}
	}

	synchronized void updateSubscriptionTopic(SubscriptionTopic theSubscriptionTopic) {
		mySubscriptionTopicDao = myDaoRegistry.getResourceDao("SubscriptionTopic");
		mySubscriptionTopicDao.update(theSubscriptionTopic, SystemRequestDetails.forAllPartitions());
	}

	private org.hl7.fhir.dstu3.model.Subscription buildMdmSubscriptionDstu3(String theId, String theCriteria) {
		org.hl7.fhir.dstu3.model.Subscription retval = new org.hl7.fhir.dstu3.model.Subscription();
		retval.setId(theId);
		retval.setReason("MDM");
		retval.setStatus(org.hl7.fhir.dstu3.model.Subscription.SubscriptionStatus.REQUESTED);
		retval.setCriteria(theCriteria);
		retval.getMeta()
				.addTag()
				.setSystem(MdmConstants.SYSTEM_MDM_MANAGED)
				.setCode(MdmConstants.CODE_HAPI_MDM_MANAGED);
		retval.addExtension()
				.setUrl(HapiExtensions.EXTENSION_SUBSCRIPTION_CROSS_PARTITION)
				.setValue(new org.hl7.fhir.dstu3.model.BooleanType().setValue(true));
		org.hl7.fhir.dstu3.model.Subscription.SubscriptionChannelComponent channel = retval.getChannel();
		channel.setType(org.hl7.fhir.dstu3.model.Subscription.SubscriptionChannelType.MESSAGE);
		channel.setEndpoint("channel:"
				+ myChannelNamer.getChannelName(IMdmSettings.EMPI_CHANNEL_NAME, new ChannelProducerSettings()));
		channel.setPayload(Constants.CT_JSON);
		return retval;
	}

	private Subscription buildMdmSubscriptionR4(String theId, String theCriteria) {
		Subscription retval = new Subscription();
		retval.setId(theId);
		retval.setReason("MDM");
		retval.setStatus(Subscription.SubscriptionStatus.REQUESTED);
		retval.setCriteria(theCriteria);
		retval.getMeta()
				.addTag()
				.setSystem(MdmConstants.SYSTEM_MDM_MANAGED)
				.setCode(MdmConstants.CODE_HAPI_MDM_MANAGED);
		retval.addExtension()
				.setUrl(HapiExtensions.EXTENSION_SUBSCRIPTION_CROSS_PARTITION)
				.setValue(new BooleanType().setValue(true));
		Subscription.SubscriptionChannelComponent channel = retval.getChannel();
		channel.setType(Subscription.SubscriptionChannelType.MESSAGE);
		channel.setEndpoint("channel:"
				+ myChannelNamer.getChannelName(IMdmSettings.EMPI_CHANNEL_NAME, new ChannelProducerSettings()));
		channel.setPayload(Constants.CT_JSON);
		return retval;
	}

	private SubscriptionTopic buildMdmSubscriptionTopicR5(List theMdmResourceTypes) {
		SubscriptionTopic subscriptionTopic = new SubscriptionTopic();
		subscriptionTopic.setId(MDM_SUBSCRIPTION_ID_PREFIX + "subscription-topic");
		subscriptionTopic
				.getMeta()
				.addTag()
				.setSystem(MdmConstants.SYSTEM_MDM_MANAGED)
				.setCode(MdmConstants.CODE_HAPI_MDM_MANAGED);
		subscriptionTopic.setStatus(Enumerations.PublicationStatus.ACTIVE);
		subscriptionTopic.setUrl(MdmConstants.SUBSCRIPTION_TOPIC_URL);
		theMdmResourceTypes.forEach(
				resourceType -> buildSubscriptionTopicResourceTriggerComponent(resourceType, subscriptionTopic));
		return subscriptionTopic;
	}

	private static void buildSubscriptionTopicResourceTriggerComponent(
			String theResourceType, SubscriptionTopic theSubscriptionTopic) {
		SubscriptionTopic.SubscriptionTopicResourceTriggerComponent triggerComponent =
				theSubscriptionTopic.addResourceTrigger();
		triggerComponent.setResource(theResourceType);
		triggerComponent.addSupportedInteraction(SubscriptionTopic.InteractionTrigger.CREATE);
		triggerComponent.addSupportedInteraction(SubscriptionTopic.InteractionTrigger.UPDATE);
	}

	private List buildMdmSubscriptionR5(SubscriptionTopic theSubscriptionTopic) {
		org.hl7.fhir.r5.model.Subscription subscription = new org.hl7.fhir.r5.model.Subscription();

		subscription.setId(MDM_SUBSCRIPTION_ID_PREFIX + "subscription");
		subscription.setReason("MDM");
		subscription.setStatus(Enumerations.SubscriptionStatusCodes.REQUESTED);

		subscription.setTopic(theSubscriptionTopic.getUrl());
		subscription
				.getMeta()
				.addTag()
				.setSystem(MdmConstants.SYSTEM_MDM_MANAGED)
				.setCode(MdmConstants.CODE_HAPI_MDM_MANAGED);

		subscription.setChannelType(new Coding()
				.setCode(CanonicalSubscriptionChannelType.MESSAGE.toCode())
				.setSystem(CanonicalSubscriptionChannelType.MESSAGE.getSystem()));

		subscription.setEndpoint("channel:"
				+ myChannelNamer.getChannelName(IMdmSettings.EMPI_CHANNEL_NAME, new ChannelProducerSettings()));
		subscription.setContentType(Constants.CT_JSON);

		return Collections.singletonList(subscription);
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy