001/*-
002 * #%L
003 * HAPI FHIR Storage api
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.jpa.binary.provider;
021
022import ca.uhn.fhir.context.BaseRuntimeElementDefinition;
023import ca.uhn.fhir.context.FhirContext;
024import ca.uhn.fhir.i18n.Msg;
025import ca.uhn.fhir.jpa.api.dao.DaoRegistry;
026import ca.uhn.fhir.jpa.api.dao.IFhirResourceDao;
027import ca.uhn.fhir.jpa.api.model.DaoMethodOutcome;
028import ca.uhn.fhir.jpa.binary.api.IBinaryStorageSvc;
029import ca.uhn.fhir.jpa.binary.api.IBinaryTarget;
030import ca.uhn.fhir.jpa.binary.api.StoredDetails;
031import ca.uhn.fhir.jpa.model.util.JpaConstants;
032import ca.uhn.fhir.rest.annotation.IdParam;
033import ca.uhn.fhir.rest.annotation.Operation;
034import ca.uhn.fhir.rest.annotation.OperationParam;
035import ca.uhn.fhir.rest.api.Constants;
036import ca.uhn.fhir.rest.api.EncodingEnum;
037import ca.uhn.fhir.rest.server.RestfulServer;
038import ca.uhn.fhir.rest.server.exceptions.InvalidRequestException;
039import ca.uhn.fhir.rest.server.servlet.ServletRequestDetails;
040import ca.uhn.fhir.util.AttachmentUtil;
041import ca.uhn.fhir.util.BinaryUtil;
042import ca.uhn.fhir.util.DateUtils;
043import ca.uhn.fhir.util.HapiExtensions;
044import com.google.common.annotations.VisibleForTesting;
045import jakarta.annotation.Nonnull;
046import jakarta.servlet.http.HttpServletRequest;
047import jakarta.servlet.http.HttpServletResponse;
048import org.apache.commons.lang3.StringUtils;
049import org.apache.commons.lang3.Validate;
050import org.hl7.fhir.instance.model.api.IBase;
051import org.hl7.fhir.instance.model.api.IBaseBinary;
052import org.hl7.fhir.instance.model.api.IBaseExtension;
053import org.hl7.fhir.instance.model.api.IBaseHasExtensions;
054import org.hl7.fhir.instance.model.api.IBaseResource;
055import org.hl7.fhir.instance.model.api.ICompositeType;
056import org.hl7.fhir.instance.model.api.IIdType;
057import org.hl7.fhir.instance.model.api.IPrimitiveType;
058import org.slf4j.Logger;
059import org.slf4j.LoggerFactory;
060import org.springframework.beans.factory.annotation.Autowired;
061
062import java.io.ByteArrayInputStream;
063import java.io.IOException;
064import java.util.Optional;
065
066import static ca.uhn.fhir.util.UrlUtil.sanitizeUrlPart;
067import static org.apache.commons.lang3.StringUtils.isBlank;
068
069/**
070 * This plain provider class can be registered with a JPA RestfulServer
071 * to provide the <code>$binary-access-read</code> and <code>$binary-access-write</code>
072 * operations that can be used to access attachment data as a raw binary.
073 */
074public class BinaryAccessProvider {
075
076        private static final Logger ourLog = LoggerFactory.getLogger(BinaryAccessProvider.class);
077
078        @Autowired
079        private FhirContext myCtx;
080
081        @Autowired
082        private DaoRegistry myDaoRegistry;
083
084        @Autowired(required = false)
085        private IBinaryStorageSvc myBinaryStorageSvc;
086
087        private Boolean addTargetAttachmentIdForTest = false;
088
089        /**
090         * $binary-access-read
091         */
092        @Operation(
093                        name = JpaConstants.OPERATION_BINARY_ACCESS_READ,
094                        global = true,
095                        manualResponse = true,
096                        idempotent = true)
097        public void binaryAccessRead(
098                        @IdParam IIdType theResourceId,
099                        @OperationParam(name = "path", min = 1, max = 1) IPrimitiveType<String> thePath,
100                        ServletRequestDetails theRequestDetails,
101                        HttpServletRequest theServletRequest,
102                        HttpServletResponse theServletResponse)
103                        throws IOException {
104
105                String path = validateResourceTypeAndPath(theResourceId, thePath);
106                IFhirResourceDao dao = getDaoForRequest(theResourceId);
107                IBaseResource resource = dao.read(theResourceId, theRequestDetails, false);
108
109                IBinaryTarget target = findAttachmentForRequest(resource, path, theRequestDetails);
110                Optional<String> attachmentId = target.getAttachmentId();
111
112                // for unit test only
113                if (addTargetAttachmentIdForTest) {
114                        attachmentId = Optional.of("1");
115                }
116
117                if (attachmentId.isPresent()) {
118
119                        String blobId = attachmentId.get();
120
121                        StoredDetails blobDetails = myBinaryStorageSvc.fetchBinaryContentDetails(theResourceId, blobId);
122                        if (blobDetails == null) {
123                                String msg = myCtx.getLocalizer().getMessage(BinaryAccessProvider.class, "unknownBlobId");
124                                throw new InvalidRequestException(Msg.code(1331) + msg);
125                        }
126
127                        theServletResponse.setStatus(200);
128                        theServletResponse.setContentType(blobDetails.getContentType());
129                        if (blobDetails.getBytes() <= Integer.MAX_VALUE) {
130                                theServletResponse.setContentLength((int) blobDetails.getBytes());
131                        }
132
133                        RestfulServer server = theRequestDetails.getServer();
134                        server.addHeadersToResponse(theServletResponse);
135
136                        theServletResponse.addHeader(Constants.HEADER_CACHE_CONTROL, Constants.CACHE_CONTROL_PRIVATE);
137                        theServletResponse.addHeader(Constants.HEADER_ETAG, '"' + blobDetails.getHash() + '"');
138                        theServletResponse.addHeader(
139                                        Constants.HEADER_LAST_MODIFIED, DateUtils.formatDate(blobDetails.getPublished()));
140
141                        myBinaryStorageSvc.writeBinaryContent(theResourceId, blobId, theServletResponse.getOutputStream());
142                        theServletResponse.getOutputStream().close();
143
144                } else {
145                        String contentType = target.getContentType();
146                        contentType = StringUtils.defaultIfBlank(contentType, Constants.CT_OCTET_STREAM);
147
148                        byte[] data = target.getData();
149                        if (data == null) {
150                                String msg = myCtx.getLocalizer()
151                                                .getMessage(
152                                                                BinaryAccessProvider.class,
153                                                                "noAttachmentDataPresent",
154                                                                sanitizeUrlPart(theResourceId),
155                                                                sanitizeUrlPart(thePath));
156                                throw new InvalidRequestException(Msg.code(1332) + msg);
157                        }
158
159                        theServletResponse.setStatus(200);
160                        theServletResponse.setContentType(contentType);
161                        theServletResponse.setContentLength(data.length);
162
163                        RestfulServer server = theRequestDetails.getServer();
164                        server.addHeadersToResponse(theServletResponse);
165
166                        theServletResponse.getOutputStream().write(data);
167                        theServletResponse.getOutputStream().close();
168                }
169        }
170
171        /**
172         * $binary-access-write
173         */
174        @SuppressWarnings("unchecked")
175        @Operation(
176                        name = JpaConstants.OPERATION_BINARY_ACCESS_WRITE,
177                        global = true,
178                        manualRequest = true,
179                        idempotent = false)
180        public IBaseResource binaryAccessWrite(
181                        @IdParam IIdType theResourceId,
182                        @OperationParam(name = "path", min = 1, max = 1) IPrimitiveType<String> thePath,
183                        ServletRequestDetails theRequestDetails,
184                        HttpServletRequest theServletRequest,
185                        HttpServletResponse theServletResponse)
186                        throws IOException {
187
188                String path = validateResourceTypeAndPath(theResourceId, thePath);
189                IFhirResourceDao dao = getDaoForRequest(theResourceId);
190                IBaseResource resource = dao.read(theResourceId, theRequestDetails, false);
191
192                IBinaryTarget target = findAttachmentForRequest(resource, path, theRequestDetails);
193
194                String requestContentType = theServletRequest.getContentType();
195                if (isBlank(requestContentType)) {
196                        throw new InvalidRequestException(Msg.code(1333) + "No content-target supplied");
197                }
198                if (EncodingEnum.forContentTypeStrict(requestContentType) != null) {
199                        throw new InvalidRequestException(
200                                        Msg.code(1334) + "This operation is for binary content, got: " + requestContentType);
201                }
202
203                long size = theServletRequest.getContentLength();
204                ourLog.trace("Request specified content length: {}", size);
205
206                String blobId = null;
207                byte[] bytes = theRequestDetails.loadRequestContents();
208
209                if (size > 0 && myBinaryStorageSvc != null) {
210                        if (bytes == null || bytes.length == 0) {
211                                throw new IllegalStateException(
212                                                Msg.code(2073)
213                                                                + "Input stream is empty! Ensure that you are uploading data, and if so, ensure that no interceptors are in use that may be consuming the input stream");
214                        }
215                        if (myBinaryStorageSvc.shouldStoreBinaryContent(size, theResourceId, requestContentType)) {
216                                StoredDetails storedDetails = myBinaryStorageSvc.storeBinaryContent(
217                                                theResourceId, null, requestContentType, new ByteArrayInputStream(bytes), theRequestDetails);
218                                size = storedDetails.getBytes();
219                                blobId = storedDetails.getBinaryContentId();
220                                Validate.notBlank(blobId, "BinaryStorageSvc returned a null blob ID"); // should not happen
221                                Validate.isTrue(size == theServletRequest.getContentLength(), "Unexpected stored size"); // Sanity check
222                        }
223                }
224
225                if (blobId == null) {
226                        size = bytes.length;
227                        target.setData(bytes);
228                } else {
229                        replaceDataWithExtension(target, blobId);
230                }
231
232                target.setContentType(requestContentType);
233                target.setSize(null);
234                if (size <= Integer.MAX_VALUE) {
235                        target.setSize((int) size);
236                }
237
238                DaoMethodOutcome outcome = dao.update(resource, theRequestDetails);
239                return outcome.getResource();
240        }
241
242        public void replaceDataWithExtension(IBinaryTarget theTarget, String theBlobId) {
243                theTarget
244                                .getTarget()
245                                .getExtension()
246                                .removeIf(t -> HapiExtensions.EXT_EXTERNALIZED_BINARY_ID.equals(t.getUrl()));
247                theTarget.setData(null);
248
249                IBaseExtension<?, ?> ext = theTarget.getTarget().addExtension();
250                ext.setUrl(HapiExtensions.EXT_EXTERNALIZED_BINARY_ID);
251                ext.setUserData(JpaConstants.EXTENSION_EXT_SYSTEMDEFINED, Boolean.TRUE);
252                IPrimitiveType<String> blobIdString =
253                                (IPrimitiveType<String>) myCtx.getElementDefinition("string").newInstance();
254                blobIdString.setValueAsString(theBlobId);
255                ext.setValue(blobIdString);
256        }
257
258        @Nonnull
259        private IBinaryTarget findAttachmentForRequest(
260                        IBaseResource theResource, String thePath, ServletRequestDetails theRequestDetails) {
261                Optional<IBase> type = myCtx.newFluentPath().evaluateFirst(theResource, thePath, IBase.class);
262                String resType = this.myCtx.getResourceType(theResource);
263                if (type.isEmpty()) {
264                        String msg = this.myCtx
265                                        .getLocalizer()
266                                        .getMessageSanitized(BinaryAccessProvider.class, "unknownPath", resType, thePath);
267                        throw new InvalidRequestException(Msg.code(1335) + msg);
268                }
269                IBase element = type.get();
270
271                Optional<IBinaryTarget> binaryTarget = toBinaryTarget(element);
272
273                if (binaryTarget.isEmpty()) {
274                        BaseRuntimeElementDefinition<?> def2 = myCtx.getElementDefinition(element.getClass());
275                        String msg = this.myCtx
276                                        .getLocalizer()
277                                        .getMessageSanitized(BinaryAccessProvider.class, "unknownType", resType, thePath, def2.getName());
278                        throw new InvalidRequestException(Msg.code(1336) + msg);
279                } else {
280                        return binaryTarget.get();
281                }
282        }
283
284        public Optional<IBinaryTarget> toBinaryTarget(IBase theElement) {
285                IBinaryTarget binaryTarget = null;
286
287                // Path is attachment
288                BaseRuntimeElementDefinition<?> def = myCtx.getElementDefinition(theElement.getClass());
289                if (def.getName().equals("Attachment")) {
290                        ICompositeType attachment = (ICompositeType) theElement;
291                        binaryTarget = new IBinaryTarget() {
292                                @Override
293                                public void setSize(Integer theSize) {
294                                        AttachmentUtil.setSize(BinaryAccessProvider.this.myCtx, attachment, theSize);
295                                }
296
297                                @Override
298                                public String getContentType() {
299                                        return AttachmentUtil.getOrCreateContentType(BinaryAccessProvider.this.myCtx, attachment)
300                                                        .getValueAsString();
301                                }
302
303                                @Override
304                                public byte[] getData() {
305                                        IPrimitiveType<byte[]> dataDt = AttachmentUtil.getOrCreateData(myCtx, attachment);
306                                        return dataDt.getValue();
307                                }
308
309                                @Override
310                                public IBaseHasExtensions getTarget() {
311                                        return (IBaseHasExtensions) AttachmentUtil.getOrCreateData(myCtx, attachment);
312                                }
313
314                                @Override
315                                public void setContentType(String theContentType) {
316                                        AttachmentUtil.setContentType(BinaryAccessProvider.this.myCtx, attachment, theContentType);
317                                }
318
319                                @Override
320                                public void setData(byte[] theBytes) {
321                                        AttachmentUtil.setData(myCtx, attachment, theBytes);
322                                }
323                        };
324                }
325
326                // Path is Binary
327                if (def.getName().equals("Binary")) {
328                        IBaseBinary binary = (IBaseBinary) theElement;
329                        binaryTarget = new IBinaryTarget() {
330                                @Override
331                                public void setSize(Integer theSize) {
332                                        // ignore
333                                }
334
335                                @Override
336                                public String getContentType() {
337                                        return binary.getContentType();
338                                }
339
340                                @Override
341                                public byte[] getData() {
342                                        return binary.getContent();
343                                }
344
345                                @Override
346                                public IBaseHasExtensions getTarget() {
347                                        return (IBaseHasExtensions) BinaryUtil.getOrCreateData(BinaryAccessProvider.this.myCtx, binary);
348                                }
349
350                                @Override
351                                public void setContentType(String theContentType) {
352                                        binary.setContentType(theContentType);
353                                }
354
355                                @Override
356                                public void setData(byte[] theBytes) {
357                                        binary.setContent(theBytes);
358                                }
359                        };
360                }
361
362                return Optional.ofNullable(binaryTarget);
363        }
364
365        private String validateResourceTypeAndPath(
366                        @IdParam IIdType theResourceId,
367                        @OperationParam(name = "path", min = 1, max = 1) IPrimitiveType<String> thePath) {
368                if (isBlank(theResourceId.getResourceType())) {
369                        throw new InvalidRequestException(Msg.code(1337) + "No resource type specified");
370                }
371                if (isBlank(theResourceId.getIdPart())) {
372                        throw new InvalidRequestException(Msg.code(1338) + "No ID specified");
373                }
374                if (thePath == null || isBlank(thePath.getValue())) {
375                        if ("Binary".equals(theResourceId.getResourceType())) {
376                                return "Binary";
377                        }
378                        throw new InvalidRequestException(Msg.code(1339) + "No path specified");
379                }
380
381                return thePath.getValue();
382        }
383
384        @Nonnull
385        private IFhirResourceDao getDaoForRequest(@IdParam IIdType theResourceId) {
386                String resourceType = theResourceId.getResourceType();
387                IFhirResourceDao dao = myDaoRegistry.getResourceDao(resourceType);
388                if (dao == null) {
389                        throw new InvalidRequestException(
390                                        Msg.code(1340) + "Unknown/unsupported resource type: " + sanitizeUrlPart(resourceType));
391                }
392                return dao;
393        }
394
395        @VisibleForTesting
396        public void setDaoRegistryForUnitTest(DaoRegistry theDaoRegistry) {
397                myDaoRegistry = theDaoRegistry;
398        }
399
400        @VisibleForTesting
401        public void setBinaryStorageSvcForUnitTest(IBinaryStorageSvc theBinaryStorageSvc) {
402                myBinaryStorageSvc = theBinaryStorageSvc;
403        }
404
405        @VisibleForTesting
406        public void setFhirContextForUnitTest(FhirContext theCtx) {
407                myCtx = theCtx;
408        }
409
410        @VisibleForTesting
411        public void setTargetAttachmentIdForUnitTest(Boolean theTargetAttachmentIdForTest) {
412                addTargetAttachmentIdForTest = theTargetAttachmentIdForTest;
413        }
414}