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 org.gbif.api.annotation.Experimental;
017import org.gbif.api.model.common.Identifier;
018import org.gbif.api.model.common.LinneanClassification;
019import org.gbif.api.model.common.LinneanClassificationKeys;
020import org.gbif.api.model.common.MediaObject;
021import org.gbif.api.util.ClassificationUtils;
022import org.gbif.api.util.IsoDateInterval;
023import org.gbif.api.vocabulary.BasisOfRecord;
024import org.gbif.api.vocabulary.Continent;
025import org.gbif.api.vocabulary.Country;
026import org.gbif.api.vocabulary.GbifRegion;
027import org.gbif.api.vocabulary.License;
028import org.gbif.api.vocabulary.OccurrenceIssue;
029import org.gbif.api.vocabulary.OccurrenceStatus;
030import org.gbif.api.vocabulary.Rank;
031import org.gbif.api.vocabulary.Sex;
032import org.gbif.api.vocabulary.TaxonomicStatus;
033import org.gbif.dwc.terms.DwcTerm;
034import org.gbif.dwc.terms.GbifTerm;
035import org.gbif.dwc.terms.Term;
036import org.gbif.dwc.terms.UnknownTerm;
037
038import java.lang.reflect.Field;
039import java.lang.reflect.Modifier;
040import java.net.URI;
041import java.util.ArrayList;
042import java.util.Arrays;
043import java.util.Collections;
044import java.util.Date;
045import java.util.EnumSet;
046import java.util.HashMap;
047import java.util.List;
048import java.util.Map;
049import java.util.Objects;
050import java.util.Optional;
051import java.util.Set;
052import java.util.stream.Collectors;
053import java.util.stream.Stream;
054
055import javax.annotation.Nullable;
056import javax.validation.constraints.Max;
057import javax.validation.constraints.Min;
058import javax.validation.constraints.NotNull;
059
060import com.fasterxml.jackson.annotation.JsonAnyGetter;
061import com.fasterxml.jackson.annotation.JsonIgnore;
062import com.fasterxml.jackson.annotation.JsonProperty;
063import com.fasterxml.jackson.databind.annotation.JsonDeserialize;
064import com.fasterxml.jackson.databind.annotation.JsonSerialize;
065
066import io.swagger.v3.oas.annotations.ExternalDocumentation;
067import io.swagger.v3.oas.annotations.media.Schema;
068import lombok.EqualsAndHashCode;
069import lombok.ToString;
070
071/**
072 * Represents an Occurrence as interpreted by GBIF, adding typed properties on top of the verbatim ones.
073 */
074@EqualsAndHashCode
075@ToString
076@SuppressWarnings("unused")
077public class Occurrence extends VerbatimOccurrence implements LinneanClassification, LinneanClassificationKeys {
078
079  public static final String GEO_DATUM = "WGS84";
080  // keep names of ALL properties of this class in a set for jackson serialization, see #properties()
081  private static final Set<String> PROPERTIES = Collections.unmodifiableSet(
082    Stream.concat(
083      // we need to these JSON properties manually because we have a fixed getter but no field for it
084      Stream.of(DwcTerm.geodeticDatum.simpleName(), "class", DwcTerm.countryCode.simpleName(), GbifTerm.gbifRegion.simpleName(), GbifTerm.publishedByGbifRegion.simpleName()),
085      Stream.concat(Arrays.stream(Occurrence.class.getDeclaredFields()),
086        Arrays.stream(VerbatimOccurrence.class.getDeclaredFields()))
087        .filter(field -> !Modifier.isStatic(field.getModifiers()))
088        .map(Field::getName)).collect(Collectors.toSet()));
089
090  // occurrence fields
091
092  // OpenAPI documentation comes from the enumeration.
093  private BasisOfRecord basisOfRecord;
094
095  @Schema(
096    description = "The number of individuals present at the time of the Occurrence.",
097    externalDocs = @ExternalDocumentation(
098      description = "Darwin Core definition",
099      url = "https://rs.tdwg.org/dwc/terms/individualCount"
100    )
101  )
102  private Integer individualCount;
103
104  // OpenAPI documentation comes from the enumeration.
105  private OccurrenceStatus occurrenceStatus;
106
107  // OpenAPI documentation comes from the enumeration.
108  private String sex;
109
110  @Schema(
111    description = "The age class or life stage of the Organism(s) at the time the Occurrence was recorded.\n\n" +
112      "Values are aligned to the [GBIF LifeStage vocabulary](https://registry.gbif.org/vocabulary/LifeStage/concepts)",
113    externalDocs = @ExternalDocumentation(
114      description = "Darwin Core definition",
115      url = "https://rs.tdwg.org/dwc/terms/lifeStage"
116    )
117  )
118  private String lifeStage;
119
120  @Schema(
121    description = "Statement about whether an organism or organisms have been introduced to a given place and time " +
122      "through the direct or indirect activity of modern humans.\n\n" +
123      "Values are aligned to the [GBIF EstablishmentMeans vocabulary](https://registry.gbif.org/vocabulary/EstablishmentMeans/concepts)," +
124      "which is derived from the [Darwin Core EstablishmentMeans vocabulary](https://dwc.tdwg.org/em/).",
125    externalDocs = @ExternalDocumentation(
126      description = "Darwin Core definition",
127      url = "https://rs.tdwg.org/dwc/terms/establishmentMeans"
128    )
129  )
130  private String establishmentMeans;
131
132  @Schema(
133    description = "The degree to which an Organism survives, reproduces, and expands its range at the given " +
134      "place and time.\n\n" +
135      "Values are aligned to the [GBIF DegreeOfEstablishment vocabulary](https://registry.gbif.org/vocabulary/DegreeOfEstablishment/concepts)," +
136      "which is derived from the [Darwin Core DegreeOfEstablishment vocabulary](https://dwc.tdwg.org/doe/).",
137    externalDocs = @ExternalDocumentation(
138      description = "Darwin Core definition",
139      url = "https://rs.tdwg.org/dwc/terms/degreeOfEstablishment"
140    )
141  )
142  private String degreeOfEstablishment;
143
144  @Schema(
145    description = "The process by which an Organism came to be in a given place at a given time.\n\n" +
146    "Values are aligned to the [GBIF Pathway vocabulary](https://registry.gbif.org/vocabulary/Pathway/concepts)," +
147    "which is derived from the [Darwin Core Pathway vocabulary](https://dwc.tdwg.org/pw/).",
148    externalDocs = @ExternalDocumentation(
149      description = "Darwin Core definition",
150      url = "https://rs.tdwg.org/dwc/terms/pathway"
151    )
152  )
153  private String pathway;
154
155  // taxonomy as NUB keys → LinneanClassificationKeys
156
157  @Schema(
158    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the most specific " +
159      "(lowest rank) taxon for this occurrence.  This could be a synonym, see `acceptedTaxonKey`.",
160    externalDocs = @ExternalDocumentation(
161      description = "Darwin Core definition",
162      url = "https://rs.tdwg.org/dwc/terms/"
163    )
164  )
165  private Integer taxonKey;
166
167  @Schema(
168    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the kingdom of this" +
169      "occurrence.",
170    externalDocs = @ExternalDocumentation(
171      description = "Darwin Core definition",
172      url = "https://rs.tdwg.org/dwc/terms/"
173    )
174  )
175  private Integer kingdomKey;
176
177  @Schema(
178    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the phylum of this" +
179      "occurrence.",
180    externalDocs = @ExternalDocumentation(
181      description = "Darwin Core definition",
182      url = "https://rs.tdwg.org/dwc/terms/"
183    )
184  )
185  private Integer phylumKey;
186
187  @Schema(
188    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the class of this" +
189      "occurrence.",
190    externalDocs = @ExternalDocumentation(
191      description = "Darwin Core definition",
192      url = "https://rs.tdwg.org/dwc/terms/"
193    )
194  )
195  private Integer classKey;
196
197  @Schema(
198    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the order of this" +
199      "occurrence.",
200    externalDocs = @ExternalDocumentation(
201      description = "Darwin Core definition",
202      url = "https://rs.tdwg.org/dwc/terms/"
203    )
204  )
205  private Integer orderKey;
206
207  @Schema(
208    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the family of this" +
209      "occurrence.",
210    externalDocs = @ExternalDocumentation(
211      description = "Darwin Core definition",
212      url = "https://rs.tdwg.org/dwc/terms/"
213    )
214  )
215  private Integer familyKey;
216
217  @Schema(
218    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the genus of this" +
219      "occurrence.",
220    externalDocs = @ExternalDocumentation(
221      description = "Darwin Core definition",
222      url = "https://rs.tdwg.org/dwc/terms/"
223    )
224  )
225  private Integer genusKey;
226
227  @Schema(
228    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the subgenus of this" +
229      "occurrence.",
230    externalDocs = @ExternalDocumentation(
231      description = "Darwin Core definition",
232      url = "https://rs.tdwg.org/dwc/terms/"
233    )
234  )
235  private Integer subgenusKey;
236
237  @Schema(
238    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the species of this" +
239      "occurrence.",
240    externalDocs = @ExternalDocumentation(
241      description = "Darwin Core definition",
242      url = "https://rs.tdwg.org/dwc/terms/"
243    )
244  )
245  private Integer speciesKey;
246
247  @Schema(
248    description = "A taxon key from the [GBIF backbone](https://doi.org/10.15468/39omei) for the accepted taxon of " +
249      "this occurrence.",
250    externalDocs = @ExternalDocumentation(
251      description = "Darwin Core definition",
252      url = "https://rs.tdwg.org/dwc/terms/"
253    )
254  )
255  private Integer acceptedTaxonKey;
256
257  // taxonomy as name strings → LinneanClassification
258
259  @Schema(
260    description = "The scientific name (including authorship) for the taxon from the " +
261      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.  This could be a synonym, see " +
262      "also `acceptedScientificName`.",
263    externalDocs = @ExternalDocumentation(
264      description = "Darwin Core definition",
265      url = "https://rs.tdwg.org/dwc/terms/"
266    )
267  )
268  private String scientificName;  // the interpreted name matching taxonKey
269
270  @Schema(
271    description = "The accepted scientific name (including authorship) for the taxon from the " +
272      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
273    externalDocs = @ExternalDocumentation(
274      description = "Darwin Core definition",
275      url = "https://rs.tdwg.org/dwc/terms/"
276    )
277  )
278  private String acceptedScientificName;
279
280  @Schema(
281    description = "The kingdom name (excluding authorship) for the kingdom from the " +
282      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
283    externalDocs = @ExternalDocumentation(
284      description = "Darwin Core definition",
285      url = "https://rs.tdwg.org/dwc/terms/"
286    )
287  )
288  private String kingdom;
289
290  @Schema(
291    description = "The phylum name (excluding authorship) for the phylum from the " +
292      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
293    externalDocs = @ExternalDocumentation(
294      description = "Darwin Core definition",
295      url = "https://rs.tdwg.org/dwc/terms/"
296    )
297  )
298  private String phylum;
299
300  @Schema(
301    description = "The class name (excluding authorship) for the class from the " +
302      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
303    externalDocs = @ExternalDocumentation(
304      description = "Darwin Core definition",
305      url = "https://rs.tdwg.org/dwc/terms/"
306    )
307  )
308  @JsonProperty("class")
309  private String clazz;
310
311  @Schema(
312    description = "The order name (excluding authorship) for the order from the " +
313      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
314    externalDocs = @ExternalDocumentation(
315      description = "Darwin Core definition",
316      url = "https://rs.tdwg.org/dwc/terms/"
317    )
318  )
319  private String order;
320
321  @Schema(
322    description = "The family name (excluding authorship) for the family from the " +
323      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
324    externalDocs = @ExternalDocumentation(
325      description = "Darwin Core definition",
326      url = "https://rs.tdwg.org/dwc/terms/"
327    )
328  )
329  private String family;
330
331  @Schema(
332    description = "The genus name (excluding authorship) for the genus from the " +
333      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
334    externalDocs = @ExternalDocumentation(
335      description = "Darwin Core definition",
336      url = "https://rs.tdwg.org/dwc/terms/"
337    )
338  )
339  private String genus;
340
341  @Schema(
342    description = "The subgenus name (excluding authorship) for the subgenus from the " +
343      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
344    externalDocs = @ExternalDocumentation(
345      description = "Darwin Core definition",
346      url = "https://rs.tdwg.org/dwc/terms/"
347    )
348  )
349  private String subgenus;
350
351  @Schema(
352    description = "The species name (excluding authorship) for the species from the " +
353      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
354    externalDocs = @ExternalDocumentation(
355      description = "Darwin Core definition",
356      url = "https://rs.tdwg.org/dwc/terms/"
357    )
358  )
359  private String species;
360
361  // atomised scientific name
362
363  @Schema(
364    description = "The genus name part of the species name from the " +
365      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
366    externalDocs = @ExternalDocumentation(
367      description = "Darwin Core definition",
368      url = "https://rs.tdwg.org/dwc/terms/genericName"
369    )
370  )
371  private String genericName;
372
373  @Schema(
374    description = "The specific name part of the species name from the " +
375      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
376    externalDocs = @ExternalDocumentation(
377      description = "Darwin Core definition",
378      url = "https://rs.tdwg.org/dwc/terms/specificEpithet"
379    )
380  )
381  private String specificEpithet;
382
383  @Schema(
384    description = "The infraspecific name part of the species name from the " +
385      "[GBIF backbone](https://doi.org/10.15468/39omei) matched to this occurrence.",
386    externalDocs = @ExternalDocumentation(
387      description = "Darwin Core definition",
388      url = "https://rs.tdwg.org/dwc/terms/infraspecificEpithet"
389    )
390  )
391  private String infraspecificEpithet;
392
393  @Schema(
394    description = "The taxonomic rank of the most specific name in the scientificName.",
395    externalDocs = @ExternalDocumentation(
396      description = "Darwin Core definition",
397      url = "https://rs.tdwg.org/dwc/terms/taxonRank"
398    )
399  )
400  private Rank taxonRank;
401
402  @Schema(
403    description = "The status of the use of the scientificName as a label for a taxon.",
404    externalDocs = @ExternalDocumentation(
405      description = "Darwin Core definition",
406      url = "https://rs.tdwg.org/dwc/terms/taxonomicStatus"
407    )
408  )
409  private TaxonomicStatus taxonomicStatus;
410
411  @Schema(
412    description = "The IUCN Red List Category of the taxon of this occurrence.\n\n" +
413      "See the [GBIF vocabulary](https://rs.gbif.org/vocabulary/iucn/threat_status/) for the values and their " +
414      "definitions, and the [IUCN Red List of Threatened Species dataset in GBIF](https://doi.org/10.15468/0qnb58) " +
415      "for the version of the Red List GBIF's interpretation procedures are using.",
416    externalDocs = @ExternalDocumentation(
417      description = "GBIF vocabulary",
418      url = "https://rs.gbif.org/vocabulary/iucn/threat_status/"
419    )
420  )
421  private String iucnRedListCategory;
422
423  // identification
424
425  @Schema(
426    description = "The date on which the subject was determined as representing the Taxon.",
427    externalDocs = @ExternalDocumentation(
428      description = "Darwin Core definition",
429      url = "https://rs.tdwg.org/dwc/terms/dateIdentified"
430    )
431  )
432  private Date dateIdentified;
433
434  // location
435
436  @Schema(
437    description = "The geographic latitude (in decimal degrees, using the WGS84 datum) of the geographic centre " +
438      "of the location of the occurrence.",
439    externalDocs = @ExternalDocumentation(
440      description = "Darwin Core definition",
441      url = "https://rs.tdwg.org/dwc/terms/decimalLatitude"
442    )
443  )
444  private Double decimalLatitude;
445
446  @Schema(
447    description = "The geographic longitude (in decimal degrees, using the WGS84 datum) of the geographic centre " +
448      "of the location of the occurrence.",
449    externalDocs = @ExternalDocumentation(
450      description = "Darwin Core definition",
451      url = "https://rs.tdwg.org/dwc/terms/decimalLongitude"
452    )
453  )
454  private Double decimalLongitude;
455
456  //coordinatePrecision and coordinateUncertaintyInMeters should be BigDecimal see POR-2795
457
458  @Schema(
459    description = "A decimal representation of the precision of the coordinates given in the decimalLatitude and decimalLongitude.",
460    externalDocs = @ExternalDocumentation(
461      description = "Darwin Core definition",
462      url = "https://rs.tdwg.org/dwc/terms/coordinatePrecision"
463    )
464  )
465  private Double coordinatePrecision;
466
467  @Schema(
468    description = "The horizontal distance (in metres) from the given decimalLatitude and decimalLongitude " +
469      "describing the smallest circle containing the whole of the Location.",
470    externalDocs = @ExternalDocumentation(
471      description = "Darwin Core definition",
472      url = "https://rs.tdwg.org/dwc/terms/coordinateUncertaintyInMeters"
473    )
474  )
475  private Double coordinateUncertaintyInMeters;
476
477  @Schema(
478    description = "**Deprecated.**  This value is always null.  It is an obsolete Darwin Core term.",
479    externalDocs = @ExternalDocumentation(
480      description = "Darwin Core definition",
481      url = "https://rs.tdwg.org/dwc/terms/"
482    )
483  )
484  @Deprecated //see getter
485  private Double coordinateAccuracy;
486
487  @Schema(
488    description = "Elevation (altitude) in metres above sea level.  This is not a current Darwin Core term."
489  )
490  private Double elevation;
491
492  @Schema(
493    description = "The value of the potential error associated with the elevation.  This is not a current Darwin Core term."
494  )
495  private Double elevationAccuracy;
496
497  @Schema(
498    description = "Depth in metres below sea level.  This is not a current Darwin Core term."
499  )
500  private Double depth;
501
502  @Schema(
503    description = "The value of the potential error associated with the depth.  This is not a current Darwin Core term."
504  )
505  private Double depthAccuracy;
506
507  // OpenAPI documentation from enumeration
508  private Continent continent;
509
510  @JsonSerialize(using = Country.IsoSerializer.class)
511  @JsonDeserialize(using = Country.IsoDeserializer.class)
512  private Country country;
513
514  @Schema(
515    description = "The name of the next-smaller administrative region than country (state, province, canton, " +
516      "department, region, etc.) in which the occurrence occurs.\n\n" +
517      "This value is unaltered by GBIF's processing; see also the GADM fields.",
518    externalDocs = @ExternalDocumentation(
519      description = "Darwin Core definition",
520      url = "https://rs.tdwg.org/dwc/terms/stateProvince"
521    )
522  )
523  private String stateProvince;
524
525  @Schema(
526    description = "The administrative divisions according to the [GADM database](https://gadm.org/).\n\n" +
527      "This value is applied by GBIF's processing without consideration of the `stateProvince`, `county` or `locality` fields."
528  )
529  private Gadm gadm = new Gadm();
530
531  @Schema(
532    description = "The name of the water body in which the Location occurs.",
533    externalDocs = @ExternalDocumentation(
534      description = "Darwin Core definition",
535      url = "https://rs.tdwg.org/dwc/terms/waterBody"
536    )
537  )
538  private String waterBody;
539
540  @Schema(
541    description = "The distance in metres of the occurrence from a centroid known to be applied to occurrences " +
542      "during georeferencing.  This can potentially indicate low-precision georeferencing, check the values of " +
543      "`coordinateUncertaintyInMeters` and `georeferenceRemarks`."
544  )
545  private Double distanceFromCentroidInMeters;
546
547  @Schema(
548    description = "A list (concatenated and separated) of geographic names less specific than the information captured in the dwc:locality term.",
549    externalDocs = @ExternalDocumentation(
550      description = "Darwin Core definition",
551      url = "https://rs.tdwg.org/dwc/terms/higherGeography"
552    )
553  )
554  private String higherGeography;
555
556  @Schema(
557    description = "A list (concatenated and separated) of names of people, groups, or organizations who determined the georeference (spatial representation) for the dcterms:Location.",
558    externalDocs = @ExternalDocumentation(
559      description = "Darwin Core definition",
560      url = "https://rs.tdwg.org/dwc/terms/georeferencedBy"
561    )
562  )
563  private String georeferencedBy;
564
565  // recording event
566
567  @Schema(
568    description = "The four-digit year in which the event occurred, according to the Common Era calendar.",
569    externalDocs = @ExternalDocumentation(
570      description = "Darwin Core definition",
571      url = "https://rs.tdwg.org/dwc/terms/year"
572    )
573  )
574  private Integer year;
575
576  @Schema(
577    description = "The integer month in which the Event occurred.",
578    externalDocs = @ExternalDocumentation(
579      description = "Darwin Core definition",
580      url = "https://rs.tdwg.org/dwc/terms/month"
581    )
582  )
583  private Integer month;
584
585  @Schema(
586    description = "The integer day of the month on which the Event occurred.",
587    externalDocs = @ExternalDocumentation(
588      description = "Darwin Core definition",
589      url = "https://rs.tdwg.org/dwc/terms/day"
590    )
591  )
592  private Integer day;
593
594  @Schema(
595    description = "The date-time during which an Event occurred. For occurrences, this is the date-time when the " +
596      "event was recorded. Not suitable for a time in a geological context.",
597    externalDocs = @ExternalDocumentation(
598      description = "Darwin Core definition",
599      url = "https://rs.tdwg.org/dwc/terms/eventDate"
600    )
601  )
602  private IsoDateInterval eventDate;
603
604  @Schema(
605    description = "The latest integer day of the year on which the Event occurred (1 for 1 January, 365 for " +
606      "31 December, except in a leap year, in which case it is 366).",
607    externalDocs = @ExternalDocumentation(
608      description = "Darwin Core definition",
609      url = "https://rs.tdwg.org/dwc/terms/endDayOfYear"
610    )
611  )
612  private Integer startDayOfYear;
613
614  @Schema(
615    description = "The earliest integer day of the year on which the Event occurred (1 for 1 January, 365 for " +
616      "31 December, except in a leap year, in which case it is 366).",
617    externalDocs = @ExternalDocumentation(
618      description = "Darwin Core definition",
619      url = "https://rs.tdwg.org/dwc/terms/startDayOfYear"
620    )
621  )
622  private Integer endDayOfYear;
623
624  @Schema(
625    description = "A list (concatenated and separated) of nomenclatural types (type status, typified scientific name, " +
626      "publication) applied to the occurrence.",
627    externalDocs = @ExternalDocumentation(
628      description = "Darwin Core definition",
629      url = "https://rs.tdwg.org/dwc/terms/typeStatus"
630    )
631  )
632  private String typeStatus;
633
634  // extracted from type status, but we should propose a new dwc term for this!
635  // for example: "Paratype of Taeniopteryx metequi Ricker & Ross" is status=Paratype, typifiedName=Taeniopteryx metequi Ricker & Ross
636  @Schema(
637    description = "The scientific name that is based on the type specimen.\n\n" +
638      "This is not yet a Darwin Core term, see the [proposal to add it](https://github.com/tdwg/dwc/issues/28)."
639  )
640  private String typifiedName; // missing from DwC
641
642  @Schema(
643    description = "A specific interpretation issue found during processing and interpretation of the record.\n\n" +
644      "See the https://links.gbif.org/occurrence-issues[list of occurrence issues] and the " +
645      "https://gbif.github.io/gbif-api/apidocs/org/gbif/api/vocabulary/OccurrenceIssue.html[OccurrenceIssue enumeration] " +
646      "for possible values and definitions."
647  )
648  private Set<OccurrenceIssue> issues = EnumSet.noneOf(OccurrenceIssue.class);
649
650  // record level
651
652  @Schema(
653    description = "The most recent date-time on which the occurrence was changed, according to the publisher.",
654    externalDocs = @ExternalDocumentation(
655      description = "Dublin Core definition",
656      url = "https://purl.org/dc/terms/modified"
657    )
658  )
659  private Date modified;  // interpreted dc:modified, i.e. date changed in source
660
661  @Schema(
662    description = "The time this occurrence was last processed by GBIF's interpretation system “Pipelines”.\n\n" +
663      "This is the time the record was last changed in GBIF, **not** the time the record was last changed by the " +
664      "publisher.  Data is also reprocessed when we changed the taxonomic backbone, geographic data sources or " +
665      "other interpretation procedures.\n\n" +
666      "An earlier interpretation system distinguished between “parsing” and “interpretation”, but in the current " +
667      "system there is only one process — the two dates will always be the same."
668  )
669  private Date lastInterpreted;
670
671  @Schema(
672    description = "A related resource that is referenced, cited, or otherwise pointed to by the described resource.",
673    externalDocs = @ExternalDocumentation(
674      description = "Dublin Core definition",
675      url = "https://purl.org/dc/terms/references"
676    )
677  )
678  private URI references;
679
680  @Schema(
681    description = "A legal document giving official permission to do something with the occurrence.",
682    externalDocs = @ExternalDocumentation(
683      description = "Dublin Core definition",
684      url = "https://purl.org/dc/terms/license"
685    )
686  )
687  private License license;
688
689  @Schema(
690    description = "A number or enumeration value for the quantity of organisms.",
691    externalDocs = @ExternalDocumentation(
692      description = "Darwin Core definition",
693      url = "https://rs.tdwg.org/dwc/terms/organismQuantity"
694    )
695  )
696  private Double organismQuantity;
697
698  @Schema(
699    description = "The type of quantification system used for the quantity of organisms.",
700    externalDocs = @ExternalDocumentation(
701      description = "Darwin Core definition",
702      url = "https://rs.tdwg.org/dwc/terms/organismQuantityType"
703    )
704  )
705  private String organismQuantityType;
706
707  @Schema(
708    description = "The unit of measurement of the size (time duration, length, area, or volume) of a sample in a sampling event.",
709    externalDocs = @ExternalDocumentation(
710      description = "Darwin Core definition",
711      url = "https://rs.tdwg.org/dwc/terms/sampleSizeUnit"
712    )
713  )
714  private String sampleSizeUnit;
715
716  @Schema(
717    description = "A numeric value for a measurement of the size (time duration, length, area, or volume) of a sample in a sampling event.",
718    externalDocs = @ExternalDocumentation(
719      description = "Darwin Core definition",
720      url = "https://rs.tdwg.org/dwc/terms/sampleSizeValue"
721    )
722  )
723  private Double sampleSizeValue;
724
725  @Schema(
726    description = "The relative measurement of the quantity of the organism (i.e. without absolute units)."
727  )
728  private Double relativeOrganismQuantity;
729
730  @Schema(
731    description = "Flag occurrence when associatedSequences/extension exists"
732  )
733  private boolean isSequenced;
734
735  @Schema(
736      description = "A list (concatenated and separated) of identifiers (publication, global unique identifier, URI) "
737        + "of genetic sequence information associated with the material entity.",
738      externalDocs =
739          @ExternalDocumentation(
740              description = "Darwin Core definition",
741              url = "https://rs.tdwg.org/dwc/terms/associatedSequences"))
742  private String associatedSequences;
743
744  // interpreted extension data
745
746  @Schema(
747    description = "Alternative identifiers for the occurrence.",
748    externalDocs = @ExternalDocumentation(
749      description = "GBIF Alternative Identifiers extension",
750      url = "https://rs.gbif.org/terms/1.0/Identifier"
751    )
752  )
753  private List<Identifier> identifiers = new ArrayList<>();
754
755  @Schema(
756    description = "Multimedia related to te occurrence.",
757    externalDocs = @ExternalDocumentation(
758      description = "GBIF Multimedia extension",
759      url = "https://rs.gbif.org/terms/1.0/Multimedia"
760    )
761  )
762  private List<MediaObject> media = new ArrayList<>();
763
764  @Schema(
765    description = "Measurements or facts about the the occurrence.",
766    externalDocs = @ExternalDocumentation(
767      description = "Darwin Core definition",
768      url = "https://rs.tdwg.org/dwc/terms/MeasurementOrFact"
769    )
770  )
771  private List<MeasurementOrFact> facts = new ArrayList<>();
772
773  @Schema(
774    description = "Relationships between occurrences.",
775    externalDocs = @ExternalDocumentation(
776      description = "Darwin Core definition",
777      url = "https://rs.tdwg.org/dwc/terms/ResourceRelationship"
778    )
779  )
780  private List<OccurrenceRelation> relations = new ArrayList<>();
781
782  @Schema(
783    description = "A list of the globally unique identifiers for the person, people, groups, or organizations " +
784      "responsible for recording the original Occurrence.",
785    externalDocs = @ExternalDocumentation(
786      description = "Darwin Core definition",
787      url = "https://rs.tdwg.org/dwc/terms/recordedByID"
788    )
789  )
790  @JsonProperty("recordedByIDs")
791  private List<AgentIdentifier> recordedByIds = new ArrayList<>();
792
793  @Schema(
794    description = "A list of the globally unique identifiers for the person, people, groups, or organizations " +
795      "responsible for assigning the Taxon to the occurrence.",
796    externalDocs = @ExternalDocumentation(
797      description = "Darwin Core definition",
798      url = "https://rs.tdwg.org/dwc/terms/identifiedByID"
799    )
800  )
801  @JsonProperty("identifiedByIDs")
802  private List<AgentIdentifier> identifiedByIds = new ArrayList<>();
803
804  @Schema(
805    description = "**Experimental.**  The UUID of the institution holding the specimen occurrence, from GRSciColl."
806  )
807  @Experimental
808  private String institutionKey;
809
810  @Schema(
811    description = "**Experimental.**  The UUID of the collection containing the specimen occurrence, from GRSciColl."
812  )
813  @Experimental
814  private String collectionKey;
815
816  @Schema(
817    description = "**Experimental.**  Whether the occurrence belongs to a machine-calculated cluster of probable " +
818      "duplicate occurrences.",
819    externalDocs = @ExternalDocumentation(
820      description = "GBIF Data Blog",
821      url = "https://data-blog.gbif.org/post/clustering-occurrences/"
822    )
823  )
824  @Experimental
825  private boolean isInCluster;
826
827  @Schema(
828    description = "An identifier for the set of data. May be a global unique identifier or an identifier specific to " +
829      "a collection or institution.",
830    externalDocs = @ExternalDocumentation(
831      description = "Darwin Core definition",
832      url = "https://rs.tdwg.org/dwc/terms/datasetID"
833    )
834  )
835  private String datasetID;
836
837  @Schema(
838    description = "The name identifying the data set from which the record was derived.",
839    externalDocs = @ExternalDocumentation(
840      description = "Darwin Core definition",
841      url = "https://rs.tdwg.org/dwc/terms/"
842    )
843  )
844  private String datasetName;
845
846  @Schema(
847    description = "A list (concatenated and separated) of previous or alternate fully qualified catalogue numbers " +
848      "or other human-used identifiers for the same occurrence, whether in the current or any other data set or collection.",
849    externalDocs = @ExternalDocumentation(
850      description = "Darwin Core definition",
851      url = "https://rs.tdwg.org/dwc/terms/otherCatalogNumbers"
852    )
853  )
854  private String otherCatalogNumbers;
855
856  @Schema(
857    description = "The full name of the earliest possible geochronologic eon or lowest chrono-stratigraphic eonothem or the informal name (\"Precambrian\") attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
858    externalDocs = @ExternalDocumentation(
859      description = "Darwin Core definition",
860      url = "https://rs.tdwg.org/dwc/terms/earliestEonOrLowestEonothem"
861    )
862  )
863  private String earliestEonOrLowestEonothem;
864
865  @Schema(
866    description = "The full name of the latest possible geochronologic eon or highest chrono-stratigraphic eonothem or the informal name (\"Precambrian\") attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
867    externalDocs = @ExternalDocumentation(
868      description = "Darwin Core definition",
869      url = "https://rs.tdwg.org/dwc/terms/latestEonOrHighestEonothem"
870    )
871  )
872  private String latestEonOrHighestEonothem;
873
874  @Schema(
875    description = "The full name of the earliest possible geochronologic era or lowest chronostratigraphic erathem attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
876    externalDocs = @ExternalDocumentation(
877      description = "Darwin Core definition",
878      url = "https://rs.tdwg.org/dwc/terms/earliestEraOrLowestErathem"
879    )
880  )
881  private String earliestEraOrLowestErathem;
882
883  @Schema(
884    description = "The full name of the latest possible geochronologic era or highest chronostratigraphic erathem attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
885    externalDocs = @ExternalDocumentation(
886      description = "Darwin Core definition",
887      url = "https://rs.tdwg.org/dwc/terms/latestEraOrHighestErathem"
888    )
889  )
890  private String latestEraOrHighestErathem;
891
892  @Schema(
893    description = "The full name of the earliest possible geochronologic period or lowest chronostratigraphic system attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
894    externalDocs = @ExternalDocumentation(
895      description = "Darwin Core definition",
896      url = "https://rs.tdwg.org/dwc/terms/earliestPeriodOrLowestSystem"
897    )
898  )
899  private String earliestPeriodOrLowestSystem;
900
901  @Schema(
902    description = "The full name of the latest possible geochronologic period or highest chronostratigraphic system attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
903    externalDocs = @ExternalDocumentation(
904      description = "Darwin Core definition",
905      url = "https://rs.tdwg.org/dwc/terms/latestPeriodOrHighestSystem"
906    )
907  )
908  private String latestPeriodOrHighestSystem;
909
910  @Schema(
911    description = "The full name of the earliest possible geochronologic epoch or lowest chronostratigraphic series attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
912    externalDocs = @ExternalDocumentation(
913      description = "Darwin Core definition",
914      url = "https://rs.tdwg.org/dwc/terms/earliestEpochOrLowestSeries"
915    )
916  )
917  private String earliestEpochOrLowestSeries;
918
919  @Schema(
920    description = "The full name of the latest possible geochronologic epoch or highest chronostratigraphic series attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
921    externalDocs = @ExternalDocumentation(
922      description = "Darwin Core definition",
923      url = "https://rs.tdwg.org/dwc/terms/latestEpochOrHighestSeries"
924    )
925  )
926  private String latestEpochOrHighestSeries;
927
928  @Schema(
929    description = "The full name of the earliest possible geochronologic age or lowest chronostratigraphic stage attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
930    externalDocs = @ExternalDocumentation(
931      description = "Darwin Core definition",
932      url = "https://rs.tdwg.org/dwc/terms/earliestAgeOrLowestStage"
933    )
934  )
935  private String earliestAgeOrLowestStage;
936
937  @Schema(
938    description = "The full name of the latest possible geochronologic age or highest chronostratigraphic stage attributable to the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
939    externalDocs = @ExternalDocumentation(
940      description = "Darwin Core definition",
941      url = "https://rs.tdwg.org/dwc/terms/latestAgeOrHighestStage"
942    )
943  )
944  private String latestAgeOrHighestStage;
945
946  @Schema(
947    description = "The full name of the lowest possible geological biostratigraphic zone of the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
948    externalDocs = @ExternalDocumentation(
949      description = "Darwin Core definition",
950      url = "https://rs.tdwg.org/dwc/terms/lowestBiostratigraphicZone"
951    )
952  )
953  private String lowestBiostratigraphicZone;
954
955  @Schema(
956    description = "The full name of the highest possible geological biostratigraphic zone of the stratigraphic horizon from which the dwc:MaterialEntity was collected.",
957    externalDocs = @ExternalDocumentation(
958      description = "Darwin Core definition",
959      url = "https://rs.tdwg.org/dwc/terms/highestBiostratigraphicZone"
960    )
961  )
962  private String highestBiostratigraphicZone;
963
964  @Schema(
965    description = "The full name of the lithostratigraphic group from which the dwc:MaterialEntity was collected.",
966    externalDocs = @ExternalDocumentation(
967      description = "Darwin Core definition",
968      url = "https://rs.tdwg.org/dwc/terms/group"
969    )
970  )
971  private String group;
972
973  @Schema(
974    description = "The full name of the lithostratigraphic formation from which the dwc:MaterialEntity was collected.",
975    externalDocs = @ExternalDocumentation(
976      description = "Darwin Core definition",
977      url = "https://rs.tdwg.org/dwc/terms/formation"
978    )
979  )
980  private String formation;
981
982  @Schema(
983    description = "The full name of the lithostratigraphic member from which the dwc:MaterialEntity was collected.",
984    externalDocs = @ExternalDocumentation(
985      description = "Darwin Core definition",
986      url = "https://rs.tdwg.org/dwc/terms/member"
987    )
988  )
989  private String member;
990
991  @Schema(
992    description = "The full name of the lithostratigraphic bed from which the dwc:MaterialEntity was collected.",
993    externalDocs = @ExternalDocumentation(
994      description = "Darwin Core definition",
995      url = "https://rs.tdwg.org/dwc/terms/bed"
996    )
997  )
998  private String bed;
999
1000
1001  @Schema(
1002    description = "A person, group, or organization responsible for recording the original occurrence.",
1003    externalDocs = @ExternalDocumentation(
1004      description = "Darwin Core definition",
1005      url = "https://rs.tdwg.org/dwc/terms/recordedBy"
1006    )
1007  )
1008  private String recordedBy;
1009
1010  @Schema(
1011    description = "A list (concatenated and separated) of names of people, groups, or organizations who assigned the " +
1012      "Taxon to the occurrence.",
1013    externalDocs = @ExternalDocumentation(
1014      description = "Darwin Core definition",
1015      url = "https://rs.tdwg.org/dwc/terms/identifiedBy"
1016    )
1017  )
1018  private String identifiedBy;
1019
1020  @Schema(
1021    description = "A preparation or preservation method for a specimen.",
1022    externalDocs = @ExternalDocumentation(
1023      description = "Darwin Core definition",
1024      url = "https://rs.tdwg.org/dwc/terms/preparations"
1025    )
1026  )
1027  private String preparations;
1028
1029  @Schema(
1030    description = "The methods or protocols used during an Event, denoted by an IRI.",
1031    externalDocs = @ExternalDocumentation(
1032      description = "Darwin Core definition",
1033      url = "https://rs.tdwg.org/dwc/terms/samplingProtocol"
1034    )
1035  )
1036  private String samplingProtocol;
1037
1038  @Schema(
1039    description = "**Experimental.**  The DNA sequence ID of an occurrence."
1040  )
1041  @Experimental
1042  private List<String> dnaSequenceID = new ArrayList<>();
1043
1044  public Occurrence() {
1045
1046  }
1047
1048  /**
1049   * Create occurrence instance from existing verbatim one, copying over all data.
1050   */
1051  public Occurrence(@Nullable VerbatimOccurrence verbatim) {
1052    if (verbatim != null) {
1053      setKey(verbatim.getKey());
1054      setDatasetKey(verbatim.getDatasetKey());
1055      setPublishingOrgKey(verbatim.getPublishingOrgKey());
1056      setPublishingCountry(verbatim.getPublishingCountry());
1057      setProtocol(verbatim.getProtocol());
1058      setCrawlId(verbatim.getCrawlId());
1059      if (verbatim.getLastCrawled() != null) {
1060        setLastCrawled(new Date(verbatim.getLastCrawled().getTime()));
1061      }
1062      if (verbatim.getVerbatimFields() != null) {
1063        getVerbatimFields().putAll(verbatim.getVerbatimFields());
1064      }
1065      if (verbatim.getLastParsed() != null) {
1066        setLastParsed(verbatim.getLastParsed());
1067      }
1068      setExtensions(verbatim.getExtensions());
1069    }
1070  }
1071
1072  @Nullable
1073  public BasisOfRecord getBasisOfRecord() {
1074    return basisOfRecord;
1075  }
1076
1077  public void setBasisOfRecord(BasisOfRecord basisOfRecord) {
1078    this.basisOfRecord = basisOfRecord;
1079  }
1080
1081  @Nullable
1082  public Integer getIndividualCount() {
1083    return individualCount;
1084  }
1085
1086  public void setIndividualCount(Integer individualCount) {
1087    this.individualCount = individualCount;
1088  }
1089
1090  @Nullable
1091  public OccurrenceStatus getOccurrenceStatus() {
1092    return occurrenceStatus;
1093  }
1094
1095  public void setOccurrenceStatus(OccurrenceStatus occurrenceStatus) {
1096    this.occurrenceStatus = occurrenceStatus;
1097  }
1098
1099  @Nullable
1100  public String getSex() {
1101    return sex;
1102  }
1103
1104  public void setSex(String sex) {
1105    this.sex = sex;
1106  }
1107
1108  @Nullable
1109  public String getLifeStage() {
1110    return lifeStage;
1111  }
1112
1113  public void setLifeStage(String lifeStage) {
1114    this.lifeStage = lifeStage;
1115  }
1116
1117  @Nullable
1118  public String getEstablishmentMeans() {
1119    return establishmentMeans;
1120  }
1121
1122  public void setEstablishmentMeans(String establishmentMeans) {
1123    this.establishmentMeans = establishmentMeans;
1124  }
1125
1126  /**
1127   * The best matching, accepted GBIF backbone name usage representing this occurrence.
1128   * In case the verbatim scientific name and its classification can only be matched to a higher rank this will
1129   * represent the lowest matching rank. In the worst case this could just be for example Animalia.
1130   */
1131  @Nullable
1132  public Integer getTaxonKey() {
1133    return taxonKey;
1134  }
1135
1136  public void setTaxonKey(Integer taxonKey) {
1137    this.taxonKey = taxonKey;
1138  }
1139
1140  @Nullable
1141  @Override
1142  public Integer getKingdomKey() {
1143    return kingdomKey;
1144  }
1145
1146  @Override
1147  public void setKingdomKey(@Nullable Integer kingdomKey) {
1148    this.kingdomKey = kingdomKey;
1149  }
1150
1151  @Nullable
1152  @Override
1153  public Integer getPhylumKey() {
1154    return phylumKey;
1155  }
1156
1157  @Override
1158  public void setPhylumKey(@Nullable Integer phylumKey) {
1159    this.phylumKey = phylumKey;
1160  }
1161
1162  @Nullable
1163  @Override
1164  public Integer getClassKey() {
1165    return classKey;
1166  }
1167
1168  @Override
1169  public void setClassKey(@Nullable Integer classKey) {
1170    this.classKey = classKey;
1171  }
1172
1173  @Nullable
1174  @Override
1175  public Integer getOrderKey() {
1176    return orderKey;
1177  }
1178
1179  @Override
1180  public void setOrderKey(@Nullable Integer orderKey) {
1181    this.orderKey = orderKey;
1182  }
1183
1184  @Nullable
1185  @Override
1186  public Integer getFamilyKey() {
1187    return familyKey;
1188  }
1189
1190  @Override
1191  public void setFamilyKey(@Nullable Integer familyKey) {
1192    this.familyKey = familyKey;
1193  }
1194
1195  @Nullable
1196  @Override
1197  public Integer getGenusKey() {
1198    return genusKey;
1199  }
1200
1201  @Override
1202  public void setGenusKey(@Nullable Integer genusKey) {
1203    this.genusKey = genusKey;
1204  }
1205
1206  @Nullable
1207  @Override
1208  public Integer getSubgenusKey() {
1209    return subgenusKey;
1210  }
1211
1212  @Override
1213  public void setSubgenusKey(@Nullable Integer subgenusKey) {
1214    this.subgenusKey = subgenusKey;
1215  }
1216
1217  @Nullable
1218  @Override
1219  public Integer getHigherRankKey(Rank rank) {
1220    return ClassificationUtils.getHigherRankKey(this, rank);
1221  }
1222
1223  /**
1224   * An ordered map with entries for all higher Linnean ranks excluding the taxonKey itself.
1225   * The map starts with the highest rank, e.g. the kingdom and maps the name usage key to its canonical name.
1226   *
1227   * @return map of higher ranks
1228   */
1229  @NotNull
1230  @JsonIgnore
1231  public Map<Integer, String> getHigherClassificationMap() {
1232    return taxonKey == null ? ClassificationUtils.getHigherClassificationMap(this)
1233      : ClassificationUtils.getHigherClassificationMap(this, taxonKey, null, null);
1234  }
1235
1236  /**
1237   * The accepted species for this occurrence. In case the taxonKey is of a higher rank than species (e.g. genus)
1238   * speciesKey is null. In case taxonKey represents an infraspecific taxon the speciesKey points to the species
1239   * the infraspecies is classified as. In case of taxonKey being a species the speciesKey is the same.
1240   */
1241  @Nullable
1242  @Override
1243  public Integer getSpeciesKey() {
1244    return speciesKey;
1245  }
1246
1247  @Override
1248  public void setSpeciesKey(@Nullable Integer speciesKey) {
1249    this.speciesKey = speciesKey;
1250  }
1251
1252  /**
1253   * The accepted taxon key from the GBIF backbone.
1254   */
1255  @Nullable
1256  public Integer getAcceptedTaxonKey() {
1257    return acceptedTaxonKey;
1258  }
1259
1260  public void setAcceptedTaxonKey(Integer acceptedTaxonKey) {
1261    this.acceptedTaxonKey = acceptedTaxonKey;
1262  }
1263
1264  @Nullable
1265  public String getSpecificEpithet() {
1266    return specificEpithet;
1267  }
1268
1269  public void setSpecificEpithet(String specificEpithet) {
1270    this.specificEpithet = specificEpithet;
1271  }
1272
1273  @Nullable
1274  public String getInfraspecificEpithet() {
1275    return infraspecificEpithet;
1276  }
1277
1278  public void setInfraspecificEpithet(String infraspecificEpithet) {
1279    this.infraspecificEpithet = infraspecificEpithet;
1280  }
1281
1282  @Nullable
1283  public Rank getTaxonRank() {
1284    return taxonRank;
1285  }
1286
1287  public void setTaxonRank(Rank taxonRank) {
1288    this.taxonRank = taxonRank;
1289  }
1290
1291  /**
1292   * The status of the use of the scientificName as a label for a taxon.
1293   * The GBIF recommended controlled value vocabulary can be found at <a href="http://rs.gbif.org/vocabulary/gbif/taxonomic_status.xml">http://rs.gbif.org/vocabulary/gbif/taxonomic_status.xm</a>.
1294   */
1295  @Nullable
1296  public TaxonomicStatus getTaxonomicStatus() {
1297    return taxonomicStatus;
1298  }
1299
1300  public void setTaxonomicStatus(TaxonomicStatus taxonomicStatus) {
1301    this.taxonomicStatus = taxonomicStatus;
1302  }
1303
1304  /**
1305   * The IUCN Red List Category.
1306   */
1307  @Nullable
1308  public String getIucnRedListCategory() {
1309    return iucnRedListCategory;
1310  }
1311
1312  public void setIucnRedListCategory(String iucnRedListCategory) {
1313    this.iucnRedListCategory = iucnRedListCategory;
1314  }
1315
1316  /**
1317   * The scientific name for taxonKey from the GBIF backbone.
1318   */
1319  @Nullable
1320  public String getScientificName() {
1321    return scientificName;
1322  }
1323
1324  public void setScientificName(@Nullable String scientificName) {
1325    this.scientificName = scientificName;
1326  }
1327
1328  /**
1329   * The verbatim scientific name as provided by the source.
1330   */
1331  @Nullable
1332  @JsonProperty(access = JsonProperty.Access.WRITE_ONLY)
1333  public String getVerbatimScientificName() {
1334    return getVerbatimField(DwcTerm.scientificName);
1335  }
1336
1337  public void setVerbatimScientificName(String scientificName) {
1338    //DO NOTHING
1339  }
1340
1341  /**
1342   * The accepted scientific name for the acceptedTaxonKey from the GBIF backbone.
1343   */
1344  @Nullable
1345  public String getAcceptedScientificName() {
1346    return acceptedScientificName;
1347  }
1348
1349  public void setAcceptedScientificName(String acceptedScientificName) {
1350    this.acceptedScientificName = acceptedScientificName;
1351  }
1352
1353  @Nullable
1354  @Override
1355  public String getKingdom() {
1356    return kingdom;
1357  }
1358
1359  @Override
1360  public void setKingdom(@Nullable String kingdom) {
1361    this.kingdom = kingdom;
1362  }
1363
1364  @Nullable
1365  @Override
1366  public String getPhylum() {
1367    return phylum;
1368  }
1369
1370  @Override
1371  public void setPhylum(@Nullable String phylum) {
1372    this.phylum = phylum;
1373  }
1374
1375  @Nullable
1376  @Override
1377  public String getClazz() {
1378    return clazz;
1379  }
1380
1381  @Override
1382  public void setClazz(@Nullable String clazz) {
1383    this.clazz = clazz;
1384  }
1385
1386  @Nullable
1387  @Override
1388  public String getOrder() {
1389    return order;
1390  }
1391
1392  @Override
1393  public void setOrder(@Nullable String order) {
1394    this.order = order;
1395  }
1396
1397  @Nullable
1398  @Override
1399  public String getFamily() {
1400    return family;
1401  }
1402
1403  @Override
1404  public void setFamily(@Nullable String family) {
1405    this.family = family;
1406  }
1407
1408  @Nullable
1409  @Override
1410  public String getGenus() {
1411    return genus;
1412  }
1413
1414  @Override
1415  public void setGenus(@Nullable String genus) {
1416    this.genus = genus;
1417  }
1418
1419  @Nullable
1420  public String getGenericName() {
1421    return genericName;
1422  }
1423
1424  public void setGenericName(String genericName) {
1425    this.genericName = genericName;
1426  }
1427
1428  @Nullable
1429  @Override
1430  public String getSubgenus() {
1431    return subgenus;
1432  }
1433
1434  @Override
1435  public void setSubgenus(@Nullable String subgenus) {
1436    this.subgenus = subgenus;
1437  }
1438
1439  @Nullable
1440  @Override
1441  public String getHigherRank(Rank rank) {
1442    return ClassificationUtils.getHigherRank(this, rank);
1443  }
1444
1445  /**
1446   * The corresponding scientific name of the speciesKey from the GBIF backbone.
1447   */
1448  @Nullable
1449  @Override
1450  public String getSpecies() {
1451    return species;
1452  }
1453
1454  @Override
1455  public void setSpecies(@Nullable String species) {
1456    this.species = species;
1457  }
1458
1459  @Nullable
1460  public Date getDateIdentified() {
1461    return dateIdentified == null ? null : new Date(dateIdentified.getTime());
1462  }
1463
1464  public void setDateIdentified(@Nullable Date dateIdentified) {
1465    this.dateIdentified = dateIdentified == null ? null : new Date(dateIdentified.getTime());
1466  }
1467
1468  /**
1469   * The decimalLongitude in decimal degrees always for the WGS84 datum. If a different geodetic datum was given the verbatim
1470   * coordinates are transformed into WGS84 values.
1471   */
1472  @Nullable
1473  public Double getDecimalLongitude() {
1474    return decimalLongitude;
1475  }
1476
1477  public void setDecimalLongitude(@Nullable Double decimalLongitude) {
1478    this.decimalLongitude = decimalLongitude;
1479  }
1480
1481  @Nullable
1482  public Double getDecimalLatitude() {
1483    return decimalLatitude;
1484  }
1485
1486  public void setDecimalLatitude(@Nullable Double decimalLatitude) {
1487    this.decimalLatitude = decimalLatitude;
1488  }
1489
1490  /**
1491   * The uncertainty radius for lat/lon in meters.
1492   */
1493  @Nullable
1494  public Double getCoordinateUncertaintyInMeters() {
1495    return coordinateUncertaintyInMeters;
1496  }
1497
1498  public void setCoordinateUncertaintyInMeters(@Nullable Double coordinateUncertaintyInMeters) {
1499    this.coordinateUncertaintyInMeters = coordinateUncertaintyInMeters;
1500  }
1501
1502  @Nullable
1503  public Double getCoordinatePrecision() {
1504    return coordinatePrecision;
1505  }
1506
1507  public void setCoordinatePrecision(Double coordinatePrecision) {
1508    this.coordinatePrecision = coordinatePrecision;
1509  }
1510
1511  /**
1512   * @deprecated to be removed in the public v2 of the API (see POR-3061)
1513   * The uncertainty for latitude in decimal degrees.
1514   * Note that the longitude degrees have a different accuracy in degrees which changes with latitude and is largest at the poles.
1515   */
1516  @Nullable
1517  @Deprecated
1518  public Double getCoordinateAccuracy() {
1519    return coordinateAccuracy;
1520  }
1521
1522  public void setCoordinateAccuracy(@Nullable Double coordinateAccuracy) {
1523    this.coordinateAccuracy = coordinateAccuracy;
1524  }
1525
1526  /**
1527   * The geodetic datum for the interpreted decimal coordinates.
1528   * This is always WGS84 if a coordinate exists as we reproject other datums into WGS84.
1529   */
1530  @Schema(
1531    description = "The geodetic datum for the interpreted decimal coordinates.\n\n" +
1532      "Coordinates are reprojected to WGS84 if they exist, so this field is either null or `WGS84`."
1533  )
1534  @Nullable
1535  public String getGeodeticDatum() {
1536    if (decimalLatitude != null) {
1537      return GEO_DATUM;
1538    }
1539    return null;
1540  }
1541
1542  /**
1543   * This private method is needed for jackson deserialization only.
1544   */
1545  private void setGeodeticDatum(String datum) {
1546    // ignore, we have a static WGS84 value
1547  }
1548
1549  /**
1550   * Elevation in meters usually above sea level (altitude).
1551   * </br>
1552   * The elevation is calculated using the equation: (minimumElevationInMeters + maximumElevationInMeters) / 2.
1553   */
1554  @Nullable
1555  public Double getElevation() {
1556    return elevation;
1557  }
1558
1559  public void setElevation(@Nullable Double elevation) {
1560    this.elevation = elevation;
1561  }
1562
1563  /**
1564   * Elevation accuracy is the uncertainty for the elevation in meters.
1565   * </br>
1566   * The elevation accuracy is calculated using the equation: (maximumElevationInMeters - minimumElevationInMeters) / 2
1567   */
1568  @Nullable
1569  public Double getElevationAccuracy() {
1570    return elevationAccuracy;
1571  }
1572
1573  public void setElevationAccuracy(@Nullable Double elevationAccuracy) {
1574    this.elevationAccuracy = elevationAccuracy;
1575  }
1576
1577  /**
1578   * Depth in meters below the surface. Complimentary to elevation, the depth can be 10 meters below the surface of a
1579   * lake in 1100m (=elevation).
1580   * </br>
1581   * The depth is calculated using the equation: (minimumDepthInMeters + maximumDepthInMeters) / 2.
1582   */
1583  @Nullable
1584  public Double getDepth() {
1585    return depth;
1586  }
1587
1588  public void setDepth(@Nullable Double depth) {
1589    this.depth = depth;
1590  }
1591
1592  /**
1593   * Depth accuracy is the uncertainty for the depth in meters.
1594   * </br>
1595   * The depth accuracy is calculated using the equation: (maximumDepthInMeters - minimumDepthInMeters) / 2
1596   */
1597  @Nullable
1598  public Double getDepthAccuracy() {
1599    return depthAccuracy;
1600  }
1601
1602  public void setDepthAccuracy(@Nullable Double depthAccuracy) {
1603    this.depthAccuracy = depthAccuracy;
1604  }
1605
1606  @Nullable
1607  public Continent getContinent() {
1608    return continent;
1609  }
1610
1611  public void setContinent(@Nullable Continent continent) {
1612    this.continent = continent;
1613  }
1614
1615  @Schema(
1616    description = "The 2-letter country code (as per ISO-3166-1) of the country, territory or area in which the " +
1617      "occurrence was recorded.",
1618    externalDocs = @ExternalDocumentation(
1619      description = "Darwin Core definition",
1620      url = "https://rs.tdwg.org/dwc/terms/countryCode"
1621    )
1622  )
1623  @Nullable
1624  @JsonProperty("countryCode")
1625  public Country getCountry() {
1626    return country;
1627  }
1628
1629  public void setCountry(@Nullable Country country) {
1630    this.country = country;
1631  }
1632
1633  @Nullable
1634  @JsonProperty("gbifRegion")
1635  public GbifRegion getGbifRegion() {
1636    return Optional.ofNullable(country).map(Country::getGbifRegion).orElse(null);
1637  }
1638
1639  private void setGbifRegion(String gbifRegion) {
1640    // ignore, setter only to avoid JSON being written into the fields map
1641  }
1642
1643  /**
1644   * Renders the country title as a JSON property country in addition to the ISO 3166 2 letter countryCode being
1645   * serialized by the regular country Java property.
1646   * Made private to use it only for JSON serialization and not within Java code.
1647   */
1648  @Schema(
1649    description = "The title (as per ISO-3166-1) of the country, territory or area in which the " +
1650      "occurrence was recorded.",
1651    externalDocs = @ExternalDocumentation(
1652      description = "Darwin Core definition",
1653      url = "https://rs.tdwg.org/dwc/terms/country"
1654    )
1655  )
1656  @Nullable
1657  @JsonProperty("country")
1658  private String getCountryTitle() {
1659    return country == null ? null : country.getTitle();
1660  }
1661
1662  private void setCountryTitle(String country) {
1663    // ignore, setter only to avoid JSON being written into the fields map
1664  }
1665
1666  @Nullable
1667  public String getStateProvince() {
1668    return stateProvince;
1669  }
1670
1671  public void setStateProvince(@Nullable String stateProvince) {
1672    this.stateProvince = stateProvince;
1673  }
1674
1675  @Nullable
1676  public String getWaterBody() {
1677    return waterBody;
1678  }
1679
1680  public void setWaterBody(@Nullable String waterBody) {
1681    this.waterBody = waterBody;
1682  }
1683
1684  /**
1685   * The distance in metres from a known centroid, e.g. a country centroid.
1686   */
1687  public Double getDistanceFromCentroidInMeters() {
1688    return distanceFromCentroidInMeters;
1689  }
1690
1691  public void setDistanceFromCentroidInMeters(Double distanceFromCentroidInMeters) {
1692    this.distanceFromCentroidInMeters = distanceFromCentroidInMeters;
1693  }
1694
1695  @Nullable
1696  public String getHigherGeography() {
1697    return higherGeography;
1698  }
1699
1700  public void setHigherGeography(String higherGeography) {
1701    this.higherGeography = higherGeography;
1702  }
1703
1704  @Nullable
1705  public String getGeoreferencedBy() {
1706    return georeferencedBy;
1707  }
1708
1709  public void setGeoreferencedBy(String georeferencedBy) {
1710    this.georeferencedBy = georeferencedBy;
1711  }
1712
1713  /**
1714   * The full year of the event date.
1715   *
1716   * @return the year of the event date
1717   */
1718  @Min(1500)
1719  @Max(2030)
1720  @Nullable
1721  public Integer getYear() {
1722    return year;
1723  }
1724
1725  public void setYear(@Nullable Integer year) {
1726    this.year = year;
1727  }
1728
1729  /**
1730   * The month of the year of the event date starting with zero for january following {@link Date}.
1731   *
1732   * @return the month of the event date
1733   */
1734  @Min(1)
1735  @Max(12)
1736  @Nullable
1737  public Integer getMonth() {
1738    return month;
1739  }
1740
1741  public void setMonth(@Nullable Integer month) {
1742    this.month = month;
1743  }
1744
1745  /**
1746   * The day of the month of the event date.
1747   *
1748   * @return the day of the event date
1749   */
1750  @Min(1)
1751  @Max(31)
1752  @Nullable
1753  public Integer getDay() {
1754    return day;
1755  }
1756
1757  public void setDay(@Nullable Integer day) {
1758    this.day = day;
1759  }
1760
1761  /**
1762   * The date the occurrence was recorded or collected.
1763   */
1764  @Nullable
1765  public IsoDateInterval getEventDate() {
1766    return eventDate == null ? null : new IsoDateInterval(eventDate.getFrom(), eventDate.getTo());
1767  }
1768
1769  public void setEventDate(@Nullable IsoDateInterval eventDate) {
1770    this.eventDate = eventDate == null ? null : new IsoDateInterval(eventDate.getFrom(), eventDate.getTo());
1771  }
1772
1773  /**
1774   * The earliest integer day of the year of the event.
1775   *
1776   * @return the earliest integer day of the event date
1777   */
1778  @Min(1)
1779  @Max(366)
1780  @Nullable
1781  public Integer getStartDayOfYear() {
1782    return startDayOfYear;
1783  }
1784
1785  public void setStartDayOfYear(@Nullable Integer startDayOfYear) {
1786    this.startDayOfYear = startDayOfYear;
1787  }
1788
1789  /**
1790   * The latest integer day of the year of the event.
1791   *
1792   * @return the latest integer day of the event date
1793   */
1794  @Min(1)
1795  @Max(366)
1796  @Nullable
1797  public Integer getEndDayOfYear() {
1798    return endDayOfYear;
1799  }
1800
1801  public void setEndDayOfYear(@Nullable Integer endDayOfYear) {
1802    this.endDayOfYear = endDayOfYear;
1803  }
1804
1805  @Nullable
1806  public String getTypeStatus() {
1807    return typeStatus;
1808  }
1809
1810  public void setTypeStatus(@Nullable String typeStatus) {
1811    this.typeStatus = typeStatus;
1812  }
1813
1814  /**
1815   * The scientific name the type status of this specimen applies to.
1816   */
1817  @Nullable
1818  public String getTypifiedName() {
1819    return typifiedName;
1820  }
1821
1822  public void setTypifiedName(@Nullable String typifiedName) {
1823    this.typifiedName = typifiedName;
1824  }
1825
1826  /**
1827   * A set of issues found for this occurrence.
1828   */
1829  @NotNull
1830  public Set<OccurrenceIssue> getIssues() {
1831    return issues;
1832  }
1833
1834  public void setIssues(Set<OccurrenceIssue> issues) {
1835    Objects.requireNonNull(issues, "Issues cannot be null");
1836    EnumSet<OccurrenceIssue> set = EnumSet.noneOf(OccurrenceIssue.class);
1837    set.addAll(issues);
1838    this.issues = set;
1839  }
1840
1841  public void addIssue(OccurrenceIssue issue) {
1842    Objects.requireNonNull(issue, "Issue needs to be specified");
1843    issues.add(issue);
1844  }
1845
1846  /**
1847   * The interpreted dc:modified from the verbatim source data.
1848   * Ideally indicating when a record was last modified in the source.
1849   */
1850  @Nullable
1851  public Date getModified() {
1852    return modified == null ? null : new Date(modified.getTime());
1853  }
1854
1855  public void setModified(@Nullable Date modified) {
1856    this.modified = modified == null ? null : new Date(modified.getTime());
1857  }
1858
1859  /**
1860   * The date this occurrence last went through the interpretation phase of the GBIF indexing.
1861   */
1862  @Nullable
1863  public Date getLastInterpreted() {
1864    return lastInterpreted == null ? null : new Date(lastInterpreted.getTime());
1865  }
1866
1867  public void setLastInterpreted(@Nullable Date lastInterpreted) {
1868    this.lastInterpreted = lastInterpreted == null ? null : new Date(lastInterpreted.getTime());
1869  }
1870
1871  /**
1872   * An external link to more details, the records "homepage".
1873   */
1874  @Nullable
1875  public URI getReferences() {
1876    return references;
1877  }
1878
1879  public void setReferences(URI references) {
1880    this.references = references;
1881  }
1882
1883  /**
1884   * A number or enumeration value for the quantity of organisms.
1885   */
1886  @Nullable
1887  public Double getOrganismQuantity() {
1888    return organismQuantity;
1889  }
1890
1891  public void setOrganismQuantity(@Nullable Double organismQuantity) {
1892    this.organismQuantity = organismQuantity;
1893  }
1894
1895  /**
1896   * The type of quantification system used for the quantity of organisms.
1897   */
1898  @Nullable
1899  public String getOrganismQuantityType() {
1900    return organismQuantityType;
1901  }
1902
1903  public void setOrganismQuantityType(@Nullable String organismQuantityType) {
1904    this.organismQuantityType = organismQuantityType;
1905  }
1906
1907  /**
1908   * The unit of measurement of the size (time duration, length, area, or volume) of a sample in a sampling event.
1909   */
1910  @Nullable
1911  public String getSampleSizeUnit() {
1912    return sampleSizeUnit;
1913  }
1914
1915  public void setSampleSizeUnit(@Nullable String sampleSizeUnit) {
1916    this.sampleSizeUnit = sampleSizeUnit;
1917  }
1918
1919  /**
1920   * A numeric value for a measurement of the size (time duration, length, area, or volume) of a sample in a sampling event.
1921   */
1922  @Nullable
1923  public Double getSampleSizeValue() {
1924    return sampleSizeValue;
1925  }
1926
1927  public void setSampleSizeValue(@Nullable Double sampleSizeValue) {
1928    this.sampleSizeValue = sampleSizeValue;
1929  }
1930
1931  /**
1932   * Calculated filed organismQuantity / sampleSizeValue, if the type is identical
1933   */
1934  @Nullable
1935  public Double getRelativeOrganismQuantity() {
1936    return relativeOrganismQuantity;
1937  }
1938
1939  public void setRelativeOrganismQuantity(@Nullable Double relativeOrganismQuantity) {
1940    this.relativeOrganismQuantity = relativeOrganismQuantity;
1941  }
1942
1943  /**
1944   * Flag occurrence when associatedSequences/extension exists
1945   */
1946  public boolean getIsSequenced() {
1947    return isSequenced;
1948  }
1949
1950  public void setIsSequenced(boolean isSequenced) {
1951    this.isSequenced = isSequenced;
1952  }
1953
1954  /**
1955   * A list (concatenated and separated) of identifiers (publication, global unique identifier, URI) of genetic
1956   * sequence information associated with the material entity.
1957   */
1958  @Nullable
1959  public String getAssociatedSequences() {
1960    return associatedSequences;
1961  }
1962
1963  public void setAssociatedSequences(String associatedSequences) {
1964    this.associatedSequences = associatedSequences;
1965  }
1966
1967  /**
1968   * Applied license to the occurrence record or dataset to which this record belongs to.
1969   */
1970  @NotNull
1971  public License getLicense() {
1972    return license;
1973  }
1974
1975  public void setLicense(License license) {
1976    this.license = license;
1977  }
1978
1979  @NotNull
1980  public List<Identifier> getIdentifiers() {
1981    return identifiers;
1982  }
1983
1984  public void setIdentifiers(List<Identifier> identifiers) {
1985    this.identifiers = identifiers;
1986  }
1987
1988  @NotNull
1989  public List<MediaObject> getMedia() {
1990    return media;
1991  }
1992
1993  public void setMedia(List<MediaObject> media) {
1994    this.media = media;
1995  }
1996
1997  @NotNull
1998  public List<MeasurementOrFact> getFacts() {
1999    return facts;
2000  }
2001
2002  public void setFacts(List<MeasurementOrFact> facts) {
2003    this.facts = facts;
2004  }
2005
2006  @NotNull
2007  public List<OccurrenceRelation> getRelations() {
2008    return relations;
2009  }
2010
2011  public void setRelations(List<OccurrenceRelation> relations) {
2012    this.relations = relations;
2013  }
2014
2015  @NotNull
2016  public List<AgentIdentifier> getRecordedByIds() {
2017    return recordedByIds;
2018  }
2019
2020  public void setRecordedByIds(List<AgentIdentifier> recordedByIds) {
2021    this.recordedByIds = recordedByIds;
2022  }
2023
2024  @NotNull
2025  public List<AgentIdentifier> getIdentifiedByIds() {
2026    return identifiedByIds;
2027  }
2028
2029  public void setIdentifiedByIds(List<AgentIdentifier> identifiedByIds) {
2030    this.identifiedByIds = identifiedByIds;
2031  }
2032
2033  @NotNull
2034  public Gadm getGadm() {
2035    return gadm;
2036  }
2037
2038  public void setGadm(Gadm gadm) {
2039    this.gadm = gadm;
2040  }
2041
2042  @Nullable
2043  @Experimental
2044  public String getInstitutionKey() {
2045    return institutionKey;
2046  }
2047
2048  public void setInstitutionKey(String institutionKey) {
2049    this.institutionKey = institutionKey;
2050  }
2051
2052  @Nullable
2053  @Experimental
2054  public String getCollectionKey() {
2055    return collectionKey;
2056  }
2057
2058  public void setCollectionKey(String collectionKey) {
2059    this.collectionKey = collectionKey;
2060  }
2061
2062  public boolean getIsInCluster() {
2063    return isInCluster;
2064  }
2065
2066  public void setIsInCluster(boolean isInCluster) {
2067    this.isInCluster = isInCluster;
2068  }
2069
2070  @Nullable
2071  public String getDegreeOfEstablishment() {
2072    return degreeOfEstablishment;
2073  }
2074
2075  public void setDegreeOfEstablishment(String degreeOfEstablishment) {
2076    this.degreeOfEstablishment = degreeOfEstablishment;
2077  }
2078
2079  @Nullable
2080  public String getPathway() {
2081    return pathway;
2082  }
2083
2084  public void setPathway(String pathway) {
2085    this.pathway = pathway;
2086  }
2087
2088  public String getDatasetID() {
2089    return datasetID;
2090  }
2091
2092  public void setDatasetID(String datasetID) {
2093    this.datasetID = datasetID;
2094  }
2095
2096  public String getDatasetName() {
2097    return datasetName;
2098  }
2099
2100  public void setDatasetName(String datasetName) {
2101    this.datasetName = datasetName;
2102  }
2103
2104  public String getOtherCatalogNumbers() {
2105    return otherCatalogNumbers;
2106  }
2107
2108  public void setOtherCatalogNumbers(String otherCatalogNumbers) {
2109    this.otherCatalogNumbers = otherCatalogNumbers;
2110  }
2111
2112  @Nullable
2113  public String getEarliestEonOrLowestEonothem() {
2114    return earliestEonOrLowestEonothem;
2115  }
2116
2117  public void setEarliestEonOrLowestEonothem(String earliestEonOrLowestEonothem) {
2118    this.earliestEonOrLowestEonothem = earliestEonOrLowestEonothem;
2119  }
2120
2121  @Nullable
2122  public String getLatestEonOrHighestEonothem() {
2123    return latestEonOrHighestEonothem;
2124  }
2125
2126  public void setLatestEonOrHighestEonothem(String latestEonOrHighestEonothem) {
2127    this.latestEonOrHighestEonothem = latestEonOrHighestEonothem;
2128  }
2129
2130  @Nullable
2131  public String getEarliestEraOrLowestErathem() {
2132    return earliestEraOrLowestErathem;
2133  }
2134
2135  public void setEarliestEraOrLowestErathem(String earliestEraOrLowestErathem) {
2136    this.earliestEraOrLowestErathem = earliestEraOrLowestErathem;
2137  }
2138
2139  @Nullable
2140  public String getLatestEraOrHighestErathem() {
2141    return latestEraOrHighestErathem;
2142  }
2143
2144  public void setLatestEraOrHighestErathem(String latestEraOrHighestErathem) {
2145    this.latestEraOrHighestErathem = latestEraOrHighestErathem;
2146  }
2147
2148  @Nullable
2149  public String getEarliestPeriodOrLowestSystem() {
2150    return earliestPeriodOrLowestSystem;
2151  }
2152
2153  public void setEarliestPeriodOrLowestSystem(String earliestPeriodOrLowestSystem) {
2154    this.earliestPeriodOrLowestSystem = earliestPeriodOrLowestSystem;
2155  }
2156
2157  @Nullable
2158  public String getLatestPeriodOrHighestSystem() {
2159    return latestPeriodOrHighestSystem;
2160  }
2161
2162  public void setLatestPeriodOrHighestSystem(String latestPeriodOrHighestSystem) {
2163    this.latestPeriodOrHighestSystem = latestPeriodOrHighestSystem;
2164  }
2165
2166  @Nullable
2167  public String getEarliestEpochOrLowestSeries() {
2168    return earliestEpochOrLowestSeries;
2169  }
2170
2171  public void setEarliestEpochOrLowestSeries(String earliestEpochOrLowestSeries) {
2172    this.earliestEpochOrLowestSeries = earliestEpochOrLowestSeries;
2173  }
2174
2175  @Nullable
2176  public String getLatestEpochOrHighestSeries() {
2177    return latestEpochOrHighestSeries;
2178  }
2179
2180  public void setLatestEpochOrHighestSeries(String latestEpochOrHighestSeries) {
2181    this.latestEpochOrHighestSeries = latestEpochOrHighestSeries;
2182  }
2183
2184  @Nullable
2185  public String getEarliestAgeOrLowestStage() {
2186    return earliestAgeOrLowestStage;
2187  }
2188
2189  public void setEarliestAgeOrLowestStage(String earliestAgeOrLowestStage) {
2190    this.earliestAgeOrLowestStage = earliestAgeOrLowestStage;
2191  }
2192
2193  @Nullable
2194  public String getLatestAgeOrHighestStage() {
2195    return latestAgeOrHighestStage;
2196  }
2197
2198  public void setLatestAgeOrHighestStage(String latestAgeOrHighestStage) {
2199    this.latestAgeOrHighestStage = latestAgeOrHighestStage;
2200  }
2201
2202  @Nullable
2203  public String getLowestBiostratigraphicZone() {
2204    return lowestBiostratigraphicZone;
2205  }
2206
2207  public void setLowestBiostratigraphicZone(String lowestBiostratigraphicZone) {
2208    this.lowestBiostratigraphicZone = lowestBiostratigraphicZone;
2209  }
2210
2211  @Nullable
2212  public String getHighestBiostratigraphicZone() {
2213    return highestBiostratigraphicZone;
2214  }
2215
2216  public void setHighestBiostratigraphicZone(String highestBiostratigraphicZone) {
2217    this.highestBiostratigraphicZone = highestBiostratigraphicZone;
2218  }
2219
2220  @Nullable
2221  public String getGroup() {
2222    return group;
2223  }
2224
2225  public void setGroup(String group) {
2226    this.group = group;
2227  }
2228
2229  @Nullable
2230  public String getFormation() {
2231    return formation;
2232  }
2233
2234  public void setFormation(String formation) {
2235    this.formation = formation;
2236  }
2237
2238  @Nullable
2239  public String getMember() {
2240    return member;
2241  }
2242
2243  public void setMember(String member) {
2244    this.member = member;
2245  }
2246
2247  @Nullable
2248  public String getBed() {
2249    return bed;
2250  }
2251
2252  public void setBed(String bed) {
2253    this.bed = bed;
2254  }
2255
2256  public String getRecordedBy() {
2257    return recordedBy;
2258  }
2259
2260  public void setRecordedBy(String recordedBy) {
2261    this.recordedBy = recordedBy;
2262  }
2263
2264  public String getIdentifiedBy() {
2265    return identifiedBy;
2266  }
2267
2268  public void setIdentifiedBy(String identifiedBy) {
2269    this.identifiedBy = identifiedBy;
2270  }
2271
2272  public String getPreparations() {
2273    return preparations;
2274  }
2275
2276  public void setPreparations(String preparations) {
2277    this.preparations = preparations;
2278  }
2279
2280  public String getSamplingProtocol() {
2281    return samplingProtocol;
2282  }
2283
2284  public void setSamplingProtocol(String samplingProtocol) {
2285    this.samplingProtocol = samplingProtocol;
2286  }
2287
2288  public List<String> getDnaSequenceID() {
2289    return dnaSequenceID;
2290  }
2291
2292  public void setDnaSequenceID(List<String> dnaSequenceID) {
2293    this.dnaSequenceID = dnaSequenceID;
2294  }
2295
2296  /**
2297   * Convenience method checking if any spatial validation rule has not passed.
2298   * Primarily used to indicate that the record should not be displayed on a map.
2299   */
2300  @JsonIgnore
2301  public boolean hasSpatialIssue() {
2302    for (OccurrenceIssue rule : OccurrenceIssue.GEOSPATIAL_RULES) {
2303      if (issues.contains(rule)) {
2304        return true;
2305      }
2306    }
2307    return false;
2308  }
2309
2310  /**
2311   * This private method is only for serialization via jackson and not exposed anywhere else!
2312   * It maps the verbatimField terms into properties with their simple name or qualified names for UnknownTerms.
2313   */
2314  @JsonAnyGetter
2315  private Map<String, String> jsonVerbatimFields() {
2316    Map<String, String> extendedProps = new HashMap<>();
2317    for (Map.Entry<Term, String> prop : getVerbatimFields().entrySet()) {
2318      Term t = prop.getKey();
2319      if (t instanceof UnknownTerm || PROPERTIES.contains(t.simpleName())) {
2320        extendedProps.put(t.qualifiedName(), prop.getValue());
2321      } else {
2322        // render all terms in controlled enumerations as simple names only - unless we have a property of that name already!
2323        extendedProps.put(t.simpleName(), prop.getValue());
2324      }
2325    }
2326    return extendedProps;
2327  }
2328}