001/*
002 * #%L
003 * HAPI FHIR - Server Framework
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.server.method;
021
022import ca.uhn.fhir.context.ConfigurationException;
023import ca.uhn.fhir.context.FhirContext;
024import ca.uhn.fhir.context.FhirVersionEnum;
025import ca.uhn.fhir.i18n.Msg;
026import ca.uhn.fhir.rest.annotation.Sort;
027import ca.uhn.fhir.rest.api.Constants;
028import ca.uhn.fhir.rest.api.SortOrderEnum;
029import ca.uhn.fhir.rest.api.SortSpec;
030import ca.uhn.fhir.rest.api.server.RequestDetails;
031import ca.uhn.fhir.rest.param.ParameterUtil;
032import ca.uhn.fhir.rest.server.exceptions.InternalErrorException;
033import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
034
035import java.lang.reflect.Method;
036import java.util.Collection;
037import java.util.StringTokenizer;
038
039import static org.apache.commons.lang3.StringUtils.isNotBlank;
040
041public class SortParameter implements IParameter {
042
043        private FhirContext myContext;
044
045        public SortParameter(FhirContext theContext) {
046                myContext = theContext;
047        }
048
049        @Override
050        public void initializeTypes(
051                        Method theMethod,
052                        Class<? extends Collection<?>> theOuterCollectionType,
053                        Class<? extends Collection<?>> theInnerCollectionType,
054                        Class<?> theParameterType) {
055                if (theOuterCollectionType != null || theInnerCollectionType != null) {
056                        throw new ConfigurationException(Msg.code(443) + "Method '" + theMethod.getName() + "' in type '"
057                                        + theMethod.getDeclaringClass().getCanonicalName() + "' is annotated with @" + Sort.class.getName()
058                                        + " but can not be of collection type");
059                }
060                if (!theParameterType.equals(SortSpec.class)) {
061                        throw new ConfigurationException(Msg.code(444) + "Method '" + theMethod.getName() + "' in type '"
062                                        + theMethod.getDeclaringClass().getCanonicalName() + "' is annotated with @" + Sort.class.getName()
063                                        + " but is an invalid type, must be: " + SortSpec.class.getCanonicalName());
064                }
065        }
066
067        @Override
068        public Object translateQueryParametersIntoServerArgument(
069                        RequestDetails theRequest, BaseMethodBinding theMethodBinding)
070                        throws InternalErrorException, InvalidRequestException {
071                if (!theRequest.getParameters().containsKey(Constants.PARAM_SORT)) {
072                        if (!theRequest.getParameters().containsKey(Constants.PARAM_SORT_ASC)) {
073                                if (!theRequest.getParameters().containsKey(Constants.PARAM_SORT_DESC)) {
074                                        return null;
075                                }
076                        }
077                }
078
079                SortSpec outerSpec = null;
080                SortSpec innerSpec = null;
081                for (String nextParamName : theRequest.getParameters().keySet()) {
082                        SortOrderEnum order;
083                        if (Constants.PARAM_SORT.equals(nextParamName)) {
084                                order = null;
085                        } else if (Constants.PARAM_SORT_ASC.equals(nextParamName)) {
086                                order = SortOrderEnum.ASC;
087                        } else if (Constants.PARAM_SORT_DESC.equals(nextParamName)) {
088                                order = SortOrderEnum.DESC;
089                        } else {
090                                continue;
091                        }
092
093                        String[] values = theRequest.getParameters().get(nextParamName);
094                        if (values != null) {
095
096                                for (String nextValue : values) {
097
098                                        if (myContext.getVersion().getVersion().isNewerThan(FhirVersionEnum.DSTU2) && order == null) {
099                                                StringTokenizer tok = new StringTokenizer(nextValue, ",");
100                                                while (tok.hasMoreTokens()) {
101                                                        String next = tok.nextToken();
102                                                        if (isNotBlank(next) && !next.equals("-")) {
103                                                                order = SortOrderEnum.ASC;
104                                                                if (next.startsWith("-")) {
105                                                                        order = SortOrderEnum.DESC;
106                                                                        next = next.substring(1);
107                                                                }
108
109                                                                SortSpec spec = new SortSpec();
110                                                                spec.setOrder(order);
111                                                                spec.setParamName(next);
112                                                                if (innerSpec == null) {
113                                                                        outerSpec = spec;
114                                                                        innerSpec = spec;
115                                                                } else {
116                                                                        innerSpec.setChain(spec);
117                                                                        innerSpec = spec;
118                                                                }
119                                                        }
120                                                }
121
122                                        } else {
123
124                                                if (isNotBlank(nextValue)) {
125                                                        SortSpec spec = new SortSpec();
126                                                        spec.setOrder(order);
127                                                        spec.setParamName(nextValue);
128                                                        if (innerSpec == null) {
129                                                                outerSpec = spec;
130                                                                innerSpec = spec;
131                                                        } else {
132                                                                innerSpec.setChain(spec);
133                                                                innerSpec = spec;
134                                                        }
135                                                }
136                                        }
137                                }
138                        }
139                }
140
141                return outerSpec;
142        }
143
144        public static String createSortStringDstu3(SortSpec ss) {
145                StringBuilder val = new StringBuilder();
146                while (ss != null) {
147
148                        if (isNotBlank(ss.getParamName())) {
149                                if (val.length() > 0) {
150                                        val.append(',');
151                                }
152                                if (ss.getOrder() == SortOrderEnum.DESC) {
153                                        val.append('-');
154                                }
155                                val.append(ParameterUtil.escape(ss.getParamName()));
156                        }
157
158                        ss = ss.getChain();
159                }
160
161                String string = val.toString();
162                return string;
163        }
164}