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.api; 021 022import ca.uhn.fhir.model.api.IModelJson; 023import ca.uhn.fhir.rest.server.util.JsonDateDeserializer; 024import ca.uhn.fhir.rest.server.util.JsonDateSerializer; 025import com.fasterxml.jackson.annotation.JsonProperty; 026import com.fasterxml.jackson.databind.annotation.JsonDeserialize; 027import com.fasterxml.jackson.databind.annotation.JsonSerialize; 028import com.google.common.hash.HashingInputStream; 029import jakarta.annotation.Nonnull; 030import org.apache.commons.lang3.builder.ToStringBuilder; 031 032import java.util.Date; 033 034public class StoredDetails implements IModelJson { 035 036 @JsonProperty(value = "binaryContentId") 037 private String myBinaryContentId; 038 039 /** 040 * This field exists to fix a break that changing this property name caused. 041 * in 7.2.0 we went from blobId to binaryContentId. However this did not consider installations using filesystem 042 * mode storage in which the data on disk was not updated, and needed to be Serialized/Deserialized at runtime. 043 * Existing stored details used `blobId`. This causes Jackson deserialization failures which are tough to recover 044 * from without manually modifying all those stored details 045 * on disk. 046 * This field is a relic to support old blobs post-upgrade to 7.2.0. It is not ever surfaced to the user, and is proxied 047 * into `myBinaryContentId` when needed. 048 */ 049 @JsonProperty(value = "blobId") 050 private String myBlobId; 051 052 @JsonProperty("bytes") 053 private long myBytes; 054 055 @JsonProperty("contentType") 056 private String myContentType; 057 058 @JsonProperty("hash") 059 private String myHash; 060 061 @JsonProperty("published") 062 @JsonSerialize(using = JsonDateSerializer.class) 063 @JsonDeserialize(using = JsonDateDeserializer.class) 064 private Date myPublished; 065 066 /** 067 * Constructor 068 */ 069 @SuppressWarnings("unused") 070 public StoredDetails() { 071 super(); 072 } 073 074 /** 075 * Constructor 076 */ 077 public StoredDetails( 078 @Nonnull String theBinaryContentId, 079 long theBytes, 080 @Nonnull String theContentType, 081 HashingInputStream theIs, 082 Date thePublished) { 083 myBinaryContentId = theBinaryContentId; 084 myBytes = theBytes; 085 myContentType = theContentType; 086 myHash = theIs.hash().toString(); 087 myPublished = thePublished; 088 } 089 090 @Override 091 public String toString() { 092 return new ToStringBuilder(this) 093 .append("binaryContentId", getBinaryContentId()) 094 .append("bytes", myBytes) 095 .append("contentType", myContentType) 096 .append("hash", myHash) 097 .append("published", myPublished) 098 .toString(); 099 } 100 101 public String getHash() { 102 return myHash; 103 } 104 105 public StoredDetails setHash(String theHash) { 106 myHash = theHash; 107 return this; 108 } 109 110 public Date getPublished() { 111 return myPublished; 112 } 113 114 public StoredDetails setPublished(Date thePublished) { 115 myPublished = thePublished; 116 return this; 117 } 118 119 @Nonnull 120 public String getContentType() { 121 return myContentType; 122 } 123 124 public StoredDetails setContentType(String theContentType) { 125 myContentType = theContentType; 126 return this; 127 } 128 129 @Nonnull 130 public String getBinaryContentId() { 131 if (myBinaryContentId == null && myBlobId != null) { 132 return myBlobId; 133 } else { 134 return myBinaryContentId; 135 } 136 } 137 138 public StoredDetails setBinaryContentId(String theBinaryContentId) { 139 myBinaryContentId = theBinaryContentId; 140 return this; 141 } 142 143 public long getBytes() { 144 return myBytes; 145 } 146 147 public StoredDetails setBytes(long theBytes) { 148 myBytes = theBytes; 149 return this; 150 } 151}