001/* 002 * #%L 003 * HAPI FHIR - Client 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.client.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.param.ParameterUtil; 031import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 032import org.hl7.fhir.instance.model.api.IBaseResource; 033 034import java.lang.reflect.Method; 035import java.util.ArrayList; 036import java.util.Collection; 037import java.util.List; 038import java.util.Map; 039 040import static org.apache.commons.lang3.StringUtils.isNotBlank; 041 042public class SortParameter implements IParameter { 043 044 private FhirContext myContext; 045 046 public SortParameter(FhirContext theContext) { 047 myContext = theContext; 048 } 049 050 @Override 051 public void initializeTypes( 052 Method theMethod, 053 Class<? extends Collection<?>> theOuterCollectionType, 054 Class<? extends Collection<?>> theInnerCollectionType, 055 Class<?> theParameterType) { 056 if (theOuterCollectionType != null || theInnerCollectionType != null) { 057 throw new ConfigurationException(Msg.code(1463) + "Method '" + theMethod.getName() + "' in type '" 058 + theMethod.getDeclaringClass().getCanonicalName() + "' is annotated with @" + Sort.class.getName() 059 + " but can not be of collection type"); 060 } 061 if (!theParameterType.equals(SortSpec.class)) { 062 throw new ConfigurationException(Msg.code(1464) + "Method '" + theMethod.getName() + "' in type '" 063 + theMethod.getDeclaringClass().getCanonicalName() + "' is annotated with @" + Sort.class.getName() 064 + " but is an invalid type, must be: " + SortSpec.class.getCanonicalName()); 065 } 066 } 067 068 @Override 069 public void translateClientArgumentIntoQueryArgument( 070 FhirContext theContext, 071 Object theSourceClientArgument, 072 Map<String, List<String>> theTargetQueryArguments, 073 IBaseResource theTargetResource) 074 throws InternalErrorException { 075 SortSpec ss = (SortSpec) theSourceClientArgument; 076 077 if (myContext.getVersion().getVersion().isNewerThan(FhirVersionEnum.DSTU2)) { 078 String string = createSortStringDstu3(ss); 079 if (string.length() > 0) { 080 if (!theTargetQueryArguments.containsKey(Constants.PARAM_SORT)) { 081 theTargetQueryArguments.put(Constants.PARAM_SORT, new ArrayList<String>()); 082 } 083 theTargetQueryArguments.get(Constants.PARAM_SORT).add(string); 084 } 085 086 } else { 087 088 while (ss != null) { 089 String name; 090 if (ss.getOrder() == null) { 091 name = Constants.PARAM_SORT; 092 } else if (ss.getOrder() == SortOrderEnum.ASC) { 093 name = Constants.PARAM_SORT_ASC; 094 } else { 095 name = Constants.PARAM_SORT_DESC; 096 } 097 098 if (ss.getParamName() != null) { 099 if (!theTargetQueryArguments.containsKey(name)) { 100 theTargetQueryArguments.put(name, new ArrayList<String>()); 101 } 102 103 theTargetQueryArguments.get(name).add(ss.getParamName()); 104 } 105 ss = ss.getChain(); 106 } 107 } 108 } 109 110 public static String createSortStringDstu3(SortSpec ss) { 111 StringBuilder val = new StringBuilder(); 112 while (ss != null) { 113 114 if (isNotBlank(ss.getParamName())) { 115 if (val.length() > 0) { 116 val.append(','); 117 } 118 if (ss.getOrder() == SortOrderEnum.DESC) { 119 val.append('-'); 120 } 121 val.append(ParameterUtil.escape(ss.getParamName())); 122 } 123 124 ss = ss.getChain(); 125 } 126 127 String string = val.toString(); 128 return string; 129 } 130}