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 public Occurrence() { 1039 1040 } 1041 1042 /** 1043 * Create occurrence instance from existing verbatim one, copying over all data. 1044 */ 1045 public Occurrence(@Nullable VerbatimOccurrence verbatim) { 1046 if (verbatim != null) { 1047 setKey(verbatim.getKey()); 1048 setDatasetKey(verbatim.getDatasetKey()); 1049 setPublishingOrgKey(verbatim.getPublishingOrgKey()); 1050 setPublishingCountry(verbatim.getPublishingCountry()); 1051 setProtocol(verbatim.getProtocol()); 1052 setCrawlId(verbatim.getCrawlId()); 1053 if (verbatim.getLastCrawled() != null) { 1054 setLastCrawled(new Date(verbatim.getLastCrawled().getTime())); 1055 } 1056 if (verbatim.getVerbatimFields() != null) { 1057 getVerbatimFields().putAll(verbatim.getVerbatimFields()); 1058 } 1059 if (verbatim.getLastParsed() != null) { 1060 setLastParsed(verbatim.getLastParsed()); 1061 } 1062 setExtensions(verbatim.getExtensions()); 1063 } 1064 } 1065 1066 @Nullable 1067 public BasisOfRecord getBasisOfRecord() { 1068 return basisOfRecord; 1069 } 1070 1071 public void setBasisOfRecord(BasisOfRecord basisOfRecord) { 1072 this.basisOfRecord = basisOfRecord; 1073 } 1074 1075 @Nullable 1076 public Integer getIndividualCount() { 1077 return individualCount; 1078 } 1079 1080 public void setIndividualCount(Integer individualCount) { 1081 this.individualCount = individualCount; 1082 } 1083 1084 @Nullable 1085 public OccurrenceStatus getOccurrenceStatus() { 1086 return occurrenceStatus; 1087 } 1088 1089 public void setOccurrenceStatus(OccurrenceStatus occurrenceStatus) { 1090 this.occurrenceStatus = occurrenceStatus; 1091 } 1092 1093 @Nullable 1094 public String getSex() { 1095 return sex; 1096 } 1097 1098 public void setSex(String sex) { 1099 this.sex = sex; 1100 } 1101 1102 @Nullable 1103 public String getLifeStage() { 1104 return lifeStage; 1105 } 1106 1107 public void setLifeStage(String lifeStage) { 1108 this.lifeStage = lifeStage; 1109 } 1110 1111 @Nullable 1112 public String getEstablishmentMeans() { 1113 return establishmentMeans; 1114 } 1115 1116 public void setEstablishmentMeans(String establishmentMeans) { 1117 this.establishmentMeans = establishmentMeans; 1118 } 1119 1120 /** 1121 * The best matching, accepted GBIF backbone name usage representing this occurrence. 1122 * In case the verbatim scientific name and its classification can only be matched to a higher rank this will 1123 * represent the lowest matching rank. In the worst case this could just be for example Animalia. 1124 */ 1125 @Nullable 1126 public Integer getTaxonKey() { 1127 return taxonKey; 1128 } 1129 1130 public void setTaxonKey(Integer taxonKey) { 1131 this.taxonKey = taxonKey; 1132 } 1133 1134 @Nullable 1135 @Override 1136 public Integer getKingdomKey() { 1137 return kingdomKey; 1138 } 1139 1140 @Override 1141 public void setKingdomKey(@Nullable Integer kingdomKey) { 1142 this.kingdomKey = kingdomKey; 1143 } 1144 1145 @Nullable 1146 @Override 1147 public Integer getPhylumKey() { 1148 return phylumKey; 1149 } 1150 1151 @Override 1152 public void setPhylumKey(@Nullable Integer phylumKey) { 1153 this.phylumKey = phylumKey; 1154 } 1155 1156 @Nullable 1157 @Override 1158 public Integer getClassKey() { 1159 return classKey; 1160 } 1161 1162 @Override 1163 public void setClassKey(@Nullable Integer classKey) { 1164 this.classKey = classKey; 1165 } 1166 1167 @Nullable 1168 @Override 1169 public Integer getOrderKey() { 1170 return orderKey; 1171 } 1172 1173 @Override 1174 public void setOrderKey(@Nullable Integer orderKey) { 1175 this.orderKey = orderKey; 1176 } 1177 1178 @Nullable 1179 @Override 1180 public Integer getFamilyKey() { 1181 return familyKey; 1182 } 1183 1184 @Override 1185 public void setFamilyKey(@Nullable Integer familyKey) { 1186 this.familyKey = familyKey; 1187 } 1188 1189 @Nullable 1190 @Override 1191 public Integer getGenusKey() { 1192 return genusKey; 1193 } 1194 1195 @Override 1196 public void setGenusKey(@Nullable Integer genusKey) { 1197 this.genusKey = genusKey; 1198 } 1199 1200 @Nullable 1201 @Override 1202 public Integer getSubgenusKey() { 1203 return subgenusKey; 1204 } 1205 1206 @Override 1207 public void setSubgenusKey(@Nullable Integer subgenusKey) { 1208 this.subgenusKey = subgenusKey; 1209 } 1210 1211 @Nullable 1212 @Override 1213 public Integer getHigherRankKey(Rank rank) { 1214 return ClassificationUtils.getHigherRankKey(this, rank); 1215 } 1216 1217 /** 1218 * An ordered map with entries for all higher Linnean ranks excluding the taxonKey itself. 1219 * The map starts with the highest rank, e.g. the kingdom and maps the name usage key to its canonical name. 1220 * 1221 * @return map of higher ranks 1222 */ 1223 @NotNull 1224 @JsonIgnore 1225 public Map<Integer, String> getHigherClassificationMap() { 1226 return taxonKey == null ? ClassificationUtils.getHigherClassificationMap(this) 1227 : ClassificationUtils.getHigherClassificationMap(this, taxonKey, null, null); 1228 } 1229 1230 /** 1231 * The accepted species for this occurrence. In case the taxonKey is of a higher rank than species (e.g. genus) 1232 * speciesKey is null. In case taxonKey represents an infraspecific taxon the speciesKey points to the species 1233 * the infraspecies is classified as. In case of taxonKey being a species the speciesKey is the same. 1234 */ 1235 @Nullable 1236 @Override 1237 public Integer getSpeciesKey() { 1238 return speciesKey; 1239 } 1240 1241 @Override 1242 public void setSpeciesKey(@Nullable Integer speciesKey) { 1243 this.speciesKey = speciesKey; 1244 } 1245 1246 /** 1247 * The accepted taxon key from the GBIF backbone. 1248 */ 1249 @Nullable 1250 public Integer getAcceptedTaxonKey() { 1251 return acceptedTaxonKey; 1252 } 1253 1254 public void setAcceptedTaxonKey(Integer acceptedTaxonKey) { 1255 this.acceptedTaxonKey = acceptedTaxonKey; 1256 } 1257 1258 @Nullable 1259 public String getSpecificEpithet() { 1260 return specificEpithet; 1261 } 1262 1263 public void setSpecificEpithet(String specificEpithet) { 1264 this.specificEpithet = specificEpithet; 1265 } 1266 1267 @Nullable 1268 public String getInfraspecificEpithet() { 1269 return infraspecificEpithet; 1270 } 1271 1272 public void setInfraspecificEpithet(String infraspecificEpithet) { 1273 this.infraspecificEpithet = infraspecificEpithet; 1274 } 1275 1276 @Nullable 1277 public Rank getTaxonRank() { 1278 return taxonRank; 1279 } 1280 1281 public void setTaxonRank(Rank taxonRank) { 1282 this.taxonRank = taxonRank; 1283 } 1284 1285 /** 1286 * The status of the use of the scientificName as a label for a taxon. 1287 * 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>. 1288 */ 1289 @Nullable 1290 public TaxonomicStatus getTaxonomicStatus() { 1291 return taxonomicStatus; 1292 } 1293 1294 public void setTaxonomicStatus(TaxonomicStatus taxonomicStatus) { 1295 this.taxonomicStatus = taxonomicStatus; 1296 } 1297 1298 /** 1299 * The IUCN Red List Category. 1300 */ 1301 @Nullable 1302 public String getIucnRedListCategory() { 1303 return iucnRedListCategory; 1304 } 1305 1306 public void setIucnRedListCategory(String iucnRedListCategory) { 1307 this.iucnRedListCategory = iucnRedListCategory; 1308 } 1309 1310 /** 1311 * The scientific name for taxonKey from the GBIF backbone. 1312 */ 1313 @Nullable 1314 public String getScientificName() { 1315 return scientificName; 1316 } 1317 1318 public void setScientificName(@Nullable String scientificName) { 1319 this.scientificName = scientificName; 1320 } 1321 1322 /** 1323 * The verbatim scientific name as provided by the source. 1324 */ 1325 @Nullable 1326 @JsonProperty(access = JsonProperty.Access.WRITE_ONLY) 1327 public String getVerbatimScientificName() { 1328 return getVerbatimField(DwcTerm.scientificName); 1329 } 1330 1331 public void setVerbatimScientificName(String scientificName) { 1332 //DO NOTHING 1333 } 1334 1335 /** 1336 * The accepted scientific name for the acceptedTaxonKey from the GBIF backbone. 1337 */ 1338 @Nullable 1339 public String getAcceptedScientificName() { 1340 return acceptedScientificName; 1341 } 1342 1343 public void setAcceptedScientificName(String acceptedScientificName) { 1344 this.acceptedScientificName = acceptedScientificName; 1345 } 1346 1347 @Nullable 1348 @Override 1349 public String getKingdom() { 1350 return kingdom; 1351 } 1352 1353 @Override 1354 public void setKingdom(@Nullable String kingdom) { 1355 this.kingdom = kingdom; 1356 } 1357 1358 @Nullable 1359 @Override 1360 public String getPhylum() { 1361 return phylum; 1362 } 1363 1364 @Override 1365 public void setPhylum(@Nullable String phylum) { 1366 this.phylum = phylum; 1367 } 1368 1369 @Nullable 1370 @Override 1371 public String getClazz() { 1372 return clazz; 1373 } 1374 1375 @Override 1376 public void setClazz(@Nullable String clazz) { 1377 this.clazz = clazz; 1378 } 1379 1380 @Nullable 1381 @Override 1382 public String getOrder() { 1383 return order; 1384 } 1385 1386 @Override 1387 public void setOrder(@Nullable String order) { 1388 this.order = order; 1389 } 1390 1391 @Nullable 1392 @Override 1393 public String getFamily() { 1394 return family; 1395 } 1396 1397 @Override 1398 public void setFamily(@Nullable String family) { 1399 this.family = family; 1400 } 1401 1402 @Nullable 1403 @Override 1404 public String getGenus() { 1405 return genus; 1406 } 1407 1408 @Override 1409 public void setGenus(@Nullable String genus) { 1410 this.genus = genus; 1411 } 1412 1413 @Nullable 1414 public String getGenericName() { 1415 return genericName; 1416 } 1417 1418 public void setGenericName(String genericName) { 1419 this.genericName = genericName; 1420 } 1421 1422 @Nullable 1423 @Override 1424 public String getSubgenus() { 1425 return subgenus; 1426 } 1427 1428 @Override 1429 public void setSubgenus(@Nullable String subgenus) { 1430 this.subgenus = subgenus; 1431 } 1432 1433 @Nullable 1434 @Override 1435 public String getHigherRank(Rank rank) { 1436 return ClassificationUtils.getHigherRank(this, rank); 1437 } 1438 1439 /** 1440 * The corresponding scientific name of the speciesKey from the GBIF backbone. 1441 */ 1442 @Nullable 1443 @Override 1444 public String getSpecies() { 1445 return species; 1446 } 1447 1448 @Override 1449 public void setSpecies(@Nullable String species) { 1450 this.species = species; 1451 } 1452 1453 @Nullable 1454 public Date getDateIdentified() { 1455 return dateIdentified == null ? null : new Date(dateIdentified.getTime()); 1456 } 1457 1458 public void setDateIdentified(@Nullable Date dateIdentified) { 1459 this.dateIdentified = dateIdentified == null ? null : new Date(dateIdentified.getTime()); 1460 } 1461 1462 /** 1463 * The decimalLongitude in decimal degrees always for the WGS84 datum. If a different geodetic datum was given the verbatim 1464 * coordinates are transformed into WGS84 values. 1465 */ 1466 @Nullable 1467 public Double getDecimalLongitude() { 1468 return decimalLongitude; 1469 } 1470 1471 public void setDecimalLongitude(@Nullable Double decimalLongitude) { 1472 this.decimalLongitude = decimalLongitude; 1473 } 1474 1475 @Nullable 1476 public Double getDecimalLatitude() { 1477 return decimalLatitude; 1478 } 1479 1480 public void setDecimalLatitude(@Nullable Double decimalLatitude) { 1481 this.decimalLatitude = decimalLatitude; 1482 } 1483 1484 /** 1485 * The uncertainty radius for lat/lon in meters. 1486 */ 1487 @Nullable 1488 public Double getCoordinateUncertaintyInMeters() { 1489 return coordinateUncertaintyInMeters; 1490 } 1491 1492 public void setCoordinateUncertaintyInMeters(@Nullable Double coordinateUncertaintyInMeters) { 1493 this.coordinateUncertaintyInMeters = coordinateUncertaintyInMeters; 1494 } 1495 1496 @Nullable 1497 public Double getCoordinatePrecision() { 1498 return coordinatePrecision; 1499 } 1500 1501 public void setCoordinatePrecision(Double coordinatePrecision) { 1502 this.coordinatePrecision = coordinatePrecision; 1503 } 1504 1505 /** 1506 * @deprecated to be removed in the public v2 of the API (see POR-3061) 1507 * The uncertainty for latitude in decimal degrees. 1508 * Note that the longitude degrees have a different accuracy in degrees which changes with latitude and is largest at the poles. 1509 */ 1510 @Nullable 1511 @Deprecated 1512 public Double getCoordinateAccuracy() { 1513 return coordinateAccuracy; 1514 } 1515 1516 public void setCoordinateAccuracy(@Nullable Double coordinateAccuracy) { 1517 this.coordinateAccuracy = coordinateAccuracy; 1518 } 1519 1520 /** 1521 * The geodetic datum for the interpreted decimal coordinates. 1522 * This is always WGS84 if a coordinate exists as we reproject other datums into WGS84. 1523 */ 1524 @Schema( 1525 description = "The geodetic datum for the interpreted decimal coordinates.\n\n" + 1526 "Coordinates are reprojected to WGS84 if they exist, so this field is either null or `WGS84`." 1527 ) 1528 @Nullable 1529 public String getGeodeticDatum() { 1530 if (decimalLatitude != null) { 1531 return GEO_DATUM; 1532 } 1533 return null; 1534 } 1535 1536 /** 1537 * This private method is needed for jackson deserialization only. 1538 */ 1539 private void setGeodeticDatum(String datum) { 1540 // ignore, we have a static WGS84 value 1541 } 1542 1543 /** 1544 * Elevation in meters usually above sea level (altitude). 1545 * </br> 1546 * The elevation is calculated using the equation: (minimumElevationInMeters + maximumElevationInMeters) / 2. 1547 */ 1548 @Nullable 1549 public Double getElevation() { 1550 return elevation; 1551 } 1552 1553 public void setElevation(@Nullable Double elevation) { 1554 this.elevation = elevation; 1555 } 1556 1557 /** 1558 * Elevation accuracy is the uncertainty for the elevation in meters. 1559 * </br> 1560 * The elevation accuracy is calculated using the equation: (maximumElevationInMeters - minimumElevationInMeters) / 2 1561 */ 1562 @Nullable 1563 public Double getElevationAccuracy() { 1564 return elevationAccuracy; 1565 } 1566 1567 public void setElevationAccuracy(@Nullable Double elevationAccuracy) { 1568 this.elevationAccuracy = elevationAccuracy; 1569 } 1570 1571 /** 1572 * Depth in meters below the surface. Complimentary to elevation, the depth can be 10 meters below the surface of a 1573 * lake in 1100m (=elevation). 1574 * </br> 1575 * The depth is calculated using the equation: (minimumDepthInMeters + maximumDepthInMeters) / 2. 1576 */ 1577 @Nullable 1578 public Double getDepth() { 1579 return depth; 1580 } 1581 1582 public void setDepth(@Nullable Double depth) { 1583 this.depth = depth; 1584 } 1585 1586 /** 1587 * Depth accuracy is the uncertainty for the depth in meters. 1588 * </br> 1589 * The depth accuracy is calculated using the equation: (maximumDepthInMeters - minimumDepthInMeters) / 2 1590 */ 1591 @Nullable 1592 public Double getDepthAccuracy() { 1593 return depthAccuracy; 1594 } 1595 1596 public void setDepthAccuracy(@Nullable Double depthAccuracy) { 1597 this.depthAccuracy = depthAccuracy; 1598 } 1599 1600 @Nullable 1601 public Continent getContinent() { 1602 return continent; 1603 } 1604 1605 public void setContinent(@Nullable Continent continent) { 1606 this.continent = continent; 1607 } 1608 1609 @Schema( 1610 description = "The 2-letter country code (as per ISO-3166-1) of the country, territory or area in which the " + 1611 "occurrence was recorded.", 1612 externalDocs = @ExternalDocumentation( 1613 description = "Darwin Core definition", 1614 url = "https://rs.tdwg.org/dwc/terms/countryCode" 1615 ) 1616 ) 1617 @Nullable 1618 @JsonProperty("countryCode") 1619 public Country getCountry() { 1620 return country; 1621 } 1622 1623 public void setCountry(@Nullable Country country) { 1624 this.country = country; 1625 } 1626 1627 @Nullable 1628 @JsonProperty("gbifRegion") 1629 public GbifRegion getGbifRegion() { 1630 return Optional.ofNullable(country).map(Country::getGbifRegion).orElse(null); 1631 } 1632 1633 private void setGbifRegion(String gbifRegion) { 1634 // ignore, setter only to avoid JSON being written into the fields map 1635 } 1636 1637 /** 1638 * Renders the country title as a JSON property country in addition to the ISO 3166 2 letter countryCode being 1639 * serialized by the regular country Java property. 1640 * Made private to use it only for JSON serialization and not within Java code. 1641 */ 1642 @Schema( 1643 description = "The title (as per ISO-3166-1) of the country, territory or area in which the " + 1644 "occurrence was recorded.", 1645 externalDocs = @ExternalDocumentation( 1646 description = "Darwin Core definition", 1647 url = "https://rs.tdwg.org/dwc/terms/country" 1648 ) 1649 ) 1650 @Nullable 1651 @JsonProperty("country") 1652 private String getCountryTitle() { 1653 return country == null ? null : country.getTitle(); 1654 } 1655 1656 private void setCountryTitle(String country) { 1657 // ignore, setter only to avoid JSON being written into the fields map 1658 } 1659 1660 @Nullable 1661 public String getStateProvince() { 1662 return stateProvince; 1663 } 1664 1665 public void setStateProvince(@Nullable String stateProvince) { 1666 this.stateProvince = stateProvince; 1667 } 1668 1669 @Nullable 1670 public String getWaterBody() { 1671 return waterBody; 1672 } 1673 1674 public void setWaterBody(@Nullable String waterBody) { 1675 this.waterBody = waterBody; 1676 } 1677 1678 /** 1679 * The distance in metres from a known centroid, e.g. a country centroid. 1680 */ 1681 public Double getDistanceFromCentroidInMeters() { 1682 return distanceFromCentroidInMeters; 1683 } 1684 1685 public void setDistanceFromCentroidInMeters(Double distanceFromCentroidInMeters) { 1686 this.distanceFromCentroidInMeters = distanceFromCentroidInMeters; 1687 } 1688 1689 @Nullable 1690 public String getHigherGeography() { 1691 return higherGeography; 1692 } 1693 1694 public void setHigherGeography(String higherGeography) { 1695 this.higherGeography = higherGeography; 1696 } 1697 1698 @Nullable 1699 public String getGeoreferencedBy() { 1700 return georeferencedBy; 1701 } 1702 1703 public void setGeoreferencedBy(String georeferencedBy) { 1704 this.georeferencedBy = georeferencedBy; 1705 } 1706 1707 /** 1708 * The full year of the event date. 1709 * 1710 * @return the year of the event date 1711 */ 1712 @Min(1500) 1713 @Max(2030) 1714 @Nullable 1715 public Integer getYear() { 1716 return year; 1717 } 1718 1719 public void setYear(@Nullable Integer year) { 1720 this.year = year; 1721 } 1722 1723 /** 1724 * The month of the year of the event date starting with zero for january following {@link Date}. 1725 * 1726 * @return the month of the event date 1727 */ 1728 @Min(1) 1729 @Max(12) 1730 @Nullable 1731 public Integer getMonth() { 1732 return month; 1733 } 1734 1735 public void setMonth(@Nullable Integer month) { 1736 this.month = month; 1737 } 1738 1739 /** 1740 * The day of the month of the event date. 1741 * 1742 * @return the day of the event date 1743 */ 1744 @Min(1) 1745 @Max(31) 1746 @Nullable 1747 public Integer getDay() { 1748 return day; 1749 } 1750 1751 public void setDay(@Nullable Integer day) { 1752 this.day = day; 1753 } 1754 1755 /** 1756 * The date the occurrence was recorded or collected. 1757 */ 1758 @Nullable 1759 public IsoDateInterval getEventDate() { 1760 return eventDate == null ? null : new IsoDateInterval(eventDate.getFrom(), eventDate.getTo()); 1761 } 1762 1763 public void setEventDate(@Nullable IsoDateInterval eventDate) { 1764 this.eventDate = eventDate == null ? null : new IsoDateInterval(eventDate.getFrom(), eventDate.getTo()); 1765 } 1766 1767 /** 1768 * The earliest integer day of the year of the event. 1769 * 1770 * @return the earliest integer day of the event date 1771 */ 1772 @Min(1) 1773 @Max(366) 1774 @Nullable 1775 public Integer getStartDayOfYear() { 1776 return startDayOfYear; 1777 } 1778 1779 public void setStartDayOfYear(@Nullable Integer startDayOfYear) { 1780 this.startDayOfYear = startDayOfYear; 1781 } 1782 1783 /** 1784 * The latest integer day of the year of the event. 1785 * 1786 * @return the latest integer day of the event date 1787 */ 1788 @Min(1) 1789 @Max(366) 1790 @Nullable 1791 public Integer getEndDayOfYear() { 1792 return endDayOfYear; 1793 } 1794 1795 public void setEndDayOfYear(@Nullable Integer endDayOfYear) { 1796 this.endDayOfYear = endDayOfYear; 1797 } 1798 1799 @Nullable 1800 public String getTypeStatus() { 1801 return typeStatus; 1802 } 1803 1804 public void setTypeStatus(@Nullable String typeStatus) { 1805 this.typeStatus = typeStatus; 1806 } 1807 1808 /** 1809 * The scientific name the type status of this specimen applies to. 1810 */ 1811 @Nullable 1812 public String getTypifiedName() { 1813 return typifiedName; 1814 } 1815 1816 public void setTypifiedName(@Nullable String typifiedName) { 1817 this.typifiedName = typifiedName; 1818 } 1819 1820 /** 1821 * A set of issues found for this occurrence. 1822 */ 1823 @NotNull 1824 public Set<OccurrenceIssue> getIssues() { 1825 return issues; 1826 } 1827 1828 public void setIssues(Set<OccurrenceIssue> issues) { 1829 Objects.requireNonNull(issues, "Issues cannot be null"); 1830 EnumSet<OccurrenceIssue> set = EnumSet.noneOf(OccurrenceIssue.class); 1831 set.addAll(issues); 1832 this.issues = set; 1833 } 1834 1835 public void addIssue(OccurrenceIssue issue) { 1836 Objects.requireNonNull(issue, "Issue needs to be specified"); 1837 issues.add(issue); 1838 } 1839 1840 /** 1841 * The interpreted dc:modified from the verbatim source data. 1842 * Ideally indicating when a record was last modified in the source. 1843 */ 1844 @Nullable 1845 public Date getModified() { 1846 return modified == null ? null : new Date(modified.getTime()); 1847 } 1848 1849 public void setModified(@Nullable Date modified) { 1850 this.modified = modified == null ? null : new Date(modified.getTime()); 1851 } 1852 1853 /** 1854 * The date this occurrence last went through the interpretation phase of the GBIF indexing. 1855 */ 1856 @Nullable 1857 public Date getLastInterpreted() { 1858 return lastInterpreted == null ? null : new Date(lastInterpreted.getTime()); 1859 } 1860 1861 public void setLastInterpreted(@Nullable Date lastInterpreted) { 1862 this.lastInterpreted = lastInterpreted == null ? null : new Date(lastInterpreted.getTime()); 1863 } 1864 1865 /** 1866 * An external link to more details, the records "homepage". 1867 */ 1868 @Nullable 1869 public URI getReferences() { 1870 return references; 1871 } 1872 1873 public void setReferences(URI references) { 1874 this.references = references; 1875 } 1876 1877 /** 1878 * A number or enumeration value for the quantity of organisms. 1879 */ 1880 @Nullable 1881 public Double getOrganismQuantity() { 1882 return organismQuantity; 1883 } 1884 1885 public void setOrganismQuantity(@Nullable Double organismQuantity) { 1886 this.organismQuantity = organismQuantity; 1887 } 1888 1889 /** 1890 * The type of quantification system used for the quantity of organisms. 1891 */ 1892 @Nullable 1893 public String getOrganismQuantityType() { 1894 return organismQuantityType; 1895 } 1896 1897 public void setOrganismQuantityType(@Nullable String organismQuantityType) { 1898 this.organismQuantityType = organismQuantityType; 1899 } 1900 1901 /** 1902 * The unit of measurement of the size (time duration, length, area, or volume) of a sample in a sampling event. 1903 */ 1904 @Nullable 1905 public String getSampleSizeUnit() { 1906 return sampleSizeUnit; 1907 } 1908 1909 public void setSampleSizeUnit(@Nullable String sampleSizeUnit) { 1910 this.sampleSizeUnit = sampleSizeUnit; 1911 } 1912 1913 /** 1914 * A numeric value for a measurement of the size (time duration, length, area, or volume) of a sample in a sampling event. 1915 */ 1916 @Nullable 1917 public Double getSampleSizeValue() { 1918 return sampleSizeValue; 1919 } 1920 1921 public void setSampleSizeValue(@Nullable Double sampleSizeValue) { 1922 this.sampleSizeValue = sampleSizeValue; 1923 } 1924 1925 /** 1926 * Calculated filed organismQuantity / sampleSizeValue, if the type is identical 1927 */ 1928 @Nullable 1929 public Double getRelativeOrganismQuantity() { 1930 return relativeOrganismQuantity; 1931 } 1932 1933 public void setRelativeOrganismQuantity(@Nullable Double relativeOrganismQuantity) { 1934 this.relativeOrganismQuantity = relativeOrganismQuantity; 1935 } 1936 1937 /** 1938 * Flag occurrence when associatedSequences/extension exists 1939 */ 1940 public boolean getIsSequenced() { 1941 return isSequenced; 1942 } 1943 1944 public void setIsSequenced(boolean isSequenced) { 1945 this.isSequenced = isSequenced; 1946 } 1947 1948 /** 1949 * A list (concatenated and separated) of identifiers (publication, global unique identifier, URI) of genetic 1950 * sequence information associated with the material entity. 1951 */ 1952 @Nullable 1953 public String getAssociatedSequences() { 1954 return associatedSequences; 1955 } 1956 1957 public void setAssociatedSequences(String associatedSequences) { 1958 this.associatedSequences = associatedSequences; 1959 } 1960 1961 /** 1962 * Applied license to the occurrence record or dataset to which this record belongs to. 1963 */ 1964 @NotNull 1965 public License getLicense() { 1966 return license; 1967 } 1968 1969 public void setLicense(License license) { 1970 this.license = license; 1971 } 1972 1973 @NotNull 1974 public List<Identifier> getIdentifiers() { 1975 return identifiers; 1976 } 1977 1978 public void setIdentifiers(List<Identifier> identifiers) { 1979 this.identifiers = identifiers; 1980 } 1981 1982 @NotNull 1983 public List<MediaObject> getMedia() { 1984 return media; 1985 } 1986 1987 public void setMedia(List<MediaObject> media) { 1988 this.media = media; 1989 } 1990 1991 @NotNull 1992 public List<MeasurementOrFact> getFacts() { 1993 return facts; 1994 } 1995 1996 public void setFacts(List<MeasurementOrFact> facts) { 1997 this.facts = facts; 1998 } 1999 2000 @NotNull 2001 public List<OccurrenceRelation> getRelations() { 2002 return relations; 2003 } 2004 2005 public void setRelations(List<OccurrenceRelation> relations) { 2006 this.relations = relations; 2007 } 2008 2009 @NotNull 2010 public List<AgentIdentifier> getRecordedByIds() { 2011 return recordedByIds; 2012 } 2013 2014 public void setRecordedByIds(List<AgentIdentifier> recordedByIds) { 2015 this.recordedByIds = recordedByIds; 2016 } 2017 2018 @NotNull 2019 public List<AgentIdentifier> getIdentifiedByIds() { 2020 return identifiedByIds; 2021 } 2022 2023 public void setIdentifiedByIds(List<AgentIdentifier> identifiedByIds) { 2024 this.identifiedByIds = identifiedByIds; 2025 } 2026 2027 @NotNull 2028 public Gadm getGadm() { 2029 return gadm; 2030 } 2031 2032 public void setGadm(Gadm gadm) { 2033 this.gadm = gadm; 2034 } 2035 2036 @Nullable 2037 @Experimental 2038 public String getInstitutionKey() { 2039 return institutionKey; 2040 } 2041 2042 public void setInstitutionKey(String institutionKey) { 2043 this.institutionKey = institutionKey; 2044 } 2045 2046 @Nullable 2047 @Experimental 2048 public String getCollectionKey() { 2049 return collectionKey; 2050 } 2051 2052 public void setCollectionKey(String collectionKey) { 2053 this.collectionKey = collectionKey; 2054 } 2055 2056 public boolean getIsInCluster() { 2057 return isInCluster; 2058 } 2059 2060 public void setIsInCluster(boolean isInCluster) { 2061 this.isInCluster = isInCluster; 2062 } 2063 2064 @Nullable 2065 public String getDegreeOfEstablishment() { 2066 return degreeOfEstablishment; 2067 } 2068 2069 public void setDegreeOfEstablishment(String degreeOfEstablishment) { 2070 this.degreeOfEstablishment = degreeOfEstablishment; 2071 } 2072 2073 @Nullable 2074 public String getPathway() { 2075 return pathway; 2076 } 2077 2078 public void setPathway(String pathway) { 2079 this.pathway = pathway; 2080 } 2081 2082 public String getDatasetID() { 2083 return datasetID; 2084 } 2085 2086 public void setDatasetID(String datasetID) { 2087 this.datasetID = datasetID; 2088 } 2089 2090 public String getDatasetName() { 2091 return datasetName; 2092 } 2093 2094 public void setDatasetName(String datasetName) { 2095 this.datasetName = datasetName; 2096 } 2097 2098 public String getOtherCatalogNumbers() { 2099 return otherCatalogNumbers; 2100 } 2101 2102 public void setOtherCatalogNumbers(String otherCatalogNumbers) { 2103 this.otherCatalogNumbers = otherCatalogNumbers; 2104 } 2105 2106 @Nullable 2107 public String getEarliestEonOrLowestEonothem() { 2108 return earliestEonOrLowestEonothem; 2109 } 2110 2111 public void setEarliestEonOrLowestEonothem(String earliestEonOrLowestEonothem) { 2112 this.earliestEonOrLowestEonothem = earliestEonOrLowestEonothem; 2113 } 2114 2115 @Nullable 2116 public String getLatestEonOrHighestEonothem() { 2117 return latestEonOrHighestEonothem; 2118 } 2119 2120 public void setLatestEonOrHighestEonothem(String latestEonOrHighestEonothem) { 2121 this.latestEonOrHighestEonothem = latestEonOrHighestEonothem; 2122 } 2123 2124 @Nullable 2125 public String getEarliestEraOrLowestErathem() { 2126 return earliestEraOrLowestErathem; 2127 } 2128 2129 public void setEarliestEraOrLowestErathem(String earliestEraOrLowestErathem) { 2130 this.earliestEraOrLowestErathem = earliestEraOrLowestErathem; 2131 } 2132 2133 @Nullable 2134 public String getLatestEraOrHighestErathem() { 2135 return latestEraOrHighestErathem; 2136 } 2137 2138 public void setLatestEraOrHighestErathem(String latestEraOrHighestErathem) { 2139 this.latestEraOrHighestErathem = latestEraOrHighestErathem; 2140 } 2141 2142 @Nullable 2143 public String getEarliestPeriodOrLowestSystem() { 2144 return earliestPeriodOrLowestSystem; 2145 } 2146 2147 public void setEarliestPeriodOrLowestSystem(String earliestPeriodOrLowestSystem) { 2148 this.earliestPeriodOrLowestSystem = earliestPeriodOrLowestSystem; 2149 } 2150 2151 @Nullable 2152 public String getLatestPeriodOrHighestSystem() { 2153 return latestPeriodOrHighestSystem; 2154 } 2155 2156 public void setLatestPeriodOrHighestSystem(String latestPeriodOrHighestSystem) { 2157 this.latestPeriodOrHighestSystem = latestPeriodOrHighestSystem; 2158 } 2159 2160 @Nullable 2161 public String getEarliestEpochOrLowestSeries() { 2162 return earliestEpochOrLowestSeries; 2163 } 2164 2165 public void setEarliestEpochOrLowestSeries(String earliestEpochOrLowestSeries) { 2166 this.earliestEpochOrLowestSeries = earliestEpochOrLowestSeries; 2167 } 2168 2169 @Nullable 2170 public String getLatestEpochOrHighestSeries() { 2171 return latestEpochOrHighestSeries; 2172 } 2173 2174 public void setLatestEpochOrHighestSeries(String latestEpochOrHighestSeries) { 2175 this.latestEpochOrHighestSeries = latestEpochOrHighestSeries; 2176 } 2177 2178 @Nullable 2179 public String getEarliestAgeOrLowestStage() { 2180 return earliestAgeOrLowestStage; 2181 } 2182 2183 public void setEarliestAgeOrLowestStage(String earliestAgeOrLowestStage) { 2184 this.earliestAgeOrLowestStage = earliestAgeOrLowestStage; 2185 } 2186 2187 @Nullable 2188 public String getLatestAgeOrHighestStage() { 2189 return latestAgeOrHighestStage; 2190 } 2191 2192 public void setLatestAgeOrHighestStage(String latestAgeOrHighestStage) { 2193 this.latestAgeOrHighestStage = latestAgeOrHighestStage; 2194 } 2195 2196 @Nullable 2197 public String getLowestBiostratigraphicZone() { 2198 return lowestBiostratigraphicZone; 2199 } 2200 2201 public void setLowestBiostratigraphicZone(String lowestBiostratigraphicZone) { 2202 this.lowestBiostratigraphicZone = lowestBiostratigraphicZone; 2203 } 2204 2205 @Nullable 2206 public String getHighestBiostratigraphicZone() { 2207 return highestBiostratigraphicZone; 2208 } 2209 2210 public void setHighestBiostratigraphicZone(String highestBiostratigraphicZone) { 2211 this.highestBiostratigraphicZone = highestBiostratigraphicZone; 2212 } 2213 2214 @Nullable 2215 public String getGroup() { 2216 return group; 2217 } 2218 2219 public void setGroup(String group) { 2220 this.group = group; 2221 } 2222 2223 @Nullable 2224 public String getFormation() { 2225 return formation; 2226 } 2227 2228 public void setFormation(String formation) { 2229 this.formation = formation; 2230 } 2231 2232 @Nullable 2233 public String getMember() { 2234 return member; 2235 } 2236 2237 public void setMember(String member) { 2238 this.member = member; 2239 } 2240 2241 @Nullable 2242 public String getBed() { 2243 return bed; 2244 } 2245 2246 public void setBed(String bed) { 2247 this.bed = bed; 2248 } 2249 2250 public String getRecordedBy() { 2251 return recordedBy; 2252 } 2253 2254 public void setRecordedBy(String recordedBy) { 2255 this.recordedBy = recordedBy; 2256 } 2257 2258 public String getIdentifiedBy() { 2259 return identifiedBy; 2260 } 2261 2262 public void setIdentifiedBy(String identifiedBy) { 2263 this.identifiedBy = identifiedBy; 2264 } 2265 2266 public String getPreparations() { 2267 return preparations; 2268 } 2269 2270 public void setPreparations(String preparations) { 2271 this.preparations = preparations; 2272 } 2273 2274 public String getSamplingProtocol() { 2275 return samplingProtocol; 2276 } 2277 2278 public void setSamplingProtocol(String samplingProtocol) { 2279 this.samplingProtocol = samplingProtocol; 2280 } 2281 2282 /** 2283 * Convenience method checking if any spatial validation rule has not passed. 2284 * Primarily used to indicate that the record should not be displayed on a map. 2285 */ 2286 @JsonIgnore 2287 public boolean hasSpatialIssue() { 2288 for (OccurrenceIssue rule : OccurrenceIssue.GEOSPATIAL_RULES) { 2289 if (issues.contains(rule)) { 2290 return true; 2291 } 2292 } 2293 return false; 2294 } 2295 2296 /** 2297 * This private method is only for serialization via jackson and not exposed anywhere else! 2298 * It maps the verbatimField terms into properties with their simple name or qualified names for UnknownTerms. 2299 */ 2300 @JsonAnyGetter 2301 private Map<String, String> jsonVerbatimFields() { 2302 Map<String, String> extendedProps = new HashMap<>(); 2303 for (Map.Entry<Term, String> prop : getVerbatimFields().entrySet()) { 2304 Term t = prop.getKey(); 2305 if (t instanceof UnknownTerm || PROPERTIES.contains(t.simpleName())) { 2306 extendedProps.put(t.qualifiedName(), prop.getValue()); 2307 } else { 2308 // render all terms in controlled enumerations as simple names only - unless we have a property of that name already! 2309 extendedProps.put(t.simpleName(), prop.getValue()); 2310 } 2311 } 2312 return extendedProps; 2313 } 2314}