001/* 002 * Licensed under the Apache License, Version 2.0 (the "License"); 003 * you may not use this file except in compliance with the License. 004 * You may obtain a copy of the License at 005 * 006 * http://www.apache.org/licenses/LICENSE-2.0 007 * 008 * Unless required by applicable law or agreed to in writing, software 009 * distributed under the License is distributed on an "AS IS" BASIS, 010 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 011 * See the License for the specific language governing permissions and 012 * limitations under the License. 013 */ 014package org.gbif.api.model.occurrence; 015 016import com.fasterxml.jackson.databind.JsonNode; 017 018import io.swagger.v3.oas.annotations.Hidden; 019import io.swagger.v3.oas.annotations.media.ArraySchema; 020import io.swagger.v3.oas.annotations.media.Schema; 021 022import org.gbif.api.jackson.DownloadRequestSerde; 023 024import java.io.Serializable; 025import java.util.Collection; 026import java.util.Collections; 027import java.util.HashSet; 028import java.util.Objects; 029import java.util.Set; 030import java.util.StringJoiner; 031import java.util.stream.Collectors; 032import java.util.stream.Stream; 033 034import javax.annotation.Nullable; 035 036import org.apache.commons.lang3.StringUtils; 037 038import com.fasterxml.jackson.annotation.JsonIgnore; 039import com.fasterxml.jackson.annotation.JsonProperty; 040import com.fasterxml.jackson.databind.annotation.JsonDeserialize; 041 042/** 043 * Represents a request to download occurrence records. 044 * This is the base class for specific type of downloads: predicate based downloads and SQL downloads. 045 */ 046@Schema( 047 description = "An occurrence download request.", 048 oneOf = {PredicateDownloadRequest.class, SqlDownloadRequest.class} 049) 050@SuppressWarnings("unused") 051@JsonDeserialize(using = DownloadRequestSerde.class) 052public abstract class DownloadRequest implements Serializable { 053 054 private static final String DELIMITER = ","; 055 056 @Schema( 057 description = "The GBIF username of the initiator of the download request." 058 ) 059 @JsonProperty("creator") 060 private String creator; 061 062 @ArraySchema( 063 schema = @Schema( 064 description = "An email addresses to notify when the download finishes." 065 ) 066 ) 067 @JsonProperty("notificationAddresses") 068 private Set<String> notificationAddresses; 069 070 @Schema( 071 description = "Whether to send a notification email when the download finishes." 072 ) 073 @JsonProperty("sendNotification") 074 private Boolean sendNotification; 075 076 @Schema( 077 description = "The data format of the download." 078 ) 079 @JsonProperty("format") 080 private DownloadFormat format; 081 082 @Hidden 083 @JsonProperty("type") 084 private DownloadType type; 085 086 @Schema( 087 description = "A user-specified description of the download, such as the intended purpose or a tag " + 088 "for later reference." 089 ) 090 @JsonProperty("description") 091 private String description; 092 093 @JsonProperty("machineDescription") 094 private JsonNode machineDescription; 095 096 /** 097 * Default constructor. 098 */ 099 public DownloadRequest() { 100 // Empty constructor required to create instances from the data access layer. 101 } 102 103 public DownloadRequest(String creator, Collection<String> notificationAddresses, 104 Boolean sendNotification, DownloadFormat format, 105 DownloadType downloadType, String description, 106 JsonNode machineDescription 107 ) { 108 this.creator = creator; 109 this.notificationAddresses = notificationAddresses == null ? Collections.emptySet() : 110 Collections.unmodifiableSet(new HashSet<>(notificationAddresses)); 111 this.sendNotification = sendNotification; 112 this.format = format; 113 this.type = downloadType; 114 this.description = description; 115 this.machineDescription = machineDescription; 116 } 117 118 /** 119 * @return the user account that initiated the download 120 */ 121 @Nullable 122 public String getCreator() { 123 return creator; 124 } 125 126 public void setCreator(String creator) { 127 this.creator = creator; 128 } 129 130 /** 131 * @return set of email addresses for notifications 132 */ 133 @Nullable 134 public Set<String> getNotificationAddresses() { 135 return notificationAddresses; 136 } 137 138 public void setNotificationAddresses(Set<String> notificationAddresses) { 139 this.notificationAddresses = notificationAddresses; 140 } 141 142 /** 143 * Returns the notification addresses as single string. The emails are separated by ','. 144 */ 145 @Nullable 146 @JsonIgnore 147 public String getNotificationAddressesAsString() { 148 if (notificationAddresses != null) { 149 return notificationAddresses.stream() 150 .filter(Objects::nonNull) 151 .map(String::trim) 152 .collect(Collectors.joining(DELIMITER)); 153 } 154 return null; 155 } 156 157 /** 158 * Sets the notificationAddresses using a single String value that is split by ','. 159 */ 160 public void setNotificationAddressesAsString(String notificationAddressesAsString) { 161 if (notificationAddressesAsString != null) { 162 notificationAddresses = Stream.of(notificationAddressesAsString.split(DELIMITER)) 163 .filter(StringUtils::isNotEmpty) 164 .map(String::trim) 165 .collect(Collectors.toSet()); 166 } 167 } 168 169 @Nullable 170 public Boolean getSendNotification() { 171 return sendNotification; 172 } 173 174 /** 175 * This parameter determines if the requested download must be notified to the created once it's ready. 176 */ 177 public void setSendNotification(boolean sendNotification) { 178 this.sendNotification = sendNotification; 179 } 180 181 public DownloadFormat getFormat() { 182 return format; 183 } 184 185 /** 186 * This parameter determines the output format of the requested download. 187 */ 188 public void setFormat(DownloadFormat format) { 189 this.format = format; 190 } 191 192 public DownloadType getType() { 193 return type; 194 } 195 196 /** 197 * Download type: Occurrence or Event. 198 */ 199 public void setType(DownloadType type) { 200 this.type = type; 201 } 202 203 public String getDescription() { 204 return description; 205 } 206 207 public void setDescription(String description) { 208 this.description = description; 209 } 210 211 public JsonNode getMachineDescription() { 212 return machineDescription; 213 } 214 215 public void setMachineDescription(JsonNode machineDescription) { 216 this.machineDescription = machineDescription; 217 } 218 219 @JsonIgnore 220 public String getFileExtension() { 221 return format.getExtension(); 222 } 223 224 @Override 225 public boolean equals(Object o) { 226 if (this == o) { 227 return true; 228 } 229 if (o == null || getClass() != o.getClass()) { 230 return false; 231 } 232 DownloadRequest that = (DownloadRequest) o; 233 return sendNotification == that.sendNotification && 234 Objects.equals(creator, that.creator) && 235 Objects.equals(notificationAddresses, that.notificationAddresses) && 236 format == that.format && 237 type == that.type && 238 Objects.equals(description, that.description);// && 239// Objects.equals(machineDescription, that.machineDescription); 240 } 241 242 @Override 243 public int hashCode() { 244 return Objects.hash(creator, notificationAddresses, sendNotification, format, type, description, machineDescription); 245 } 246 247 @Override 248 public String toString() { 249 return new StringJoiner(", ", DownloadRequest.class.getSimpleName() + "[", "]") 250 .add("creator='" + creator + "'") 251 .add("notificationAddresses=" + notificationAddresses) 252 .add("sendNotification=" + sendNotification) 253 .add("format=" + format) 254 .add("type=" + type) 255 .add("description=" + description) 256 .add("machineDescription=" + machineDescription) 257 .toString(); 258 } 259}