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.param;
021
022import ca.uhn.fhir.context.FhirContext;
023import ca.uhn.fhir.i18n.Msg;
024import ca.uhn.fhir.model.base.composite.BaseCodingDt;
025import ca.uhn.fhir.model.base.composite.BaseIdentifierDt;
026import ca.uhn.fhir.model.primitive.UriDt;
027import ca.uhn.fhir.rest.api.Constants;
028import org.apache.commons.lang3.StringUtils;
029import org.apache.commons.lang3.builder.EqualsBuilder;
030import org.apache.commons.lang3.builder.HashCodeBuilder;
031import org.apache.commons.lang3.builder.ToStringBuilder;
032import org.apache.commons.lang3.builder.ToStringStyle;
033import org.hl7.fhir.instance.model.api.IBaseCoding;
034
035import static org.apache.commons.lang3.StringUtils.defaultString;
036import static org.apache.commons.lang3.StringUtils.isNotBlank;
037
038public class TokenParam extends BaseParam /*implements IQueryParameterType*/ {
039
040        private TokenParamModifier myModifier;
041        private String mySystem;
042        private String myValue;
043
044        private Boolean myMdmExpand;
045
046        /**
047         * Constructor
048         */
049        public TokenParam() {
050                super();
051        }
052
053        /**
054         * Constructor which copies the {@link InternalCodingDt#getSystemElement() system} and
055         * {@link InternalCodingDt#getCodeElement() code} from a {@link InternalCodingDt} instance and adds it as a parameter
056         *
057         * @param theCodingDt The coding
058         */
059        public TokenParam(BaseCodingDt theCodingDt) {
060                this(
061                                toSystemValue(theCodingDt.getSystemElement()),
062                                theCodingDt.getCodeElement().getValue());
063        }
064
065        /**
066         * Constructor which copies the {@link BaseIdentifierDt#getSystemElement() system} and
067         * {@link BaseIdentifierDt#getValueElement() value} from a {@link BaseIdentifierDt} instance and adds it as a
068         * parameter
069         *
070         * @param theIdentifierDt The identifier
071         */
072        public TokenParam(BaseIdentifierDt theIdentifierDt) {
073                this(
074                                toSystemValue(theIdentifierDt.getSystemElement()),
075                                theIdentifierDt.getValueElement().getValue());
076        }
077
078        /**
079         * Construct a {@link TokenParam} from the {@link IBaseCoding#getSystem()} () system} and
080         * {@link IBaseCoding#getCode()} () code} of a {@link IBaseCoding} instance.
081         *
082         * @param theCoding The coding
083         */
084        public TokenParam(IBaseCoding theCoding) {
085                this(theCoding.getSystem(), theCoding.getCode());
086        }
087
088        public TokenParam(String theSystem, String theValue) {
089                setSystem(theSystem);
090                setValue(theValue);
091        }
092
093        public TokenParam(String theSystem, String theValue, boolean theText) {
094                if (theText && isNotBlank(theSystem)) {
095                        throw new IllegalArgumentException(
096                                        Msg.code(1938)
097                                                        + "theSystem can not be non-blank if theText is true (:text searches do not include a system). In other words, set the first parameter to null for a text search");
098                }
099                setSystem(theSystem);
100                setValue(theValue);
101                setText(theText);
102        }
103
104        /**
105         * Constructor that takes a code but no system
106         */
107        public TokenParam(String theCode) {
108                this(null, theCode);
109        }
110
111        public boolean isMdmExpand() {
112                return myMdmExpand != null && myMdmExpand;
113        }
114
115        public TokenParam setMdmExpand(boolean theMdmExpand) {
116                myMdmExpand = theMdmExpand;
117                return this;
118        }
119
120        @Override
121        String doGetQueryParameterQualifier() {
122                if (getModifier() != null) {
123                        return getModifier().getValue();
124                }
125                return null;
126        }
127
128        /**
129         * {@inheritDoc}
130         */
131        @Override
132        String doGetValueAsQueryToken(FhirContext theContext) {
133                if (getSystem() != null) {
134                        if (getValue() != null) {
135                                return ParameterUtil.escape(StringUtils.defaultString(getSystem()))
136                                                + '|'
137                                                + ParameterUtil.escape(getValue());
138                        } else {
139                                return ParameterUtil.escape(StringUtils.defaultString(getSystem())) + '|';
140                        }
141                }
142                return ParameterUtil.escape(getValue());
143        }
144
145        /**
146         * {@inheritDoc}
147         */
148        @Override
149        void doSetValueAsQueryToken(FhirContext theContext, String theParamName, String theQualifier, String theParameter) {
150                setModifier(null);
151                setSystem(null);
152
153                if (theQualifier != null) {
154                        if (Constants.PARAMQUALIFIER_MDM.equals(theQualifier)) {
155                                setMdmExpand(true);
156                        }
157
158                        TokenParamModifier modifier = TokenParamModifier.forValue(theQualifier);
159                        setModifier(modifier);
160
161                        if (modifier == TokenParamModifier.TEXT) {
162                                setValue(ParameterUtil.unescape(theParameter));
163                                return;
164                        }
165                }
166
167                if (theParameter == null) {
168                        setValue(null);
169                } else {
170                        int barIndex = ParameterUtil.nonEscapedIndexOf(theParameter, '|');
171                        if (barIndex != -1) {
172                                setSystem(theParameter.substring(0, barIndex));
173                                setValue(ParameterUtil.unescape(theParameter.substring(barIndex + 1)));
174                        } else {
175                                setValue(ParameterUtil.unescape(theParameter));
176                        }
177                }
178        }
179
180        /**
181         * Returns the modifier for this token
182         */
183        public TokenParamModifier getModifier() {
184                return myModifier;
185        }
186
187        public TokenParam setModifier(TokenParamModifier theModifier) {
188                myModifier = theModifier;
189                return this;
190        }
191
192        /**
193         * Returns the system for this token. Note that if a {@link #getModifier()} is being used, the entire value of the
194         * parameter will be placed in {@link #getValue() value} and this method will return <code>null</code>.
195         * <p
196         * Also note that this value may be <code>null</code> or <code>""</code> (empty string) and that
197         * each of these have a different meaning. When a token is passed on a URL and it has no
198         * vertical bar (often meaning "return values that match the given code in any codesystem")
199         * this method will return <code>null</code>. When a token is passed on a URL and it has
200         * a vetical bar but nothing before the bar (often meaning "return values that match the
201         * given code but that have no codesystem) this method will return <code>""</code>
202         * </p>
203         */
204        public String getSystem() {
205                return mySystem;
206        }
207
208        public TokenParam setSystem(String theSystem) {
209                mySystem = theSystem;
210                return this;
211        }
212
213        /**
214         * Returns the value for the token (generally the value to the right of the
215         * vertical bar on the URL)
216         */
217        public String getValue() {
218                return myValue;
219        }
220
221        public TokenParam setValue(String theValue) {
222                myValue = theValue;
223                return this;
224        }
225
226        public InternalCodingDt getValueAsCoding() {
227                return new InternalCodingDt(mySystem, myValue);
228        }
229
230        public String getValueNotNull() {
231                return defaultString(myValue);
232        }
233
234        public boolean isEmpty() {
235                return StringUtils.isBlank(mySystem) && StringUtils.isBlank(myValue) && getMissing() == null;
236        }
237
238        /**
239         * Returns true if {@link #getModifier()} returns {@link TokenParamModifier#TEXT}
240         */
241        public boolean isText() {
242                return myModifier == TokenParamModifier.TEXT;
243        }
244
245        /**
246         * @deprecated Use {@link #setModifier(TokenParamModifier)} instead
247         */
248        @Deprecated
249        public TokenParam setText(boolean theText) {
250                if (theText) {
251                        myModifier = TokenParamModifier.TEXT;
252                } else {
253                        myModifier = null;
254                }
255                return this;
256        }
257
258        @Override
259        public String toString() {
260                ToStringBuilder builder = new ToStringBuilder(this, ToStringStyle.SHORT_PREFIX_STYLE);
261                builder.append("system", defaultString(getSystem()));
262                if (myModifier != null) {
263                        builder.append(":" + myModifier.getValue());
264                }
265                builder.append("value", getValue());
266                if (getMissing() != null) {
267                        builder.append(":missing", getMissing());
268                }
269                return builder.toString();
270        }
271
272        @Override
273        public boolean equals(Object theO) {
274                if (this == theO) {
275                        return true;
276                }
277
278                if (theO == null || getClass() != theO.getClass()) {
279                        return false;
280                }
281
282                TokenParam that = (TokenParam) theO;
283
284                EqualsBuilder b = new EqualsBuilder();
285                b.append(myModifier, that.myModifier);
286                b.append(mySystem, that.mySystem);
287                b.append(myValue, that.myValue);
288                return b.isEquals();
289        }
290
291        @Override
292        public int hashCode() {
293                HashCodeBuilder b = new HashCodeBuilder(17, 37);
294                b.append(myModifier);
295                b.append(mySystem);
296                b.append(myValue);
297                return b.toHashCode();
298        }
299
300        private static String toSystemValue(UriDt theSystem) {
301                return theSystem.getValueAsString();
302        }
303}