001/*-
002 * #%L
003 * HAPI FHIR - Master Data Management
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.mdm.provider;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.mdm.api.IMdmControllerSvc;
025import ca.uhn.fhir.mdm.api.MdmMatchResultEnum;
026import ca.uhn.fhir.mdm.api.paging.MdmPageLinkBuilder;
027import ca.uhn.fhir.mdm.api.paging.MdmPageLinkTuple;
028import ca.uhn.fhir.mdm.api.paging.MdmPageRequest;
029import ca.uhn.fhir.mdm.api.params.MdmHistorySearchParameters;
030import ca.uhn.fhir.mdm.model.MdmTransactionContext;
031import ca.uhn.fhir.mdm.model.mdmevents.MdmLinkJson;
032import ca.uhn.fhir.mdm.model.mdmevents.MdmLinkWithRevisionJson;
033import ca.uhn.fhir.rest.api.server.RequestDetails;
034import ca.uhn.fhir.rest.server.TransactionLogMessages;
035import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
036import ca.uhn.fhir.rest.server.provider.ProviderConstants;
037import ca.uhn.fhir.rest.server.servlet.ServletRequestDetails;
038import ca.uhn.fhir.util.ParametersUtil;
039import jakarta.annotation.Nonnull;
040import jakarta.annotation.Nullable;
041import org.hl7.fhir.instance.model.api.IBase;
042import org.hl7.fhir.instance.model.api.IBaseParameters;
043import org.hl7.fhir.instance.model.api.IPrimitiveType;
044import org.springframework.data.domain.Page;
045
046import java.util.Arrays;
047import java.util.Collection;
048import java.util.Collections;
049import java.util.Comparator;
050import java.util.List;
051import java.util.Map;
052import java.util.Objects;
053import java.util.Optional;
054import java.util.stream.Collectors;
055
056public abstract class BaseMdmProvider {
057
058        protected final FhirContext myFhirContext;
059        protected final IMdmControllerSvc myMdmControllerSvc;
060
061        public BaseMdmProvider(FhirContext theFhirContext, IMdmControllerSvc theMdmControllerSvc) {
062                myFhirContext = theFhirContext;
063                myMdmControllerSvc = theMdmControllerSvc;
064        }
065
066        protected void validateMergeParameters(
067                        IPrimitiveType<String> theFromGoldenResourceId, IPrimitiveType<String> theToGoldenResourceId) {
068                validateNotNull(ProviderConstants.MDM_MERGE_GR_FROM_GOLDEN_RESOURCE_ID, theFromGoldenResourceId);
069                validateNotNull(ProviderConstants.MDM_MERGE_GR_TO_GOLDEN_RESOURCE_ID, theToGoldenResourceId);
070                if (theFromGoldenResourceId.getValue().equals(theToGoldenResourceId.getValue())) {
071                        throw new InvalidRequestException(
072                                        Msg.code(1493) + "fromGoldenResourceId must be different from toGoldenResourceId");
073                }
074        }
075
076        private void validateNotNull(String theName, IPrimitiveType<String> theString) {
077                if (theString == null || theString.getValue() == null) {
078                        throw new InvalidRequestException(Msg.code(1494) + theName + " cannot be null");
079                }
080        }
081
082        protected void validateMdmLinkHistoryParameters(
083                        List<IPrimitiveType<String>> theGoldenResourceIds, List<IPrimitiveType<String>> theSourceIds) {
084                validateBothCannotBeNullOrEmpty(
085                                ProviderConstants.MDM_QUERY_LINKS_GOLDEN_RESOURCE_ID,
086                                theGoldenResourceIds,
087                                ProviderConstants.MDM_QUERY_LINKS_RESOURCE_ID,
088                                theSourceIds);
089        }
090
091        private void validateBothCannotBeNullOrEmpty(
092                        String theFirstName,
093                        List<IPrimitiveType<String>> theFirstList,
094                        String theSecondName,
095                        List<IPrimitiveType<String>> theSecondList) {
096                if ((theFirstList == null || theFirstList.isEmpty()) && (theSecondList == null || theSecondList.isEmpty())) {
097                        throw new InvalidRequestException(Msg.code(2292) + "Please include either [" + theFirstName + "]s, ["
098                                        + theSecondName + "]s, or both in your search inputs.");
099                }
100        }
101
102        protected void validateUpdateLinkParameters(
103                        IPrimitiveType<String> theGoldenResourceId,
104                        IPrimitiveType<String> theResourceId,
105                        IPrimitiveType<String> theMatchResult) {
106                validateNotNull(ProviderConstants.MDM_UPDATE_LINK_GOLDEN_RESOURCE_ID, theGoldenResourceId);
107                validateNotNull(ProviderConstants.MDM_UPDATE_LINK_RESOURCE_ID, theResourceId);
108                validateNotNull(ProviderConstants.MDM_UPDATE_LINK_MATCH_RESULT, theMatchResult);
109                MdmMatchResultEnum matchResult = MdmMatchResultEnum.valueOf(theMatchResult.getValue());
110                switch (matchResult) {
111                        case NO_MATCH:
112                        case MATCH:
113                                break;
114                        default:
115                                throw new InvalidRequestException(Msg.code(1495) + ProviderConstants.MDM_UPDATE_LINK + " illegal "
116                                                + ProviderConstants.MDM_UPDATE_LINK_MATCH_RESULT + " value '" + matchResult + "'.  Must be "
117                                                + MdmMatchResultEnum.NO_MATCH + " or " + MdmMatchResultEnum.MATCH);
118                }
119        }
120
121        protected void validateNotDuplicateParameters(
122                        IPrimitiveType<String> theGoldenResourceId, IPrimitiveType<String> theResourceId) {
123                validateNotNull(ProviderConstants.MDM_UPDATE_LINK_GOLDEN_RESOURCE_ID, theGoldenResourceId);
124                validateNotNull(ProviderConstants.MDM_UPDATE_LINK_RESOURCE_ID, theResourceId);
125        }
126
127        protected void validateCreateLinkParameters(
128                        IPrimitiveType<String> theGoldenResourceId,
129                        IPrimitiveType<String> theResourceId,
130                        @Nullable IPrimitiveType<String> theMatchResult) {
131                validateNotNull(ProviderConstants.MDM_CREATE_LINK_GOLDEN_RESOURCE_ID, theGoldenResourceId);
132                validateNotNull(ProviderConstants.MDM_CREATE_LINK_RESOURCE_ID, theResourceId);
133                if (theMatchResult != null) {
134                        MdmMatchResultEnum matchResult = MdmMatchResultEnum.valueOf(theMatchResult.getValue());
135                        switch (matchResult) {
136                                case NO_MATCH:
137                                case POSSIBLE_MATCH:
138                                case MATCH:
139                                        break;
140                                default:
141                                        throw new InvalidRequestException(Msg.code(1496) + ProviderConstants.MDM_CREATE_LINK + " illegal "
142                                                        + ProviderConstants.MDM_CREATE_LINK_MATCH_RESULT + " value '" + matchResult + "'.  Must be "
143                                                        + MdmMatchResultEnum.NO_MATCH + ", " + MdmMatchResultEnum.MATCH + " or "
144                                                        + MdmMatchResultEnum.POSSIBLE_MATCH);
145                        }
146                }
147        }
148
149        protected MdmTransactionContext createMdmContext(
150                        RequestDetails theRequestDetails,
151                        MdmTransactionContext.OperationType theOperationType,
152                        String theResourceType) {
153                TransactionLogMessages transactionLogMessages =
154                                TransactionLogMessages.createFromTransactionGuid(theRequestDetails.getTransactionGuid());
155                MdmTransactionContext mdmTransactionContext =
156                                new MdmTransactionContext(transactionLogMessages, theOperationType);
157                mdmTransactionContext.setResourceType(theResourceType);
158                return mdmTransactionContext;
159        }
160
161        @Nonnull
162        protected List<String> convertToStringsIncludingCommaDelimitedIfNotNull(
163                        List<IPrimitiveType<String>> thePrimitiveTypeStrings) {
164                if (thePrimitiveTypeStrings == null) {
165                        return Collections.emptyList();
166                }
167
168                return thePrimitiveTypeStrings.stream()
169                                .map(this::extractStringOrNull)
170                                .filter(Objects::nonNull)
171                                .map(input -> Arrays.asList(input.split(",")))
172                                .flatMap(Collection::stream)
173                                .collect(Collectors.toUnmodifiableList());
174        }
175
176        protected String extractStringOrNull(IPrimitiveType<String> theString) {
177                if (theString == null) {
178                        return null;
179                }
180                return theString.getValue();
181        }
182
183        protected IBaseParameters parametersFromMdmLinks(
184                        Page<MdmLinkJson> theMdmLinkStream,
185                        boolean theIncludeResultAndSource,
186                        ServletRequestDetails theServletRequestDetails,
187                        MdmPageRequest thePageRequest) {
188                IBaseParameters retval = ParametersUtil.newInstance(myFhirContext);
189                addPagingParameters(retval, theMdmLinkStream, theServletRequestDetails, thePageRequest);
190
191                long numDuplicates = theMdmLinkStream.getTotalElements();
192                ParametersUtil.addParameterToParametersLong(myFhirContext, retval, "total", numDuplicates);
193
194                theMdmLinkStream.getContent().forEach(mdmLink -> {
195                        IBase resultPart = ParametersUtil.addParameterToParameters(myFhirContext, retval, "link");
196                        ParametersUtil.addPartString(myFhirContext, resultPart, "goldenResourceId", mdmLink.getGoldenResourceId());
197                        ParametersUtil.addPartString(myFhirContext, resultPart, "sourceResourceId", mdmLink.getSourceId());
198
199                        if (theIncludeResultAndSource) {
200                                ParametersUtil.addPartString(
201                                                myFhirContext,
202                                                resultPart,
203                                                "matchResult",
204                                                mdmLink.getMatchResult().name());
205                                ParametersUtil.addPartString(
206                                                myFhirContext,
207                                                resultPart,
208                                                "linkSource",
209                                                mdmLink.getLinkSource().name());
210                                ParametersUtil.addPartBoolean(myFhirContext, resultPart, "eidMatch", mdmLink.getEidMatch());
211                                ParametersUtil.addPartBoolean(
212                                                myFhirContext, resultPart, "hadToCreateNewResource", mdmLink.getLinkCreatedNewResource());
213                                ParametersUtil.addPartDecimal(myFhirContext, resultPart, "score", mdmLink.getScore());
214                                ParametersUtil.addPartDecimal(myFhirContext, resultPart, "linkCreated", (double)
215                                                mdmLink.getCreated().getTime());
216                                ParametersUtil.addPartDecimal(myFhirContext, resultPart, "linkUpdated", (double)
217                                                mdmLink.getUpdated().getTime());
218                        }
219                });
220                return retval;
221        }
222
223        protected void parametersFromMdmLinkRevisions(
224                        IBaseParameters theRetVal,
225                        List<MdmLinkWithRevisionJson> theMdmLinkRevisions,
226                        ServletRequestDetails theRequestDetails) {
227                if (theMdmLinkRevisions.isEmpty()) {
228                        final IBase resultPart = ParametersUtil.addParameterToParameters(
229                                        myFhirContext, theRetVal, "historical links not found for query parameters");
230
231                        ParametersUtil.addPartString(
232                                        myFhirContext, resultPart, "theResults", "historical links not found for query parameters");
233                }
234
235                theMdmLinkRevisions.forEach(mdmLinkRevision -> parametersFromMdmLinkRevision(
236                                theRetVal,
237                                mdmLinkRevision,
238                                findInitialMatchResult(theMdmLinkRevisions, mdmLinkRevision, theRequestDetails)));
239        }
240
241        private MdmMatchResultEnum findInitialMatchResult(
242                        List<MdmLinkWithRevisionJson> theRevisionList,
243                        MdmLinkWithRevisionJson theToMatch,
244                        ServletRequestDetails theRequestDetails) {
245                String sourceId = theToMatch.getMdmLink().getSourceId();
246                String goldenId = theToMatch.getMdmLink().getGoldenResourceId();
247
248                // In the REDIRECT case, both the goldenResourceId and sourceResourceId fields are actually both
249                // golden resources. Because of this, based on our history query, it's possible not all links
250                // involving that golden resource will show up in the results (eg. query for goldenResourceId = GR/1
251                // but sourceResourceId = GR/1 in the link history). Hence, we should re-query to find the initial
252                // match result.
253                if (theToMatch.getMdmLink().getMatchResult() == MdmMatchResultEnum.REDIRECT) {
254                        MdmHistorySearchParameters params = new MdmHistorySearchParameters()
255                                        .setSourceIds(List.of(sourceId, goldenId))
256                                        .setGoldenResourceIds(List.of(sourceId, goldenId));
257
258                        List<MdmLinkWithRevisionJson> result = myMdmControllerSvc.queryLinkHistory(params, theRequestDetails);
259                        // If there is a POSSIBLE_DUPLICATE, a user merged two resources with *pre-existing* POSSIBLE_DUPLICATE link
260                        // so the initial match result is POSSIBLE_DUPLICATE
261                        // If no POSSIBLE_DUPLICATE, a user merged two *unlinked* GRs, so the initial match result is REDIRECT
262                        return containsPossibleDuplicate(result)
263                                        ? MdmMatchResultEnum.POSSIBLE_DUPLICATE
264                                        : MdmMatchResultEnum.REDIRECT;
265                }
266
267                // Get first match result with given source and golden ID
268                Optional<MdmLinkWithRevisionJson> theEarliestRevision = theRevisionList.stream()
269                                .filter(revision -> revision.getMdmLink().getSourceId().equals(sourceId))
270                                .filter(revision -> revision.getMdmLink().getGoldenResourceId().equals(goldenId))
271                                .min(Comparator.comparing(MdmLinkWithRevisionJson::getRevisionNumber));
272
273                return theEarliestRevision.isPresent()
274                                ? theEarliestRevision.get().getMdmLink().getMatchResult()
275                                : theToMatch.getMdmLink().getMatchResult();
276        }
277
278        private static boolean containsPossibleDuplicate(List<MdmLinkWithRevisionJson> result) {
279                return result.stream().anyMatch(t -> t.getMdmLink().getMatchResult() == MdmMatchResultEnum.POSSIBLE_DUPLICATE);
280        }
281
282        private void parametersFromMdmLinkRevision(
283                        IBaseParameters theRetVal,
284                        MdmLinkWithRevisionJson theMdmLinkRevision,
285                        MdmMatchResultEnum theInitialAutoResult) {
286                final IBase resultPart = ParametersUtil.addParameterToParameters(myFhirContext, theRetVal, "historical link");
287                final MdmLinkJson mdmLink = theMdmLinkRevision.getMdmLink();
288
289                ParametersUtil.addPartString(myFhirContext, resultPart, "goldenResourceId", mdmLink.getGoldenResourceId());
290                ParametersUtil.addPartString(
291                                myFhirContext,
292                                resultPart,
293                                "revisionTimestamp",
294                                theMdmLinkRevision.getRevisionTimestamp().toString());
295                ParametersUtil.addPartString(myFhirContext, resultPart, "sourceResourceId", mdmLink.getSourceId());
296                ParametersUtil.addPartString(
297                                myFhirContext,
298                                resultPart,
299                                "matchResult",
300                                mdmLink.getMatchResult().name());
301                ParametersUtil.addPartString(
302                                myFhirContext, resultPart, "linkSource", mdmLink.getLinkSource().name());
303                ParametersUtil.addPartBoolean(myFhirContext, resultPart, "eidMatch", mdmLink.getEidMatch());
304                ParametersUtil.addPartBoolean(
305                                myFhirContext, resultPart, "hadToCreateNewResource", mdmLink.getLinkCreatedNewResource());
306                ParametersUtil.addPartDecimal(myFhirContext, resultPart, "score", mdmLink.getScore());
307                ParametersUtil.addPartDecimal(myFhirContext, resultPart, "linkCreated", (double)
308                                mdmLink.getCreated().getTime());
309                ParametersUtil.addPartDecimal(myFhirContext, resultPart, "linkUpdated", (double)
310                                mdmLink.getUpdated().getTime());
311
312                IBase matchResultMapSubpart = ParametersUtil.createPart(myFhirContext, resultPart, "matchResultMap");
313
314                IBase matchedRulesSubpart = ParametersUtil.createPart(myFhirContext, matchResultMapSubpart, "matchedRules");
315                for (Map.Entry<String, MdmMatchResultEnum> entry : mdmLink.getRule()) {
316                        ParametersUtil.addPartString(myFhirContext, matchedRulesSubpart, "rule", entry.getKey());
317                }
318
319                ParametersUtil.addPartString(
320                                myFhirContext, matchResultMapSubpart, "initialMatchResult", theInitialAutoResult.name());
321        }
322
323        protected void addPagingParameters(
324                        IBaseParameters theParameters,
325                        Page<MdmLinkJson> theCurrentPage,
326                        ServletRequestDetails theServletRequestDetails,
327                        MdmPageRequest thePageRequest) {
328                MdmPageLinkTuple mdmPageLinkTuple =
329                                MdmPageLinkBuilder.buildMdmPageLinks(theServletRequestDetails, theCurrentPage, thePageRequest);
330
331                if (mdmPageLinkTuple.getPreviousLink().isPresent()) {
332                        ParametersUtil.addParameterToParametersUri(
333                                        myFhirContext,
334                                        theParameters,
335                                        "prev",
336                                        mdmPageLinkTuple.getPreviousLink().get());
337                }
338
339                ParametersUtil.addParameterToParametersUri(
340                                myFhirContext, theParameters, "self", mdmPageLinkTuple.getSelfLink());
341
342                if (mdmPageLinkTuple.getNextLink().isPresent()) {
343                        ParametersUtil.addParameterToParametersUri(
344                                        myFhirContext,
345                                        theParameters,
346                                        "next",
347                                        mdmPageLinkTuple.getNextLink().get());
348                }
349        }
350}