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}