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.gclient;
021
022import ca.uhn.fhir.context.FhirContext;
023import org.apache.commons.lang3.Validate;
024import org.hl7.fhir.instance.model.api.IIdType;
025
026import java.util.Arrays;
027import java.util.Collection;
028
029import static org.apache.commons.lang3.StringUtils.isNotBlank;
030
031public class ReferenceClientParam extends BaseClientParam implements IParam {
032
033        private String myName;
034
035        public ReferenceClientParam(String theName) {
036                myName = theName;
037        }
038
039        @Override
040        public String getParamName() {
041                return myName;
042        }
043
044        /**
045         * Include a chained search. For example:
046         * <pre>
047         * Bundle resp = ourClient
048         *   .search()
049         *   .forResource(QuestionnaireResponse.class)
050         *   .where(QuestionnaireResponse.SUBJECT.hasChainedProperty(Patient.FAMILY.matches().value("SMITH")))
051         *   .returnBundle(Bundle.class)
052         *   .execute();
053         * </pre>
054         */
055        public ICriterion<ReferenceClientParam> hasChainedProperty(ICriterion<?> theCriterion) {
056                return new ReferenceChainCriterion(getParamName(), theCriterion);
057        }
058
059        /**
060         * Include a chained search with a resource type. For example:
061         * <pre>
062         * Bundle resp = ourClient
063         *   .search()
064         *   .forResource(QuestionnaireResponse.class)
065         *   .where(QuestionnaireResponse.SUBJECT.hasChainedProperty("Patient", Patient.FAMILY.matches().value("SMITH")))
066         *   .returnBundle(Bundle.class)
067         *   .execute();
068         * </pre>
069         */
070        public ICriterion<ReferenceClientParam> hasChainedProperty(String theResourceType, ICriterion<?> theCriterion) {
071                return new ReferenceChainCriterion(getParamName(), theResourceType, theCriterion);
072        }
073
074        /**
075         * Match the referenced resource if the resource has the given ID (this can be
076         * the logical ID or the absolute URL of the resource)
077         */
078        public ICriterion<ReferenceClientParam> hasId(IIdType theId) {
079                return new StringCriterion<>(getParamName(), theId.getValue());
080        }
081
082        /**
083         * Match the referenced resource if the resource has the given ID (this can be
084         * the logical ID or the absolute URL of the resource)
085         */
086        public ICriterion<ReferenceClientParam> hasId(String theId) {
087                return new StringCriterion<>(getParamName(), theId);
088        }
089
090        /**
091         * Match the referenced resource if the resource has ANY of the given IDs
092         * (this is an OR search, not an AND search), (this can be the logical ID or
093         * the absolute URL of the resource). Note that to specify an AND search,
094         * simply add a subsequent {@link IQuery#where(ICriterion) where} criteria
095         * with the same parameter.
096         */
097        public ICriterion<ReferenceClientParam> hasAnyOfIds(Collection<String> theIds) {
098                return new StringCriterion<>(getParamName(), theIds);
099        }
100
101        /**
102         * Match the referenced resource if the resource has ANY of the given IDs
103         * (this is an OR search, not an AND search), (this can be the logical ID or
104         * the absolute URL of the resource). Note that to specify an AND search,
105         * simply add a subsequent {@link IQuery#where(ICriterion) where} criteria
106         * with the same parameter.
107         */
108        public ICriterion<ReferenceClientParam> hasAnyOfIds(String... theIds) {
109                Validate.notNull(theIds, "theIds must not be null");
110                return hasAnyOfIds(Arrays.asList(theIds));
111        }
112
113        private static class ReferenceChainCriterion implements ICriterion<ReferenceClientParam>, ICriterionInternal {
114
115                private final String myResourceTypeQualifier;
116                private String myParamName;
117                private ICriterionInternal myWrappedCriterion;
118
119                ReferenceChainCriterion(String theParamName, ICriterion<?> theWrappedCriterion) {
120                        this(theParamName, null, theWrappedCriterion);
121                }
122
123                ReferenceChainCriterion(String theParamName, String theResourceType, ICriterion<?> theWrappedCriterion) {
124                        myParamName = theParamName;
125                        myResourceTypeQualifier = isNotBlank(theResourceType) ? ":" + theResourceType : "";
126                        myWrappedCriterion = (ICriterionInternal) theWrappedCriterion;
127                }
128
129                @Override
130                public String getParameterName() {
131                        return myParamName + myResourceTypeQualifier + "." + myWrappedCriterion.getParameterName();
132                }
133
134                @Override
135                public String getParameterValue(FhirContext theContext) {
136                        return myWrappedCriterion.getParameterValue(theContext);
137                }
138        }
139}