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.api;
021
022import ca.uhn.fhir.util.CoverageIgnore;
023import org.apache.commons.lang3.Validate;
024import org.hl7.fhir.instance.model.api.IBaseOperationOutcome;
025import org.hl7.fhir.instance.model.api.IBaseResource;
026import org.hl7.fhir.instance.model.api.IIdType;
027
028import java.util.ArrayList;
029import java.util.Collection;
030import java.util.HashMap;
031import java.util.List;
032import java.util.Map;
033import java.util.Optional;
034
035public class MethodOutcome {
036
037        private Boolean myCreated;
038        private IIdType myId;
039        private IBaseOperationOutcome myOperationOutcome;
040        private IBaseResource myResource;
041        private Map<String, List<String>> myResponseHeaders;
042        private Collection<Runnable> myResourceViewCallbacks;
043        private Integer myResponseStatusCode;
044
045        /**
046         * Constructor
047         */
048        public MethodOutcome() {
049                super();
050        }
051
052        /**
053         * Constructor
054         *
055         * @param theId      The ID of the created/updated resource
056         * @param theCreated If not null, indicates whether the resource was created (as opposed to being updated). This is generally not needed, since the server can assume based on the method being called
057         *                   whether the result was a creation or an update. However, it can be useful if you are implementing an update method that does a create if the ID doesn't already exist.
058         */
059        @CoverageIgnore
060        public MethodOutcome(IIdType theId, Boolean theCreated) {
061                myId = theId;
062                myCreated = theCreated;
063        }
064
065        /**
066         * Constructor
067         *
068         * @param theId                   The ID of the created/updated resource
069         * @param theBaseOperationOutcome The operation outcome to return with the response (or null for none)
070         */
071        public MethodOutcome(IIdType theId, IBaseOperationOutcome theBaseOperationOutcome) {
072                myId = theId;
073                myOperationOutcome = theBaseOperationOutcome;
074        }
075
076        /**
077         * Constructor
078         *
079         * @param theId                   The ID of the created/updated resource
080         * @param theBaseOperationOutcome The operation outcome to return with the response (or null for none)
081         * @param theCreated              If not null, indicates whether the resource was created (as opposed to being updated). This is generally not needed, since the server can assume based on the method being called
082         *                                whether the result was a creation or an update. However, it can be useful if you are implementing an update method that does a create if the ID doesn't already exist.
083         */
084        public MethodOutcome(IIdType theId, IBaseOperationOutcome theBaseOperationOutcome, Boolean theCreated) {
085                myId = theId;
086                myOperationOutcome = theBaseOperationOutcome;
087                myCreated = theCreated;
088        }
089
090        /**
091         * Constructor
092         *
093         * @param theId The ID of the created/updated resource
094         */
095        public MethodOutcome(IIdType theId) {
096                myId = theId;
097        }
098
099        /**
100         * Constructor
101         *
102         * @param theOperationOutcome The operation outcome resource to return
103         */
104        public MethodOutcome(IBaseOperationOutcome theOperationOutcome) {
105                myOperationOutcome = theOperationOutcome;
106        }
107
108        /**
109         * This will be set to {@link Boolean#TRUE} for instance of MethodOutcome which are
110         * returned to client instances, if the server has responded with an HTTP 201 Created.
111         */
112        public Boolean getCreated() {
113                return myCreated;
114        }
115
116        /**
117         * If not null, indicates whether the resource was created (as opposed to being updated). This is generally not needed, since the server can assume based on the method being called whether the
118         * result was a creation or an update. However, it can be useful if you are implementing an update method that does a create if the ID doesn't already exist.
119         * <p>
120         * Users of HAPI should only interact with this method in Server applications
121         * </p>
122         *
123         * @param theCreated If not null, indicates whether the resource was created (as opposed to being updated). This is generally not needed, since the server can assume based on the method being called
124         *                   whether the result was a creation or an update. However, it can be useful if you are implementing an update method that does a create if the ID doesn't already exist.
125         * @return Returns a reference to <code>this</code> for easy method chaining
126         */
127        public MethodOutcome setCreated(Boolean theCreated) {
128                myCreated = theCreated;
129                return this;
130        }
131
132        public IIdType getId() {
133                return myId;
134        }
135
136        /**
137         * @param theId The ID of the created/updated resource
138         * @return Returns a reference to <code>this</code> for easy method chaining
139         */
140        public MethodOutcome setId(IIdType theId) {
141                myId = theId;
142                return this;
143        }
144
145        /**
146         * Returns the {@link IBaseOperationOutcome} resource to return to the client or <code>null</code> if none.
147         *
148         * @return This method <b>will return null</b>, unlike many methods in the API.
149         */
150        public IBaseOperationOutcome getOperationOutcome() {
151                return myOperationOutcome;
152        }
153
154        /**
155         * Sets the {@link IBaseOperationOutcome} resource to return to the client. Set to <code>null</code> (which is the default) if none.
156         *
157         * @return Returns a reference to <code>this</code> for easy method chaining
158         */
159        public MethodOutcome setOperationOutcome(IBaseOperationOutcome theBaseOperationOutcome) {
160                myOperationOutcome = theBaseOperationOutcome;
161                return this;
162        }
163
164        /**
165         * <b>From a client response:</b> If the method returned an actual resource body (e.g. a create/update with
166         * "Prefer: return=representation") this field will be populated with the
167         * resource itself.
168         */
169        public IBaseResource getResource() {
170                return myResource;
171        }
172
173        /**
174         * <b>In a server response</b>: This field may be populated in server code with the final resource for operations
175         * where a resource body is being created/updated. E.g. for an update method, this field could be populated with
176         * the resource after the update is applied, with the new version ID, lastUpdate time, etc.
177         * <p>
178         * This field is optional, but if it is populated the server will return the resource body if requested to
179         * do so via the HTTP Prefer header.
180         * </p>
181         *
182         * @return Returns a reference to <code>this</code> for easy method chaining
183         * @see #registerResourceViewCallback(Runnable) to register a callback that should be invoked by the framework before the resource is shown/returned to a client
184         */
185        public MethodOutcome setResource(IBaseResource theResource) {
186                myResource = theResource;
187                return this;
188        }
189
190        /**
191         * Gets the headers for the HTTP response
192         */
193        public Map<String, List<String>> getResponseHeaders() {
194                if (myResponseHeaders == null) {
195                        myResponseHeaders = new HashMap<>();
196                }
197                return myResponseHeaders;
198        }
199
200        /**
201         * Sets the headers for the HTTP response
202         */
203        public void setResponseHeaders(Map<String, List<String>> theResponseHeaders) {
204                myResponseHeaders = theResponseHeaders;
205        }
206
207        public Optional<String> getFirstResponseHeader(String theHeader) {
208                List<String> values = getResponseHeaders().get(theHeader);
209
210                if (values == null || values.isEmpty()) {
211                        return Optional.empty();
212                } else {
213                        return Optional.of(values.get(0));
214                }
215        }
216
217        /**
218         * Registers a callback to be invoked before the resource in this object gets
219         * returned to the client. Note that this is an experimental API and may change.
220         *
221         * @param theCallback The callback
222         * @since 4.0.0
223         */
224        public void registerResourceViewCallback(Runnable theCallback) {
225                Validate.notNull(theCallback, "theCallback must not be null");
226
227                if (myResourceViewCallbacks == null) {
228                        myResourceViewCallbacks = new ArrayList<>(2);
229                }
230                myResourceViewCallbacks.add(theCallback);
231        }
232
233        /**
234         * Fires callbacks registered to {@link #registerResourceViewCallback(Runnable)} and then
235         * clears the list of registered callbacks.
236         *
237         * @since 4.0.0
238         */
239        public void fireResourceViewCallbacks() {
240                if (myResourceViewCallbacks != null) {
241                        myResourceViewCallbacks.forEach(t -> t.run());
242                        myResourceViewCallbacks.clear();
243                }
244        }
245
246        public void setCreatedUsingStatusCode(int theResponseStatusCode) {
247                if (theResponseStatusCode == Constants.STATUS_HTTP_201_CREATED) {
248                        setCreated(true);
249                }
250        }
251
252        protected boolean hasResource() {
253                return myResource != null;
254        }
255
256        public void setResponseStatusCode(int theResponseStatusCode) {
257                myResponseStatusCode = theResponseStatusCode;
258        }
259
260        public int getResponseStatusCode() {
261                return isResponseStatusCodeSet() ? myResponseStatusCode : 0;
262        }
263
264        public boolean isResponseStatusCodeSet() {
265                return myResponseStatusCode != null;
266        }
267}