001/*-
002 * #%L
003 * HAPI FHIR Storage api
004 * %%
005 * Copyright (C) 2014 - 2025 Smile CDR, Inc.
006 * %%
007 * Licensed under the Apache License, Version 2.0 (the "License");
008 * you may not use this file except in compliance with the License.
009 * You may obtain a copy of the License at
010 *
011 *      http://www.apache.org/licenses/LICENSE-2.0
012 *
013 * Unless required by applicable law or agreed to in writing, software
014 * distributed under the License is distributed on an "AS IS" BASIS,
015 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
016 * See the License for the specific language governing permissions and
017 * limitations under the License.
018 * #L%
019 */
020package ca.uhn.fhir.jpa.subscription.match.registry;
021
022import ca.uhn.fhir.context.ConfigurationException;
023import ca.uhn.fhir.context.FhirContext;
024import ca.uhn.fhir.i18n.Msg;
025import ca.uhn.fhir.interceptor.model.RequestPartitionId;
026import ca.uhn.fhir.jpa.model.config.SubscriptionSettings;
027import ca.uhn.fhir.jpa.subscription.match.matcher.matching.SubscriptionMatchingStrategy;
028import ca.uhn.fhir.jpa.subscription.model.CanonicalSubscription;
029import ca.uhn.fhir.jpa.subscription.model.CanonicalSubscriptionChannelType;
030import ca.uhn.fhir.jpa.subscription.model.CanonicalTopicSubscription;
031import ca.uhn.fhir.jpa.subscription.model.CanonicalTopicSubscriptionFilter;
032import ca.uhn.fhir.model.api.BasePrimitive;
033import ca.uhn.fhir.model.api.ExtensionDt;
034import ca.uhn.fhir.model.dstu2.resource.Subscription;
035import ca.uhn.fhir.model.primitive.BooleanDt;
036import ca.uhn.fhir.rest.api.Constants;
037import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
038import ca.uhn.fhir.rest.server.exceptions.PreconditionFailedException;
039import ca.uhn.fhir.subscription.SubscriptionConstants;
040import ca.uhn.fhir.util.HapiExtensions;
041import ca.uhn.fhir.util.SubscriptionUtil;
042import jakarta.annotation.Nonnull;
043import jakarta.annotation.Nullable;
044import org.hl7.fhir.exceptions.FHIRException;
045import org.hl7.fhir.instance.model.api.IBaseHasExtensions;
046import org.hl7.fhir.instance.model.api.IBaseMetaType;
047import org.hl7.fhir.instance.model.api.IBaseReference;
048import org.hl7.fhir.instance.model.api.IBaseResource;
049import org.hl7.fhir.instance.model.api.IPrimitiveType;
050import org.hl7.fhir.r4.model.BooleanType;
051import org.hl7.fhir.r4.model.Extension;
052import org.hl7.fhir.r5.model.Enumerations;
053import org.slf4j.Logger;
054import org.slf4j.LoggerFactory;
055import org.springframework.beans.factory.annotation.Autowired;
056
057import java.util.Collections;
058import java.util.HashMap;
059import java.util.List;
060import java.util.Map;
061import java.util.stream.Collectors;
062
063import static ca.uhn.fhir.util.HapiExtensions.EX_SEND_DELETE_MESSAGES;
064import static java.util.Objects.nonNull;
065import static java.util.stream.Collectors.mapping;
066import static java.util.stream.Collectors.toList;
067
068public class SubscriptionCanonicalizer {
069        private static final Logger ourLog = LoggerFactory.getLogger(SubscriptionCanonicalizer.class);
070
071        final FhirContext myFhirContext;
072        private final SubscriptionSettings mySubscriptionSettings;
073
074        @Autowired
075        public SubscriptionCanonicalizer(FhirContext theFhirContext, SubscriptionSettings theSubscriptionSettings) {
076                myFhirContext = theFhirContext;
077                mySubscriptionSettings = theSubscriptionSettings;
078        }
079
080        // TODO:  LD:  remove this constructor once all callers call the 2 arg constructor above
081
082        /**
083         * @deprecated All callers should invoke {@link SubscriptionCanonicalizer()} instead.
084         */
085        @Deprecated
086        public SubscriptionCanonicalizer(FhirContext theFhirContext) {
087                myFhirContext = theFhirContext;
088                mySubscriptionSettings = new SubscriptionSettings();
089        }
090
091        public CanonicalSubscription canonicalize(IBaseResource theSubscription) {
092                switch (myFhirContext.getVersion().getVersion()) {
093                        case DSTU2:
094                                return canonicalizeDstu2(theSubscription);
095                        case DSTU3:
096                                return canonicalizeDstu3(theSubscription);
097                        case R4:
098                                return canonicalizeR4(theSubscription);
099                        case R4B:
100                                return canonicalizeR4B(theSubscription);
101                        case R5:
102                                return canonicalizeR5(theSubscription);
103                        case DSTU2_HL7ORG:
104                        case DSTU2_1:
105                        default:
106                                throw new ConfigurationException(Msg.code(556) + "Subscription not supported for version: "
107                                                + myFhirContext.getVersion().getVersion());
108                }
109        }
110
111        private CanonicalSubscription canonicalizeDstu2(IBaseResource theSubscription) {
112                ca.uhn.fhir.model.dstu2.resource.Subscription subscription =
113                                (ca.uhn.fhir.model.dstu2.resource.Subscription) theSubscription;
114                CanonicalSubscription retVal = new CanonicalSubscription();
115                try {
116                        retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.fromCode(subscription.getStatus()));
117                        retVal.setChannelType(getChannelType(theSubscription));
118                        retVal.setCriteriaString(subscription.getCriteria());
119                        Subscription.Channel channel = subscription.getChannel();
120                        retVal.setEndpointUrl(channel.getEndpoint());
121                        retVal.setHeaders(channel.getHeader());
122                        retVal.setChannelExtensions(extractExtension(subscription));
123                        retVal.setIdElement(subscription.getIdElement());
124                        retVal.setPayloadString(channel.getPayload());
125                        retVal.setTags(extractTags(subscription));
126                        retVal.setCrossPartitionEnabled(handleCrossPartition(theSubscription));
127                        retVal.setSendDeleteMessages(extractDeleteExtensionDstu2(subscription));
128                } catch (FHIRException theE) {
129                        throw new InternalErrorException(Msg.code(557) + theE);
130                }
131                return retVal;
132        }
133
134        private boolean extractDeleteExtensionDstu2(ca.uhn.fhir.model.dstu2.resource.Subscription theSubscription) {
135                return theSubscription.getChannel().getUndeclaredExtensionsByUrl(EX_SEND_DELETE_MESSAGES).stream()
136                                .map(ExtensionDt::getValue)
137                                .map(BooleanDt.class::cast)
138                                .map(BasePrimitive::getValue)
139                                .findFirst()
140                                .orElse(false);
141        }
142
143        /**
144         * Extract the meta tags from the subscription and convert them to a simple string map.
145         *
146         * @param theSubscription The subscription to extract the tags from
147         * @return A map of tags System:Code
148         */
149        private Map<String, String> extractTags(IBaseResource theSubscription) {
150                Map<String, String> retVal = new HashMap<>();
151                theSubscription.getMeta().getTag().stream()
152                                .filter(t -> t.getSystem() != null && t.getCode() != null)
153                                .forEach(t -> retVal.put(t.getSystem(), t.getCode()));
154                return retVal;
155        }
156
157        private CanonicalSubscription canonicalizeDstu3(IBaseResource theSubscription) {
158                org.hl7.fhir.dstu3.model.Subscription subscription = (org.hl7.fhir.dstu3.model.Subscription) theSubscription;
159
160                CanonicalSubscription retVal = new CanonicalSubscription();
161                try {
162                        org.hl7.fhir.dstu3.model.Subscription.SubscriptionStatus status = subscription.getStatus();
163                        if (status != null) {
164                                retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.fromCode(status.toCode()));
165                        }
166                        setPartitionIdOnReturnValue(theSubscription, retVal);
167                        retVal.setChannelType(getChannelType(theSubscription));
168                        retVal.setCriteriaString(subscription.getCriteria());
169                        org.hl7.fhir.dstu3.model.Subscription.SubscriptionChannelComponent channel = subscription.getChannel();
170                        retVal.setEndpointUrl(channel.getEndpoint());
171                        retVal.setHeaders(channel.getHeader());
172                        retVal.setChannelExtensions(extractExtension(subscription));
173                        retVal.setIdElement(subscription.getIdElement());
174                        retVal.setPayloadString(channel.getPayload());
175                        retVal.setPayloadSearchCriteria(
176                                        getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_PAYLOAD_SEARCH_CRITERIA));
177                        retVal.setTags(extractTags(subscription));
178                        retVal.setCrossPartitionEnabled(handleCrossPartition(theSubscription));
179
180                        if (retVal.getChannelType() == CanonicalSubscriptionChannelType.EMAIL) {
181                                String from;
182                                String subjectTemplate;
183
184                                try {
185                                        from = channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_EMAIL_FROM);
186                                        subjectTemplate = channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_SUBJECT_TEMPLATE);
187                                } catch (FHIRException theE) {
188                                        throw new ConfigurationException(
189                                                        Msg.code(558) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
190                                }
191                                retVal.getEmailDetails().setFrom(from);
192                                retVal.getEmailDetails().setSubjectTemplate(subjectTemplate);
193                        }
194
195                        if (retVal.getChannelType() == CanonicalSubscriptionChannelType.RESTHOOK) {
196
197                                String stripVersionIds;
198                                String deliverLatestVersion;
199                                try {
200                                        stripVersionIds =
201                                                        channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_STRIP_VERSION_IDS);
202                                        deliverLatestVersion =
203                                                        channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_DELIVER_LATEST_VERSION);
204                                } catch (FHIRException theE) {
205                                        throw new ConfigurationException(
206                                                        Msg.code(559) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
207                                }
208                                retVal.getRestHookDetails().setStripVersionId(Boolean.parseBoolean(stripVersionIds));
209                                retVal.getRestHookDetails().setDeliverLatestVersion(Boolean.parseBoolean(deliverLatestVersion));
210                        }
211                        retVal.setSendDeleteMessages(extractSendDeletesDstu3(subscription));
212
213                } catch (FHIRException theE) {
214                        throw new InternalErrorException(Msg.code(560) + theE);
215                }
216                return retVal;
217        }
218
219        private Boolean extractSendDeletesDstu3(org.hl7.fhir.dstu3.model.Subscription subscription) {
220                return subscription.getChannel().getExtensionsByUrl(EX_SEND_DELETE_MESSAGES).stream()
221                                .map(org.hl7.fhir.dstu3.model.Extension::getValue)
222                                .filter(val -> val instanceof org.hl7.fhir.dstu3.model.BooleanType)
223                                .map(val -> (org.hl7.fhir.dstu3.model.BooleanType) val)
224                                .map(org.hl7.fhir.dstu3.model.BooleanType::booleanValue)
225                                .findFirst()
226                                .orElse(false);
227        }
228
229        private @Nonnull Map<String, List<String>> extractExtension(IBaseResource theSubscription) {
230                try {
231                        switch (theSubscription.getStructureFhirVersionEnum()) {
232                                case DSTU2: {
233                                        ca.uhn.fhir.model.dstu2.resource.Subscription subscription =
234                                                        (ca.uhn.fhir.model.dstu2.resource.Subscription) theSubscription;
235                                        return subscription.getChannel().getUndeclaredExtensions().stream()
236                                                        .collect(Collectors.groupingBy(
237                                                                        t -> t.getUrl(),
238                                                                        mapping(t -> t.getValueAsPrimitive().getValueAsString(), toList())));
239                                }
240                                case DSTU3: {
241                                        org.hl7.fhir.dstu3.model.Subscription subscription =
242                                                        (org.hl7.fhir.dstu3.model.Subscription) theSubscription;
243                                        return subscription.getChannel().getExtension().stream()
244                                                        .collect(Collectors.groupingBy(
245                                                                        t -> t.getUrl(),
246                                                                        mapping(t -> t.getValueAsPrimitive().getValueAsString(), toList())));
247                                }
248                                case R4: {
249                                        org.hl7.fhir.r4.model.Subscription subscription =
250                                                        (org.hl7.fhir.r4.model.Subscription) theSubscription;
251                                        return subscription.getChannel().getExtension().stream()
252                                                        .collect(Collectors.groupingBy(
253                                                                        t -> t.getUrl(),
254                                                                        mapping(
255                                                                                        t -> {
256                                                                                                return t.getValueAsPrimitive().getValueAsString();
257                                                                                        },
258                                                                                        toList())));
259                                }
260                                case R5: {
261                                        // TODO KHS fix org.hl7.fhir.r4b.model.BaseResource.getStructureFhirVersionEnum() for R4B
262                                        if (theSubscription instanceof org.hl7.fhir.r4b.model.Subscription) {
263                                                org.hl7.fhir.r4b.model.Subscription subscription =
264                                                                (org.hl7.fhir.r4b.model.Subscription) theSubscription;
265                                                return subscription.getExtension().stream()
266                                                                .collect(Collectors.groupingBy(
267                                                                                t -> t.getUrl(),
268                                                                                mapping(t -> t.getValueAsPrimitive().getValueAsString(), toList())));
269                                        } else if (theSubscription instanceof org.hl7.fhir.r5.model.Subscription) {
270                                                org.hl7.fhir.r5.model.Subscription subscription =
271                                                                (org.hl7.fhir.r5.model.Subscription) theSubscription;
272                                                return subscription.getExtension().stream()
273                                                                .collect(Collectors.groupingBy(
274                                                                                t -> t.getUrl(),
275                                                                                mapping(t -> t.getValueAsPrimitive().getValueAsString(), toList())));
276                                        }
277                                }
278                                case DSTU2_HL7ORG:
279                                case DSTU2_1:
280                                default: {
281                                        ourLog.error(
282                                                        "Failed to extract extension from subscription {}",
283                                                        theSubscription.getIdElement().toUnqualified().getValue());
284                                        break;
285                                }
286                        }
287                } catch (FHIRException theE) {
288                        ourLog.error(
289                                        "Failed to extract extension from subscription {}",
290                                        theSubscription.getIdElement().toUnqualified().getValue(),
291                                        theE);
292                }
293                return Collections.emptyMap();
294        }
295
296        private CanonicalSubscription canonicalizeR4(IBaseResource theSubscription) {
297                org.hl7.fhir.r4.model.Subscription subscription = (org.hl7.fhir.r4.model.Subscription) theSubscription;
298                CanonicalSubscription retVal = new CanonicalSubscription();
299                retVal.setStatus(subscription.getStatus());
300                org.hl7.fhir.r4.model.Subscription.SubscriptionChannelComponent channel = subscription.getChannel();
301                retVal.setHeaders(channel.getHeader());
302                retVal.setChannelExtensions(extractExtension(subscription));
303                retVal.setIdElement(subscription.getIdElement());
304                retVal.setPayloadString(channel.getPayload());
305                retVal.setPayloadSearchCriteria(
306                                getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_PAYLOAD_SEARCH_CRITERIA));
307                retVal.setTags(extractTags(subscription));
308                setPartitionIdOnReturnValue(theSubscription, retVal);
309                retVal.setCrossPartitionEnabled(handleCrossPartition(theSubscription));
310
311                List<org.hl7.fhir.r4.model.CanonicalType> profiles =
312                                subscription.getMeta().getProfile();
313                for (org.hl7.fhir.r4.model.CanonicalType next : profiles) {
314                        if (SubscriptionConstants.SUBSCRIPTION_TOPIC_PROFILE_URL.equals(next.getValueAsString())) {
315                                retVal.setTopicSubscription(true);
316                        }
317                }
318
319                if (retVal.isTopicSubscription()) {
320                        CanonicalTopicSubscription topicSubscription = retVal.getTopicSubscription();
321                        topicSubscription.setTopic(getCriteria(theSubscription));
322
323                        retVal.setEndpointUrl(channel.getEndpoint());
324                        retVal.setChannelType(getChannelType(subscription));
325
326                        for (org.hl7.fhir.r4.model.Extension next :
327                                        subscription.getCriteriaElement().getExtension()) {
328                                if (SubscriptionConstants.SUBSCRIPTION_TOPIC_FILTER_URL.equals(next.getUrl())) {
329                                        List<CanonicalTopicSubscriptionFilter> filters = CanonicalTopicSubscriptionFilter.fromQueryUrl(
330                                                        next.getValue().primitiveValue());
331                                        filters.forEach(topicSubscription::addFilter);
332                                }
333                        }
334
335                        if (channel.hasExtension(SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_HEARTBEAT_PERIOD_URL)) {
336                                org.hl7.fhir.r4.model.Extension channelHeartbeatPeriotUrlExtension = channel.getExtensionByUrl(
337                                                SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_HEARTBEAT_PERIOD_URL);
338                                topicSubscription.setHeartbeatPeriod(Integer.valueOf(
339                                                channelHeartbeatPeriotUrlExtension.getValue().primitiveValue()));
340                        }
341                        if (channel.hasExtension(SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_TIMEOUT_URL)) {
342                                org.hl7.fhir.r4.model.Extension channelTimeoutUrlExtension =
343                                                channel.getExtensionByUrl(SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_TIMEOUT_URL);
344                                topicSubscription.setTimeout(
345                                                Integer.valueOf(channelTimeoutUrlExtension.getValue().primitiveValue()));
346                        }
347                        if (channel.hasExtension(SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_MAX_COUNT)) {
348                                org.hl7.fhir.r4.model.Extension channelMaxCountExtension =
349                                                channel.getExtensionByUrl(SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_MAX_COUNT);
350                                topicSubscription.setMaxCount(
351                                                Integer.valueOf(channelMaxCountExtension.getValue().primitiveValue()));
352                        }
353
354                        // setting full-resource PayloadContent if backport-payload-content is not provided
355                        org.hl7.fhir.r5.model.Subscription.SubscriptionPayloadContent payloadContent =
356                                        org.hl7.fhir.r5.model.Subscription.SubscriptionPayloadContent.FULLRESOURCE;
357
358                        org.hl7.fhir.r4.model.Extension channelPayloadContentExtension = channel.getPayloadElement()
359                                        .getExtensionByUrl(SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_PAYLOAD_CONTENT);
360
361                        if (nonNull(channelPayloadContentExtension)) {
362                                payloadContent = org.hl7.fhir.r5.model.Subscription.SubscriptionPayloadContent.fromCode(
363                                                channelPayloadContentExtension.getValue().primitiveValue());
364                        }
365
366                        topicSubscription.setContent(payloadContent);
367                } else {
368                        retVal.setCriteriaString(getCriteria(theSubscription));
369                        retVal.setEndpointUrl(channel.getEndpoint());
370                        retVal.setChannelType(getChannelType(subscription));
371                }
372
373                if (retVal.getChannelType() == CanonicalSubscriptionChannelType.EMAIL) {
374                        String from;
375                        String subjectTemplate;
376                        try {
377                                from = channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_EMAIL_FROM);
378                                subjectTemplate = channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_SUBJECT_TEMPLATE);
379                        } catch (FHIRException theE) {
380                                throw new ConfigurationException(
381                                                Msg.code(561) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
382                        }
383                        retVal.getEmailDetails().setFrom(from);
384                        retVal.getEmailDetails().setSubjectTemplate(subjectTemplate);
385                }
386
387                if (retVal.getChannelType() == CanonicalSubscriptionChannelType.RESTHOOK) {
388                        String stripVersionIds;
389                        String deliverLatestVersion;
390                        try {
391                                stripVersionIds =
392                                                channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_STRIP_VERSION_IDS);
393                                deliverLatestVersion =
394                                                channel.getExtensionString(HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_DELIVER_LATEST_VERSION);
395                        } catch (FHIRException theE) {
396                                throw new ConfigurationException(
397                                                Msg.code(562) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
398                        }
399                        retVal.getRestHookDetails().setStripVersionId(Boolean.parseBoolean(stripVersionIds));
400                        retVal.getRestHookDetails().setDeliverLatestVersion(Boolean.parseBoolean(deliverLatestVersion));
401                }
402
403                List<Extension> topicExts = subscription.getExtensionsByUrl("http://hl7.org/fhir/subscription/topics");
404                if (!topicExts.isEmpty()) {
405                        IBaseReference ref = (IBaseReference) topicExts.get(0).getValueAsPrimitive();
406                        if (!"EventDefinition".equals(ref.getReferenceElement().getResourceType())) {
407                                throw new PreconditionFailedException(Msg.code(563) + "Topic reference must be an EventDefinition");
408                        }
409                }
410
411                Extension extension = channel.getExtensionByUrl(EX_SEND_DELETE_MESSAGES);
412                if (extension != null && extension.hasValue() && extension.getValue() instanceof BooleanType) {
413                        retVal.setSendDeleteMessages(((BooleanType) extension.getValue()).booleanValue());
414                }
415                return retVal;
416        }
417
418        private CanonicalSubscription canonicalizeR4B(IBaseResource theSubscription) {
419                org.hl7.fhir.r4b.model.Subscription subscription = (org.hl7.fhir.r4b.model.Subscription) theSubscription;
420
421                CanonicalSubscription retVal = new CanonicalSubscription();
422                org.hl7.fhir.r4b.model.Enumerations.SubscriptionStatus status = subscription.getStatus();
423                if (status != null) {
424                        retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.fromCode(status.toCode()));
425                }
426                setPartitionIdOnReturnValue(theSubscription, retVal);
427                org.hl7.fhir.r4b.model.Subscription.SubscriptionChannelComponent channel = subscription.getChannel();
428                retVal.setHeaders(channel.getHeader());
429                retVal.setChannelExtensions(extractExtension(subscription));
430                retVal.setIdElement(subscription.getIdElement());
431                retVal.setPayloadString(channel.getPayload());
432                retVal.setPayloadSearchCriteria(
433                                getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_PAYLOAD_SEARCH_CRITERIA));
434                retVal.setTags(extractTags(subscription));
435
436                List<org.hl7.fhir.r4b.model.CanonicalType> profiles =
437                                subscription.getMeta().getProfile();
438                for (org.hl7.fhir.r4b.model.CanonicalType next : profiles) {
439                        if (SubscriptionConstants.SUBSCRIPTION_TOPIC_PROFILE_URL.equals(next.getValueAsString())) {
440                                retVal.setTopicSubscription(true);
441                        }
442                }
443
444                if (retVal.isTopicSubscription()) {
445                        CanonicalTopicSubscription topicSubscription = retVal.getTopicSubscription();
446                        topicSubscription.setTopic(getCriteria(theSubscription));
447
448                        retVal.setEndpointUrl(channel.getEndpoint());
449                        retVal.setChannelType(getChannelType(subscription));
450
451                        // setting full-resource PayloadContent if backport-payload-content is not provided
452                        org.hl7.fhir.r5.model.Subscription.SubscriptionPayloadContent payloadContent =
453                                        org.hl7.fhir.r5.model.Subscription.SubscriptionPayloadContent.FULLRESOURCE;
454
455                        org.hl7.fhir.r4b.model.Extension channelPayloadContentExtension = channel.getPayloadElement()
456                                        .getExtensionByUrl(SubscriptionConstants.SUBSCRIPTION_TOPIC_CHANNEL_PAYLOAD_CONTENT);
457
458                        if (nonNull(channelPayloadContentExtension)) {
459                                payloadContent = org.hl7.fhir.r5.model.Subscription.SubscriptionPayloadContent.fromCode(
460                                                channelPayloadContentExtension.getValue().primitiveValue());
461                        }
462
463                        topicSubscription.setContent(payloadContent);
464                } else {
465                        retVal.setCriteriaString(getCriteria(theSubscription));
466                        retVal.setEndpointUrl(channel.getEndpoint());
467                        retVal.setChannelType(getChannelType(subscription));
468                }
469
470                if (retVal.getChannelType() == CanonicalSubscriptionChannelType.EMAIL) {
471                        String from;
472                        String subjectTemplate;
473                        try {
474                                from = getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_EMAIL_FROM);
475                                subjectTemplate = getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_SUBJECT_TEMPLATE);
476                        } catch (FHIRException theE) {
477                                throw new ConfigurationException(
478                                                Msg.code(564) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
479                        }
480                        retVal.getEmailDetails().setFrom(from);
481                        retVal.getEmailDetails().setSubjectTemplate(subjectTemplate);
482                }
483
484                if (retVal.getChannelType() == CanonicalSubscriptionChannelType.RESTHOOK) {
485                        String stripVersionIds;
486                        String deliverLatestVersion;
487                        try {
488                                stripVersionIds =
489                                                getExtensionString(channel, HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_STRIP_VERSION_IDS);
490                                deliverLatestVersion =
491                                                getExtensionString(channel, HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_DELIVER_LATEST_VERSION);
492                        } catch (FHIRException theE) {
493                                throw new ConfigurationException(
494                                                Msg.code(565) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
495                        }
496                        retVal.getRestHookDetails().setStripVersionId(Boolean.parseBoolean(stripVersionIds));
497                        retVal.getRestHookDetails().setDeliverLatestVersion(Boolean.parseBoolean(deliverLatestVersion));
498                }
499
500                List<org.hl7.fhir.r4b.model.Extension> topicExts =
501                                subscription.getExtensionsByUrl("http://hl7.org/fhir/subscription/topics");
502                if (!topicExts.isEmpty()) {
503                        IBaseReference ref = (IBaseReference) topicExts.get(0).getValueAsPrimitive();
504                        if (!"EventDefinition".equals(ref.getReferenceElement().getResourceType())) {
505                                throw new PreconditionFailedException(Msg.code(566) + "Topic reference must be an EventDefinition");
506                        }
507                }
508
509                org.hl7.fhir.r4b.model.Extension extension = channel.getExtensionByUrl(EX_SEND_DELETE_MESSAGES);
510                if (extension != null && extension.hasValue() && extension.hasValueBooleanType()) {
511                        retVal.setSendDeleteMessages(extension.getValueBooleanType().booleanValue());
512                }
513
514                retVal.setCrossPartitionEnabled(handleCrossPartition(theSubscription));
515
516                return retVal;
517        }
518
519        private CanonicalSubscription canonicalizeR5(IBaseResource theSubscription) {
520                org.hl7.fhir.r5.model.Subscription subscription = (org.hl7.fhir.r5.model.Subscription) theSubscription;
521
522                CanonicalSubscription retVal = new CanonicalSubscription();
523
524                setPartitionIdOnReturnValue(theSubscription, retVal);
525                retVal.setChannelExtensions(extractExtension(subscription));
526                retVal.setIdElement(subscription.getIdElement());
527                retVal.setPayloadString(subscription.getContentType());
528                retVal.setPayloadSearchCriteria(
529                                getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_PAYLOAD_SEARCH_CRITERIA));
530                retVal.setTags(extractTags(subscription));
531
532                List<org.hl7.fhir.r5.model.Extension> topicExts =
533                                subscription.getExtensionsByUrl("http://hl7.org/fhir/subscription/topics");
534                if (!topicExts.isEmpty()) {
535                        IBaseReference ref = (IBaseReference) topicExts.get(0).getValueAsPrimitive();
536                        if (!"EventDefinition".equals(ref.getReferenceElement().getResourceType())) {
537                                throw new PreconditionFailedException(Msg.code(2325) + "Topic reference must be an EventDefinition");
538                        }
539                }
540
541                // All R5 subscriptions are topic subscriptions
542                retVal.setTopicSubscription(true);
543
544                Enumerations.SubscriptionStatusCodes status = subscription.getStatus();
545                if (status != null) {
546                        switch (status) {
547                                case REQUESTED:
548                                        retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.REQUESTED);
549                                        break;
550                                case ACTIVE:
551                                        retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.ACTIVE);
552                                        break;
553                                case ERROR:
554                                        retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.ERROR);
555                                        break;
556                                case OFF:
557                                        retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.OFF);
558                                        break;
559                                case NULL:
560                                case ENTEREDINERROR:
561                                default:
562                                        ourLog.warn("Converting R5 Subscription status from {} to ERROR", status);
563                                        retVal.setStatus(org.hl7.fhir.r4.model.Subscription.SubscriptionStatus.ERROR);
564                        }
565                }
566                retVal.getTopicSubscription().setContent(subscription.getContent());
567                retVal.setEndpointUrl(subscription.getEndpoint());
568                retVal.getTopicSubscription().setTopic(subscription.getTopic());
569                retVal.setChannelType(getChannelType(subscription));
570
571                subscription.getFilterBy().forEach(filter -> retVal.getTopicSubscription()
572                                .addFilter(convertFilter(filter)));
573
574                retVal.getTopicSubscription().setHeartbeatPeriod(subscription.getHeartbeatPeriod());
575                retVal.getTopicSubscription().setMaxCount(subscription.getMaxCount());
576
577                setR5FlagsBasedOnChannelType(subscription, retVal);
578
579                retVal.setCrossPartitionEnabled(handleCrossPartition(theSubscription));
580
581                return retVal;
582        }
583
584        private void setR5FlagsBasedOnChannelType(
585                        org.hl7.fhir.r5.model.Subscription subscription, CanonicalSubscription retVal) {
586                if (retVal.getChannelType() == CanonicalSubscriptionChannelType.EMAIL) {
587                        String from;
588                        String subjectTemplate;
589                        try {
590                                from = getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_EMAIL_FROM);
591                                subjectTemplate = getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_SUBJECT_TEMPLATE);
592                        } catch (FHIRException theE) {
593                                throw new ConfigurationException(
594                                                Msg.code(2323) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
595                        }
596                        retVal.getEmailDetails().setFrom(from);
597                        retVal.getEmailDetails().setSubjectTemplate(subjectTemplate);
598                }
599
600                if (retVal.getChannelType() == CanonicalSubscriptionChannelType.RESTHOOK) {
601                        String stripVersionIds;
602                        String deliverLatestVersion;
603                        try {
604                                stripVersionIds =
605                                                getExtensionString(subscription, HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_STRIP_VERSION_IDS);
606                                deliverLatestVersion = getExtensionString(
607                                                subscription, HapiExtensions.EXT_SUBSCRIPTION_RESTHOOK_DELIVER_LATEST_VERSION);
608                        } catch (FHIRException theE) {
609                                throw new ConfigurationException(
610                                                Msg.code(2324) + "Failed to extract subscription extension(s): " + theE.getMessage(), theE);
611                        }
612                        retVal.getRestHookDetails().setStripVersionId(Boolean.parseBoolean(stripVersionIds));
613                        retVal.getRestHookDetails().setDeliverLatestVersion(Boolean.parseBoolean(deliverLatestVersion));
614                }
615        }
616
617        private CanonicalTopicSubscriptionFilter convertFilter(
618                        org.hl7.fhir.r5.model.Subscription.SubscriptionFilterByComponent theFilter) {
619                CanonicalTopicSubscriptionFilter retVal = new CanonicalTopicSubscriptionFilter();
620                retVal.setResourceType(theFilter.getResourceType());
621                retVal.setFilterParameter(theFilter.getFilterParameter());
622                retVal.setModifier(theFilter.getModifier());
623                retVal.setComparator(theFilter.getComparator());
624                retVal.setValue(theFilter.getValue());
625                return retVal;
626        }
627
628        private void setPartitionIdOnReturnValue(IBaseResource theSubscription, CanonicalSubscription retVal) {
629                RequestPartitionId requestPartitionId =
630                                (RequestPartitionId) theSubscription.getUserData(Constants.RESOURCE_PARTITION_ID);
631                if (requestPartitionId != null) {
632                        retVal.setPartitionId(requestPartitionId.getFirstPartitionIdOrNull());
633                }
634        }
635
636        private String getExtensionString(IBaseHasExtensions theBase, String theUrl) {
637                return theBase.getExtension().stream()
638                                .filter(t -> theUrl.equals(t.getUrl()))
639                                .filter(t -> t.getValue() instanceof IPrimitiveType)
640                                .map(t -> (IPrimitiveType<?>) t.getValue())
641                                .map(t -> t.getValueAsString())
642                                .findFirst()
643                                .orElse(null);
644        }
645
646        @SuppressWarnings("EnumSwitchStatementWhichMissesCases")
647        public CanonicalSubscriptionChannelType getChannelType(IBaseResource theSubscription) {
648                CanonicalSubscriptionChannelType retVal = null;
649
650                switch (myFhirContext.getVersion().getVersion()) {
651                        case DSTU2: {
652                                String channelTypeCode = ((ca.uhn.fhir.model.dstu2.resource.Subscription) theSubscription)
653                                                .getChannel()
654                                                .getType();
655                                retVal = CanonicalSubscriptionChannelType.fromCode(null, channelTypeCode);
656                                break;
657                        }
658                        case DSTU3: {
659                                org.hl7.fhir.dstu3.model.Subscription.SubscriptionChannelType type =
660                                                ((org.hl7.fhir.dstu3.model.Subscription) theSubscription)
661                                                                .getChannel()
662                                                                .getType();
663                                if (type != null) {
664                                        String channelTypeCode = type.toCode();
665                                        retVal = CanonicalSubscriptionChannelType.fromCode(null, channelTypeCode);
666                                }
667                                break;
668                        }
669                        case R4: {
670                                org.hl7.fhir.r4.model.Subscription.SubscriptionChannelType type = ((org.hl7.fhir.r4.model.Subscription)
671                                                                theSubscription)
672                                                .getChannel()
673                                                .getType();
674                                if (type != null) {
675                                        String channelTypeCode = type.toCode();
676                                        retVal = CanonicalSubscriptionChannelType.fromCode(null, channelTypeCode);
677                                }
678                                break;
679                        }
680                        case R4B: {
681                                org.hl7.fhir.r4b.model.Subscription.SubscriptionChannelType type =
682                                                ((org.hl7.fhir.r4b.model.Subscription) theSubscription)
683                                                                .getChannel()
684                                                                .getType();
685                                if (type != null) {
686                                        String channelTypeCode = type.toCode();
687                                        retVal = CanonicalSubscriptionChannelType.fromCode(null, channelTypeCode);
688                                }
689                                break;
690                        }
691                        case R5: {
692                                org.hl7.fhir.r5.model.Coding nextTypeCode =
693                                                ((org.hl7.fhir.r5.model.Subscription) theSubscription).getChannelType();
694                                CanonicalSubscriptionChannelType code =
695                                                CanonicalSubscriptionChannelType.fromCode(nextTypeCode.getSystem(), nextTypeCode.getCode());
696                                if (code != null) {
697                                        retVal = code;
698                                }
699                                break;
700                        }
701                        default:
702                                throw new IllegalStateException(Msg.code(2326) + "Unsupported Subscription FHIR version: "
703                                                + myFhirContext.getVersion().getVersion());
704                }
705
706                return retVal;
707        }
708
709        @SuppressWarnings("EnumSwitchStatementWhichMissesCases")
710        @Nullable
711        public String getCriteria(IBaseResource theSubscription) {
712                String retVal = null;
713
714                switch (myFhirContext.getVersion().getVersion()) {
715                        case DSTU2:
716                                retVal = ((Subscription) theSubscription).getCriteria();
717                                break;
718                        case DSTU3:
719                                retVal = ((org.hl7.fhir.dstu3.model.Subscription) theSubscription).getCriteria();
720                                break;
721                        case R4:
722                                retVal = ((org.hl7.fhir.r4.model.Subscription) theSubscription).getCriteria();
723                                break;
724                        case R4B:
725                                retVal = ((org.hl7.fhir.r4b.model.Subscription) theSubscription).getCriteria();
726                                break;
727                        case R5:
728                        default:
729                                throw new IllegalStateException(
730                                                Msg.code(2327) + "Subscription criteria is not supported for FHIR version: "
731                                                                + myFhirContext.getVersion().getVersion());
732                }
733
734                return retVal;
735        }
736
737        public void setMatchingStrategyTag(
738                        @Nonnull IBaseResource theSubscription, @Nullable SubscriptionMatchingStrategy theStrategy) {
739                IBaseMetaType meta = theSubscription.getMeta();
740
741                // Remove any existing strategy tag
742                meta.getTag().stream()
743                                .filter(t -> HapiExtensions.EXT_SUBSCRIPTION_MATCHING_STRATEGY.equals(t.getSystem()))
744                                .forEach(t -> {
745                                        t.setCode(null);
746                                        t.setSystem(null);
747                                        t.setDisplay(null);
748                                });
749
750                if (theStrategy == null) {
751                        return;
752                }
753
754                String value = theStrategy.toString();
755                String display;
756
757                if (theStrategy == SubscriptionMatchingStrategy.DATABASE) {
758                        display = "Database";
759                } else if (theStrategy == SubscriptionMatchingStrategy.IN_MEMORY) {
760                        display = "In-memory";
761                } else if (theStrategy == SubscriptionMatchingStrategy.TOPIC) {
762                        display = "SubscriptionTopic";
763                } else {
764                        throw new IllegalStateException(Msg.code(567) + "Unknown "
765                                        + SubscriptionMatchingStrategy.class.getSimpleName() + ": " + theStrategy);
766                }
767                meta.addTag()
768                                .setSystem(HapiExtensions.EXT_SUBSCRIPTION_MATCHING_STRATEGY)
769                                .setCode(value)
770                                .setDisplay(display);
771        }
772
773        public String getSubscriptionStatus(IBaseResource theSubscription) {
774                final IPrimitiveType<?> status = myFhirContext
775                                .newTerser()
776                                .getSingleValueOrNull(theSubscription, SubscriptionConstants.SUBSCRIPTION_STATUS, IPrimitiveType.class);
777                if (status == null) {
778                        return null;
779                }
780                return status.getValueAsString();
781        }
782
783        private boolean handleCrossPartition(IBaseResource theSubscription) {
784                RequestPartitionId requestPartitionId =
785                                (RequestPartitionId) theSubscription.getUserData(Constants.RESOURCE_PARTITION_ID);
786
787                boolean isSubscriptionCreatedOnDefaultPartition = false;
788
789                if (nonNull(requestPartitionId)) {
790                        isSubscriptionCreatedOnDefaultPartition = requestPartitionId.isDefaultPartition();
791                }
792
793                boolean isSubscriptionDefinededAsCrossPartitionSubscription =
794                                SubscriptionUtil.isDefinedAsCrossPartitionSubcription(theSubscription);
795                boolean isGlobalSettingCrossPartitionSubscriptionEnabled =
796                                mySubscriptionSettings.isCrossPartitionSubscriptionEnabled();
797
798                return isSubscriptionCreatedOnDefaultPartition
799                                && isSubscriptionDefinededAsCrossPartitionSubscription
800                                && isGlobalSettingCrossPartitionSubscriptionEnabled;
801        }
802}