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.util;
021
022import ca.uhn.fhir.interceptor.model.RequestPartitionId;
023import ca.uhn.fhir.util.UrlUtil;
024import org.apache.commons.lang3.Validate;
025import org.hibernate.engine.jdbc.internal.BasicFormatterImpl;
026import org.slf4j.Logger;
027import org.slf4j.LoggerFactory;
028
029import java.util.ArrayList;
030import java.util.Collections;
031import java.util.List;
032
033import static org.apache.commons.lang3.StringUtils.trim;
034
035public class SqlQuery {
036        private static final Logger ourLog = LoggerFactory.getLogger(SqlQuery.class);
037        private final String myThreadName = Thread.currentThread().getName();
038        private final String mySql;
039        private final List<String> myParams;
040        private final long myQueryTimestamp;
041        private final long myElapsedTime;
042        private final StackTraceElement[] myStackTrace;
043        private final int mySize;
044        private final LanguageEnum myLanguage;
045        private final String myNamespace;
046        private final RequestPartitionId myRequestPartitionId;
047
048        public SqlQuery(
049                        String theSql,
050                        List<String> theParams,
051                        long theQueryTimestamp,
052                        long theElapsedTime,
053                        StackTraceElement[] theStackTraceElements,
054                        int theSize,
055                        RequestPartitionId theRequestPartitionId) {
056                this(
057                                null,
058                                theSql,
059                                theParams,
060                                theQueryTimestamp,
061                                theElapsedTime,
062                                theStackTraceElements,
063                                theSize,
064                                LanguageEnum.SQL,
065                                theRequestPartitionId);
066        }
067
068        public SqlQuery(
069                        String theNamespace,
070                        String theSql,
071                        List<String> theParams,
072                        long theQueryTimestamp,
073                        long theElapsedTime,
074                        StackTraceElement[] theStackTraceElements,
075                        int theSize,
076                        LanguageEnum theLanguage,
077                        RequestPartitionId theRequestPartitionId) {
078                Validate.notNull(theLanguage, "theLanguage must not be null");
079
080                myNamespace = theNamespace;
081                mySql = theSql;
082                myParams = Collections.unmodifiableList(theParams);
083                myQueryTimestamp = theQueryTimestamp;
084                myElapsedTime = theElapsedTime;
085                myStackTrace = theStackTraceElements;
086                mySize = theSize;
087                myLanguage = theLanguage;
088                myRequestPartitionId = theRequestPartitionId;
089        }
090
091        public RequestPartitionId getRequestPartitionId() {
092                return myRequestPartitionId;
093        }
094
095        public String getNamespace() {
096                return myNamespace;
097        }
098
099        public long getQueryTimestamp() {
100                return myQueryTimestamp;
101        }
102
103        public long getElapsedTime() {
104                return myElapsedTime;
105        }
106
107        public String getThreadName() {
108                return myThreadName;
109        }
110
111        public String getSql(boolean theInlineParams, boolean theFormat) {
112                return getSql(theInlineParams, theFormat, false);
113        }
114
115        public LanguageEnum getLanguage() {
116                return myLanguage;
117        }
118
119        public String getSql(boolean theInlineParams, boolean theFormat, boolean theSanitizeParams) {
120                String retVal = mySql;
121                if (theFormat) {
122                        if (getLanguage() == LanguageEnum.SQL) {
123                                retVal = new BasicFormatterImpl().format(retVal);
124
125                                // BasicFormatterImpl annoyingly adds a newline at the very start of its output
126                                while (retVal.startsWith("\n")) {
127                                        retVal = retVal.substring(1);
128                                }
129                        }
130                }
131
132                if (theInlineParams) {
133                        List<String> nextParams = new ArrayList<>(myParams);
134                        int idx = 0;
135                        while (nextParams.size() > 0) {
136                                idx = retVal.indexOf("?", idx);
137                                if (idx == -1) {
138                                        break;
139                                }
140                                String nextParamValue = nextParams.remove(0);
141                                String nextSubstitution;
142                                if (nextParamValue != null) {
143                                        if (theSanitizeParams) {
144                                                nextParamValue = UrlUtil.sanitizeUrlPart(nextParamValue);
145                                        }
146                                        nextSubstitution = "'" + nextParamValue + "'";
147                                } else {
148                                        nextSubstitution = "NULL";
149                                }
150                                retVal = retVal.substring(0, idx) + nextSubstitution + retVal.substring(idx + 1);
151                                idx += nextSubstitution.length();
152                        }
153                }
154
155                return trim(retVal);
156        }
157
158        public StackTraceElement[] getStackTrace() {
159                return myStackTrace;
160        }
161
162        public int getSize() {
163                return mySize;
164        }
165
166        @Override
167        public String toString() {
168                return getSql(true, true);
169        }
170
171        public enum LanguageEnum {
172                SQL,
173                JSON
174        }
175}