001/*
002 * #%L
003 * HAPI FHIR - Core Library
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.rest.api;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.model.api.IQueryParameterOr;
024import ca.uhn.fhir.model.api.IQueryParameterType;
025
026import java.util.ArrayList;
027import java.util.StringTokenizer;
028
029import static org.apache.commons.lang3.StringUtils.isBlank;
030
031public class QualifiedParamList extends ArrayList<String> {
032
033        private static final long serialVersionUID = 1L;
034
035        private String myQualifier;
036
037        public QualifiedParamList() {
038                super();
039        }
040
041        public QualifiedParamList(int theCapacity) {
042                super(theCapacity);
043        }
044
045        public QualifiedParamList(IQueryParameterOr<?> theNextOr, FhirContext theContext) {
046                for (IQueryParameterType next : theNextOr.getValuesAsQueryTokens()) {
047                        if (myQualifier == null) {
048                                myQualifier = next.getQueryParameterQualifier();
049                        }
050                        add(next.getValueAsQueryToken(theContext));
051                }
052        }
053
054        public String getQualifier() {
055                return myQualifier;
056        }
057
058        public void setQualifier(String theQualifier) {
059                myQualifier = theQualifier;
060        }
061
062        public static QualifiedParamList singleton(String theParamValue) {
063                return singleton(null, theParamValue);
064        }
065
066        public static QualifiedParamList singleton(String theQualifier, String theParamValue) {
067                QualifiedParamList retVal = new QualifiedParamList(1);
068                retVal.setQualifier(theQualifier);
069                retVal.add(theParamValue);
070                return retVal;
071        }
072
073        public static QualifiedParamList splitQueryStringByCommasIgnoreEscape(String theQualifier, String theParams) {
074                QualifiedParamList retVal = new QualifiedParamList();
075                retVal.setQualifier(theQualifier);
076
077                StringTokenizer tok = new StringTokenizer(theParams, ",", true);
078                String prev = null;
079                while (tok.hasMoreElements()) {
080                        String str = tok.nextToken();
081                        if (isBlank(str)) {
082                                prev = null;
083                                continue;
084                        }
085
086                        if (str.equals(",")) {
087                                if (countTrailingSlashes(prev) % 2 == 1) {
088                                        int idx = retVal.size() - 1;
089                                        String existing = retVal.get(idx);
090                                        prev = existing.substring(0, existing.length() - 1) + ',';
091                                        retVal.set(idx, prev);
092                                } else {
093                                        prev = null;
094                                }
095                                continue;
096                        }
097
098                        if (prev != null && prev.length() > 0 && prev.charAt(prev.length() - 1) == ',') {
099                                int idx = retVal.size() - 1;
100                                String existing = retVal.get(idx);
101                                prev = existing + str;
102                                retVal.set(idx, prev);
103                        } else {
104                                retVal.add(str);
105                                prev = str;
106                        }
107                }
108
109                // If no value was found, at least add that empty string as a value. It should get ignored later, but at
110                // least this lets us give a sensible error message if the parameter name was bad. See
111                // ResourceProviderR4Test#testParameterWithNoValueThrowsError_InvalidChainOnCustomSearch for an example
112                if (retVal.size() == 0) {
113                        retVal.add("");
114                }
115
116                return retVal;
117        }
118
119        private static int countTrailingSlashes(String theString) {
120                if (theString == null) {
121                        return 0;
122                }
123                int retVal = 0;
124                for (int i = theString.length() - 1; i >= 0; i--) {
125                        char nextChar = theString.charAt(i);
126                        if (nextChar != '\\') {
127                                break;
128                        }
129                        retVal++;
130                }
131                return retVal;
132        }
133}