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.i18n.Msg; 024import ca.uhn.fhir.rest.api.Constants; 025import ca.uhn.fhir.rest.api.SummaryEnum; 026import ca.uhn.fhir.rest.api.server.RequestDetails; 027import ca.uhn.fhir.rest.param.binder.CollectionBinder; 028import ca.uhn.fhir.rest.server.ElementsSupportEnum; 029import ca.uhn.fhir.rest.server.exceptions.InternalErrorException; 030import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException; 031import org.apache.commons.lang3.StringUtils; 032 033import java.lang.reflect.Method; 034import java.util.Collection; 035import java.util.HashSet; 036import java.util.Set; 037import java.util.StringTokenizer; 038 039import static org.apache.commons.lang3.StringUtils.isNotBlank; 040 041public class ElementsParameter implements IParameter { 042 043 @SuppressWarnings("rawtypes") 044 private Class<? extends Collection> myInnerCollectionType; 045 046 @Override 047 @SuppressWarnings({"rawtypes", "unchecked"}) 048 public Object translateQueryParametersIntoServerArgument( 049 RequestDetails theRequest, BaseMethodBinding theMethodBinding) 050 throws InternalErrorException, InvalidRequestException { 051 Set<String> value = getElementsValueOrNull(theRequest, false); 052 if (value == null || value.isEmpty()) { 053 return null; 054 } 055 056 if (myInnerCollectionType == null) { 057 return StringUtils.join(value, ','); 058 } 059 060 try { 061 Collection retVal = myInnerCollectionType.newInstance(); 062 retVal.addAll(value); 063 return retVal; 064 } catch (InstantiationException e) { 065 throw new InternalErrorException(Msg.code(413) + "Failed to instantiate " + myInnerCollectionType, e); 066 } catch (IllegalAccessException e) { 067 throw new InternalErrorException(Msg.code(414) + "Failed to instantiate " + myInnerCollectionType, e); 068 } 069 } 070 071 @Override 072 public void initializeTypes( 073 Method theMethod, 074 Class<? extends Collection<?>> theOuterCollectionType, 075 Class<? extends Collection<?>> theInnerCollectionType, 076 Class<?> theParameterType) { 077 if (theOuterCollectionType != null) { 078 throw new ConfigurationException(Msg.code(415) + "Method '" + theMethod.getName() + "' in type '" 079 + theMethod.getDeclaringClass().getCanonicalName() + "' is of type " + SummaryEnum.class 080 + " but can not be a collection of collections"); 081 } 082 if (theInnerCollectionType != null) { 083 myInnerCollectionType = CollectionBinder.getInstantiableCollectionType( 084 theInnerCollectionType, SummaryEnum.class.getSimpleName()); 085 } 086 } 087 088 public static Set<String> getElementsValueOrNull(RequestDetails theRequest, boolean theExclude) { 089 boolean standardMode = theRequest.getServer().getElementsSupport() != ElementsSupportEnum.EXTENDED; 090 if (theExclude && standardMode) { 091 return null; 092 } 093 094 String paramName = Constants.PARAM_ELEMENTS; 095 if (theExclude) { 096 paramName = Constants.PARAM_ELEMENTS + Constants.PARAM_ELEMENTS_EXCLUDE_MODIFIER; 097 } 098 String[] elementsValues = theRequest.getParameters().get(paramName); 099 100 if (elementsValues != null && elementsValues.length > 0) { 101 Set<String> retVal = new HashSet<>(); 102 for (String next : elementsValues) { 103 StringTokenizer tok = new StringTokenizer(next, ","); 104 while (tok.hasMoreTokens()) { 105 String token = tok.nextToken(); 106 if (isNotBlank(token)) { 107 if (token.contains(".")) 108 if (standardMode) { 109 continue; 110 } 111 retVal.add(token); 112 } 113 } 114 } 115 if (retVal.isEmpty()) { 116 return null; 117 } 118 119 return retVal; 120 } 121 return null; 122 } 123}