001/*-
002 * #%L
003 * Smile CDR - CDR
004 * %%
005 * Copyright (C) 2016 - 2025 Smile CDR, Inc.
006 * %%
007 * All rights reserved.
008 * #L%
009 */
010package ca.cdr.api.util;
011
012import ca.cdr.api.model.json.TransactionLogStepJson;
013import ca.cdr.api.pub.hl7v2.model.MappingMessage;
014import ca.uhn.fhir.rest.api.server.RequestDetails;
015import ca.uhn.fhir.rest.server.servlet.ServletRequestDetails;
016import com.fasterxml.jackson.core.JsonProcessingException;
017import com.fasterxml.jackson.databind.ObjectMapper;
018import jakarta.servlet.http.HttpServletRequest;
019import org.apache.commons.lang3.Validate;
020
021import java.util.ArrayList;
022import java.util.Collections;
023import java.util.HashMap;
024import java.util.List;
025import java.util.Map;
026
027public class TransactionLogRequestDetailsUtil {
028        public static final String STEPS_LIST_KEY = TransactionLogRequestDetailsUtil.class.getName() + "_STEPS_LIST";
029        public static final String REQUEST_SUBTYPE_KEY =
030                        TransactionLogRequestDetailsUtil.class.getName() + "_REQUEST_SUBTYPE_KEY";
031        private static final String MESSAGES_LIST_KEY = TransactionLogRequestDetailsUtil.class.getName() + "_MESSAGES_LIST";
032        private static final String ADDITIONAL_JSON_MAP_KEY =
033                        TransactionLogRequestDetailsUtil.class.getName() + "_ADDITIONAL_JSON_MAP";
034
035        private static final ObjectMapper ourObjectMapper = new ObjectMapper();
036
037        private TransactionLogRequestDetailsUtil() {}
038
039        public static void addMessageToRequest(RequestDetails theDetails, MappingMessage theMessage) {
040                Validate.notNull(theDetails, "theDetails must not be null");
041                Validate.notNull(theMessage, "theMessage must not be null");
042                @SuppressWarnings("unchecked")
043                List<MappingMessage> messages =
044                                (List<MappingMessage>) theDetails.getUserData().get(MESSAGES_LIST_KEY);
045                if (messages == null) {
046                        messages = new ArrayList<>();
047                        theDetails.getUserData().put(MESSAGES_LIST_KEY, messages);
048                }
049                messages.add(theMessage);
050        }
051
052        public static List<MappingMessage> getMessagesOnRequest(RequestDetails theDetails) {
053                @SuppressWarnings("unchecked")
054                List<MappingMessage> messages =
055                                (List<MappingMessage>) theDetails.getUserData().get(MESSAGES_LIST_KEY);
056                if (messages == null) {
057                        return List.of();
058                }
059                return Collections.unmodifiableList(messages);
060        }
061
062        public static List<TransactionLogStepJson> getTransactionLogStepsFromRequest(RequestDetails theRequestDetails) {
063                @SuppressWarnings("unchecked")
064                List<TransactionLogStepJson> steps =
065                                (List<TransactionLogStepJson>) theRequestDetails.getAttribute(STEPS_LIST_KEY);
066                if (steps == null) {
067                        return List.of();
068                }
069                return Collections.unmodifiableList(steps);
070        }
071
072        /**
073         * Adds a transaction log step to the attribute which holds the list of steps for the provided request.
074         * @param theRequestDetails the request details
075         * @param theStep the transaction log step to add
076         */
077        public static void addTransactionLogStepForRequest(
078                        ServletRequestDetails theRequestDetails, TransactionLogStepJson theStep) {
079                Validate.notNull(theRequestDetails, "theRequestDetails must not be null");
080                addTransactionLogStepForRequest(theRequestDetails.getServletRequest(), theStep);
081        }
082
083        /**
084         * Adds a transaction log step to the attribute which holds the list of steps for the provided request.
085         * @param theRequest the request
086         * @param theStep the transaction log step to add
087         */
088        public static void addTransactionLogStepForRequest(HttpServletRequest theRequest, TransactionLogStepJson theStep) {
089                Validate.notNull(theRequest, "theRequest must not be null");
090                Validate.notNull(theStep, "theStep must not be null");
091                @SuppressWarnings("unchecked")
092                List<TransactionLogStepJson> steps = (List<TransactionLogStepJson>) theRequest.getAttribute(STEPS_LIST_KEY);
093                if (steps == null) {
094                        steps = new ArrayList<>();
095                        theRequest.setAttribute(STEPS_LIST_KEY, steps);
096                }
097                steps.add(theStep);
098        }
099
100        /**
101         * Utility method to add key/value pairs to a transactionLog.  The provided value needs to be serializable
102         * with Jackson since pairs are accumulated and subsequently formatted in an inline json string.
103         * IllegalArgumentException is thrown if <code>theValue</code> is not Json serializable with Jackson or <code>null</code>.
104         *      The additional log properties will be tallied and rendered as inline json in the transactionLogEventJson:
105         *{
106         *              "id": 2,
107         *      "initialTimestamp": "2023-08-29T14:00:30.096-04:00",
108         *              "type": "FHIR_REQUEST",
109         *              "subType": "FHIR_CREATE",
110         *              "outcome": "SUCCESS",
111         *              ...
112         *              "requestId": "r1bW96uhRAn2z8iU",
113         *              "additionalJson": { "diseases": [ "Tuberculosis", "Polio", "Malaria", "Dengue fever"], "clinic": "Acme Ouest" },
114         *              ...
115         *              "userFamilyName": "GenericUser",
116         *              "userGivenName": "Admin"
117         *              }
118         *
119         * @param theRequestDetails The request details where the pairs will be stored
120         * @param theKey The key for referencing to theObject.  Keys are transformed into json properties when serializing a pair.
121         * @param theValue The value needing storage.  Values are transformed into json values when serializing a pair.
122         *
123         */
124        public static void addAdditionalJsonProperty(RequestDetails theRequestDetails, String theKey, Object theValue) {
125                Validate.notNull(theRequestDetails, "theDetails must not be null");
126                Validate.notNull(theKey, "theKey must not be null");
127                Validate.notNull(theValue, "theValue must not be null");
128
129                try {
130                        // ensure that the value can be serialized to JSON
131                        ourObjectMapper.writeValueAsString(theValue);
132                } catch (JsonProcessingException e) {
133                        throw new IllegalArgumentException(
134                                        String.format(
135                                                        "Unable to add pair with key %s to the additional Json property as the provided value object could not be parsed.",
136                                                        theKey),
137                                        e);
138                }
139                @SuppressWarnings("unchecked")
140                Map<String, Object> additionalJsonPropertiesMap =
141                                (Map<String, Object>) theRequestDetails.getAttribute(ADDITIONAL_JSON_MAP_KEY);
142                if (additionalJsonPropertiesMap == null) {
143                        additionalJsonPropertiesMap = new HashMap<>();
144                        theRequestDetails.setAttribute(ADDITIONAL_JSON_MAP_KEY, additionalJsonPropertiesMap);
145                }
146                additionalJsonPropertiesMap.put(theKey, theValue);
147        }
148
149        public static Map<String, Object> getAdditionalJsonPropertiesMap(RequestDetails theRequestDetails) {
150                Validate.notNull(theRequestDetails, "theRequestDetails must not be null");
151                @SuppressWarnings("unchecked")
152                Map<String, Object> additionalJsonPropertiesMap =
153                                (Map<String, Object>) theRequestDetails.getAttribute(ADDITIONAL_JSON_MAP_KEY);
154                if (additionalJsonPropertiesMap == null) {
155                        return Map.of();
156                }
157                return additionalJsonPropertiesMap;
158        }
159}