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}