CropTraitObservationData.java

/*
 * Copyright 2026 Global Crop Diversity Trust
 * Licensed under the Apache License, Version 2.0
 * See LICENSE file in project root folder or http://www.apache.org/licenses/LICENSE-2.0
 */

package org.gringlobal.model;

import java.time.LocalDate;
import java.time.format.DateTimeParseException;
import java.util.Objects;
import javax.persistence.*;

import org.genesys.blocks.model.Copyable;

import org.gringlobal.api.exception.InvalidApiUsageException;
import org.gringlobal.custom.validation.javax.ValidCropTraitObservationData;
import org.gringlobal.model.community.CommunityCodeValues;

import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;

import com.fasterxml.jackson.annotation.JsonIdentityInfo;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.annotation.ObjectIdGenerators;

@Entity
@Table(name = "crop_trait_observation_data")
@JsonIdentityInfo(scope = CropTraitObservationData.class, generator = ObjectIdGenerators.PropertyGenerator.class, property = "id")
@Getter
@Setter
@NoArgsConstructor
@ValidCropTraitObservationData
public class CropTraitObservationData extends CooperatorOwnedModel implements Copyable<CropTraitObservationData> {

	private static final long serialVersionUID = 6269329970599931123L;

	@Id
	@JsonProperty
	@GeneratedValue(strategy = GenerationType.IDENTITY)
	@Column(name = "crop_trait_observation_data_id", columnDefinition = "int")
	private Long id;

	@ManyToOne(fetch = FetchType.LAZY, cascade = {}, optional = false)
	@JoinColumn(name = "method_id", nullable = false)
	private Method method;

	@ManyToOne(fetch = FetchType.LAZY, cascade = {}, optional = false)
	@JoinColumn(name = "inventory_id", nullable = false)
	private Inventory inventory;

	@ManyToOne(fetch = FetchType.LAZY, cascade = {}, optional = false)
	@JoinColumn(name = "crop_trait_id", nullable = false)
	private CropTrait cropTrait;

	@ManyToOne(fetch = FetchType.LAZY, cascade = {})
	@JoinColumn(name = "crop_trait_code_id")
	private CropTraitCode cropTraitCode;

	@Basic
	private int individual;

	@Basic
	@Column(name = "numeric_value")
	private Double numericValue;

	@Basic
	@Column(name = "string_value")
	private String stringValue;

	@ManyToOne(fetch = FetchType.LAZY, cascade = {})
	@JoinColumn(name = "crop_trait_observation_id")
	private CropTraitObservation cropTraitObservation;

	@Basic
	@Column
	@Lob
	private String note;

	@PrePersist
	@PreUpdate
	private void preupdate() {
		/*
		* Make sure to keep this in sync with CropTraitObservation#preupdate
		*/
		assert cropTrait != null;

		if ("Y".equals(cropTrait.getIsCoded())) {
			if (cropTraitCode != null && !cropTrait.getId().equals(cropTraitCode.getCropTrait().getId())) {
				throw new InvalidApiUsageException("cropTraitCode does not belong to cropTrait");
			}
			if (numericValue != null) {
				throw new InvalidApiUsageException("cropTrait is coded");
			} else if (stringValue != null) {
				throw new InvalidApiUsageException("cropTrait is coded");
			};
		} else {
			if (cropTraitCode != null) {
				throw new InvalidApiUsageException("cropTrait is not coded");
			}
			if (numericValue != null) {
				if (!Objects.equals(cropTrait.getDataTypeCode(), CommunityCodeValues.CROP_TRAIT_DATA_TYPE_NUMERIC.value)) {
					throw new InvalidApiUsageException("cropTrait is not numeric");
				}
				if (stringValue != null) {
					throw new InvalidApiUsageException("cropTrait is numeric");
				};
				if (cropTrait.getNumericMinimum() != null && numericValue < cropTrait.getNumericMinimum()) {
					throw new InvalidApiUsageException("Out of numeric range");
				}
				if (cropTrait.getNumericMaximum() != null && numericValue > cropTrait.getNumericMaximum()) {
					throw new InvalidApiUsageException("Out of numeric range");
				}
			}
			if (stringValue != null) {
				if (Objects.equals(cropTrait.getDataTypeCode(), CommunityCodeValues.CROP_TRAIT_DATA_TYPE_NUMERIC.value)) {
					throw new InvalidApiUsageException("cropTrait is numeric");
				}
				if (numericValue != null) {
					throw new InvalidApiUsageException("cropTrait is not numeric");
				}
				if (Objects.equals(cropTrait.getDataTypeCode(), CommunityCodeValues.CROP_TRAIT_DATA_TYPE_DATE.value)) {
					try {
						// Ensure stringValue is ISO date yyyy-MM-dd
						LocalDate.parse(stringValue);
					} catch (DateTimeParseException e) {
						throw new InvalidApiUsageException("stringValue must be in yyyy-MM-dd format");
					}
				}
				if (Objects.equals(cropTrait.getDataTypeCode(), CommunityCodeValues.CROP_TRAIT_DATA_TYPE_CHAR.value)) {
					// OK
				}
				if (cropTrait.getMaxLength() != null && stringValue.length() > cropTrait.getMaxLength())
					throw new InvalidApiUsageException("stringValue longer than crop trait maxLength " + cropTrait.getMaxLength());
				numericValue = null;
			}
		}
	}

	public CropTraitObservationData(final Long id) {
		this.id = id;
	}

	@Override
	public void lazyLoad() {
		super.lazyLoad();
		lazyLoad(this.method);
		lazyLoad(this.inventory);
		lazyLoad(this.cropTrait);
		lazyLoad(this.cropTraitCode);
	}

	@Override
	public boolean canEqual(Object other) {
		return other instanceof CropTraitObservationData;
	}
}