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.exceptions.InternalErrorException;
029import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
030
031import java.lang.reflect.Method;
032import java.util.Collection;
033import java.util.Collections;
034import java.util.HashSet;
035import java.util.Set;
036
037import static org.apache.commons.lang3.StringUtils.isBlank;
038
039public class SummaryEnumParameter implements IParameter {
040
041        @SuppressWarnings("rawtypes")
042        private Class<? extends Collection> myInnerCollectionType;
043
044        @Override
045        @SuppressWarnings({"rawtypes", "unchecked"})
046        public Object translateQueryParametersIntoServerArgument(
047                        RequestDetails theRequest, BaseMethodBinding theMethodBinding)
048                        throws InternalErrorException, InvalidRequestException {
049                Set<SummaryEnum> value = getSummaryValueOrNull(theRequest);
050                if (value == null || value.isEmpty()) {
051                        return null;
052                }
053
054                if (myInnerCollectionType == null) {
055                        return value.iterator().next();
056                }
057
058                try {
059                        Collection retVal = myInnerCollectionType.newInstance();
060                        retVal.addAll(value);
061                        return retVal;
062                } catch (InstantiationException e) {
063                        throw new InternalErrorException(Msg.code(378) + "Failed to instantiate " + myInnerCollectionType, e);
064                } catch (IllegalAccessException e) {
065                        throw new InternalErrorException(Msg.code(379) + "Failed to instantiate " + myInnerCollectionType, e);
066                }
067        }
068
069        public static Set<SummaryEnum> getSummaryValueOrNull(RequestDetails theRequest) {
070                String[] summary = theRequest.getParameters().get(Constants.PARAM_SUMMARY);
071
072                Set<SummaryEnum> retVal;
073                if (summary == null || summary.length == 0) {
074                        retVal = null;
075                } else if (isBlank(summary[0])) {
076                        retVal = null;
077                } else if (summary.length == 1 && summary[0].indexOf(',') == -1) {
078                        retVal = toCollectionOrNull(SummaryEnum.fromCode(summary[0]));
079                        if (retVal == null) {
080                                retVal = toCollectionOrNull(SummaryEnum.fromCode(summary[0].toLowerCase()));
081                        }
082                } else {
083                        retVal = new HashSet<>();
084                        for (String nextParamValue : summary) {
085                                for (String nextParamValueTok : nextParamValue.split(",")) {
086                                        SummaryEnum value = SummaryEnum.fromCode(nextParamValueTok);
087                                        if (value == null) {
088                                                value = SummaryEnum.fromCode(nextParamValueTok.toLowerCase());
089                                        }
090                                        if (value != null) {
091                                                retVal.add(value);
092                                        }
093                                }
094                        }
095                }
096
097                if (retVal != null) {
098                        if (retVal.contains(SummaryEnum.TEXT)) {
099                                if (retVal.size() > 1) {
100                                        String msg = theRequest
101                                                        .getServer()
102                                                        .getFhirContext()
103                                                        .getLocalizer()
104                                                        .getMessage(SummaryEnumParameter.class, "cantCombineText");
105                                        throw new InvalidRequestException(Msg.code(380) + msg);
106                                }
107                        }
108                }
109
110                return retVal;
111        }
112
113        private static Set<SummaryEnum> toCollectionOrNull(SummaryEnum theFromCode) {
114                if (theFromCode == null) {
115                        return null;
116                }
117                return Collections.singleton(theFromCode);
118        }
119
120        @Override
121        public void initializeTypes(
122                        Method theMethod,
123                        Class<? extends Collection<?>> theOuterCollectionType,
124                        Class<? extends Collection<?>> theInnerCollectionType,
125                        Class<?> theParameterType) {
126                if (theOuterCollectionType != null) {
127                        throw new ConfigurationException(Msg.code(381) + "Method '" + theMethod.getName() + "' in type '"
128                                        + theMethod.getDeclaringClass().getCanonicalName() + "' is of type " + SummaryEnum.class
129                                        + " but can not be a collection of collections");
130                }
131                if (theInnerCollectionType != null) {
132                        myInnerCollectionType = CollectionBinder.getInstantiableCollectionType(
133                                        theInnerCollectionType, SummaryEnum.class.getSimpleName());
134                }
135        }
136}