Descriptor.java

/*
 * Copyright 2018 Global Crop Diversity Trust
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.genesys.server.model.traits;

import com.fasterxml.jackson.annotation.*;
import lombok.Getter;
import lombok.NoArgsConstructor;
import lombok.Setter;
import org.genesys.blocks.annotations.NotCopyable;
import org.genesys.blocks.auditlog.annotations.Audited;
import org.genesys.blocks.model.*;
import org.genesys.blocks.security.model.AclAwareModel;
import org.genesys.filerepository.model.RepositoryImage;
import org.genesys.server.model.Partner;
import org.genesys.server.model.PublishState;
import org.genesys.server.model.dataset.Dataset;
import org.genesys.server.model.impl.Crop;
import org.genesys.server.model.impl.TranslatedUuidModel;
import org.genesys.server.model.vocab.ControlledVocabulary;
import org.genesys.server.model.vocab.VocabularyTerm;
import org.springframework.dao.DataIntegrityViolationException;
import org.springframework.data.elasticsearch.annotations.Document;
import org.springframework.data.elasticsearch.annotations.Field;
import org.springframework.data.elasticsearch.annotations.FieldType;

import javax.persistence.*;
import javax.validation.constraints.NotNull;
import javax.validation.constraints.Size;
import java.time.Instant;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;

/**
 * A single descriptor definition belonging to a single "owner".
 *
 * @author Matija Obreza
 */
@Entity
@Cacheable
@Table(name = "descriptor")
@Audited
@Document(indexName = "descriptor")
@JsonIdentityInfo(generator = ObjectIdGenerators.PropertyGenerator.class, property = "uuid", scope = Descriptor.class)
@Getter
@Setter
@NoArgsConstructor
public class Descriptor extends TranslatedUuidModel<DescriptorLang, Descriptor> implements SelfCleaning, Publishable, Copyable<Descriptor>, AclAwareModel {

	/** The Constant serialVersionUID. */
	private static final long serialVersionUID = 7307818236681549484L;

	/**
	 * Descriptor data types.
	 */
	public static enum DataType {
		/** The numeric. */
		NUMERIC,
		/** Free text. */
		TEXT,

		/** Scale (0-9 scale) type. */
		SCALE,
		/** Coded descriptors using controlled vocabularies. */
		CODED,
		/** Yes/No type. */
		BOOLEAN,

		/** Dates. */
		DATE
	}

	/**
	 * Descriptor classification.
	 */
	public static enum Category {

		/** Passport descriptor. */
		PASSPORT,

		/** Management descriptor. */
		MANAGEMENT,

		/** Environment and Site descriptor. */
		ENVIRONMENT,

		/** Characterization descriptor. */
		CHARACTERIZATION,

		/** Evaluation descriptor. */
		EVALUATION,

		/** Abiotic stress descriptor. */
		ABIOTICSTRESS,

		/** Biotic stress descriptor. */
		BIOTICSTRESS,

		/** Molecular marker descriptor. */
		MOLECULAR
	}

	/**
	 * User-specified version tag. E.g. "1.0", "1.1"
	 */
	@NotNull
	@Column(nullable = false, updatable = false)
	private String versionTag;

	/**
	 * Trait title in English.
	 */
	@NotNull
	@Column(nullable = false)
	private String title;

	/**
	 * Trait description in English.
	 */
	@Lob
	private String description;

	/** Descriptor data type. */
	@NotNull
	@Enumerated(EnumType.STRING)
	@Column(nullable = false)
	private DataType dataType;

	/** The key. */
	@NotNull
	@Column(name = "keyDescriptor", nullable = false)
	private boolean key;

	/** The publisher. */
	@Size(max = 200)
	@Column(length = 200)
	private String publisher;

	/** Not published by default. */
	@Enumerated(EnumType.ORDINAL)
	private PublishState state = PublishState.DRAFT;

	/** Allow only integers, no decimal numbers. */
	private Boolean integerOnly;

	/** Data constraints: minimum allowed numeric values. */
	@Column(name = "min_value")
	private Double minValue;

	/** Data constraints: maximum allowed numeric values. */
	@Column(name = "max_value")
	private Double maxValue;

	/** The uom. */
	@Size(max = 20)
	@Column(name = "uom", length = 20)
	private String uom;

	/** The preferred column name in databases and spreadsheets. */
	@Size(max = 50)
	@Column(name = "columnName", length = 50)
	private String columnName;

	/** The bibliographic citation. */
	@Column(name = "bibliographicCitation")
	@Lob
	private String bibliographicCitation;

	/** The shared controlled vocabulary used by the descriptor */
	@ManyToOne(cascade = {}, optional = true)
	@JoinColumn(name = "vocabularyId")
	@JsonIdentityReference(alwaysAsId = false)
	@JsonIgnoreProperties(value = { "terms" })
	@JsonView({ JsonViews.Public.class })
	@Field(type = FieldType.Nested)
	private ControlledVocabulary vocabulary;

	/**
	 * Vocabulary terms specific to this descriptor (99% of the cases).
	 */
	@OneToMany(fetch = FetchType.LAZY, cascade = { CascadeType.MERGE, CascadeType.PERSIST, CascadeType.REMOVE, CascadeType.REFRESH }, orphanRemoval = true)
	@JoinTable(name = "descriptor_term", joinColumns = @JoinColumn(name = "descriptorId"),
			// other side
			inverseJoinColumns = @JoinColumn(name = "termId"),
			// Index
			indexes = { @Index(columnList = "descriptorId, idx") },
			// unique constraints
			uniqueConstraints = { })
	@OrderColumn(name = "idx")
	@JsonView({ JsonViews.Public.class })
	@Field(type = FieldType.Nested)
	@JsonIgnoreProperties({ "vocabulary", "descriptor" })
	private List<VocabularyTerm> terms;

	/** The owner. */
	@NotNull
	@JsonIdentityReference(alwaysAsId = false)
	@ManyToOne(cascade = {}, optional = false)
	@JoinColumn(name = "partnerId", updatable = false)
	@JsonView({ JsonViews.Public.class })
	@Field(type = FieldType.Object)
	@JsonIgnoreProperties({ "institutes", "urls", "countryCodes", "description" })
	private Partner owner;

	/** The descriptor lists. */
	@JsonIdentityReference(alwaysAsId = false)
	@ManyToMany(fetch = FetchType.LAZY, mappedBy = "descriptors", cascade = { CascadeType.REFRESH })
	@JsonIgnore
	private List<DescriptorList> descriptorLists;

	/** The datasets. */
	@ManyToMany(fetch = FetchType.LAZY, cascade = {}, mappedBy = "descriptors")
	@JsonIgnore
	private List<Dataset> datasets;

	/** The crop. */
	@Size(max = Crop.CROP_SHORTNAME_LENGTH)
	@Column(name = "crop", length = Crop.CROP_SHORTNAME_LENGTH, nullable = true)
	private String crop;

	/** Descriptor classification. */
	@NotNull
	@Enumerated(EnumType.STRING)
	@Column(length = 20, nullable = false)
	private Category category;

	@OneToOne(fetch = FetchType.LAZY, cascade = { }, optional = true, orphanRemoval = false)
	@JoinColumn(name = "imageId", unique = true)
	private RepositoryImage image;

	@Column(name = "firstPublishedDate")
	@NotCopyable
	private Instant firstPublishedDate;

	/**
	 * Pre-persist, pre-update
	 */
	@PrePersist
	@PreUpdate
	private void preupdate() {
		if (Objects.isNull(firstPublishedDate) && Objects.equals(state, PublishState.PUBLISHED)) {
			firstPublishedDate = Instant.now();
		}
		trimStringsToNull();
	}
	
	@PreRemove
	private void preventRemoveWithImage() {
		if (image != null) {
			throw new DataIntegrityViolationException("Refusing to delete Descriptor with image");
		}
	}

	/**
	 * Owner is the ACL parent object for the descriptor
	 */
	@Override
	public AclAwareModel aclParentObject() {
		return null;
	}

	/**
	 * Instantiates a new descriptor.
	 *
	 * @param descriptor the descriptor
	 */
	public Descriptor(final Descriptor descriptor) {
		this(descriptor.getVersionTag(), descriptor.getTitle(), descriptor.description, descriptor.getDataType(), descriptor.getState(), descriptor.isKey(), descriptor
			.getIntegerOnly(), descriptor.getMinValue(), descriptor.getMaxValue(), descriptor.getColumnName(), descriptor.getUom(), descriptor.getVocabulary(), descriptor
				.getOwner(), descriptor.getDescriptorLists(), descriptor.getImage());
	}

	/**
	 * Instantiates a new descriptor.
	 *
	 * @param versionTag the version tag
	 * @param title the title
	 * @param description the description
	 * @param dataType the data type
	 * @param state the publish state
	 * @param key the key
	 * @param integerOnly the integer only
	 * @param minValue the min value
	 * @param maxValue the max value
	 * @param columnName the column name
	 * @param uom the uom
	 * @param vocabulary the vocabulary
	 * @param owner the owner
	 * @param descriptorLists the descriptor lists
	 * @param image the image
	 */
	public Descriptor(final String versionTag, final String title, final String description, final DataType dataType, final PublishState state, final boolean key,
			final Boolean integerOnly, final Double minValue, final Double maxValue, final String columnName, final String uom, final ControlledVocabulary vocabulary,
			final Partner owner, final List<DescriptorList> descriptorLists, final RepositoryImage image) {
		this.versionTag = versionTag;
		this.title = title;
		this.description = description;
		this.dataType = dataType;
		this.state = state;
		this.key = key;
		this.integerOnly = integerOnly;
		this.minValue = minValue;
		this.maxValue = maxValue;
		this.columnName = columnName;
		this.uom = uom;
		this.vocabulary = vocabulary;
		this.owner = owner;
		this.descriptorLists = descriptorLists;
		this.image = image;
	}

	/**
	 * Checks if is published.
	 *
	 * @return the published
	 */
	@Override
	public boolean isPublished() {
		return this.state == PublishState.PUBLISHED;
	}
	
	/*
	 * (non-Javadoc)
	 * @see org.genesys.blocks.model.Copyable#copy()
	 */
	@Override
	public Descriptor copy() {
		return null;
	}

	/*
	 * (non-Javadoc)
	 * @see org.genesys.blocks.model.Copyable#apply(java.lang.Object)
	 */
	@Override
	public Descriptor apply(final Descriptor source) {
		Copyable.super.apply(source);

		if (source.getTerms() != null) {
			if (this.terms == null) {
				this.terms = new ArrayList<>(source.getTerms());
			} else {
				this.terms.clear();
				this.terms.addAll(source.getTerms());
			}
		}

		// Check that CODED has terms or vocabulary
		if (this.dataType == DataType.CODED && (this.terms == null || this.terms.size() == 0) && this.vocabulary == null) {
			throw new DataIntegrityViolationException("Coded descriptor " + this.title + " requires terms or a vocabulary");
		}

		// Check that SCALE has terms
		if (this.dataType == DataType.SCALE && (this.terms == null || this.terms.size() == 0)) {
			throw new DataIntegrityViolationException("Scale descriptor " + this.title + " requires terms");
		}

		// Check that SCALE has min and max values
		if (this.dataType == DataType.SCALE && (this.minValue == null || this.maxValue == null)) {
			throw new DataIntegrityViolationException("Scale descriptor " + this.title + " requires min and max values.");
		}

		switch (this.dataType) {
		case BOOLEAN:
			this.setVocabulary(null);
			this.setMinValue(null);
			this.setMaxValue(null);
			break;
		case CODED:
			this.setMinValue(null);
			this.setMaxValue(null);
			break;
		case DATE:
		case TEXT:
		case NUMERIC:
		case SCALE:
			this.setVocabulary(null);
			break;
		default:
			break;
		}

		// When using a specified vocabulary
		if (this.vocabulary != null && this.terms != null) {
			// We don't use our terms
			this.terms.clear();
		}

		return this;
	}

	// @Override // TODO Introduce LazyLoadable
	public void lazyLoad() {
		if (this.getOwner() != null) {
			this.getOwner().getId();
			// System.err.println("Owner is " + this.getOwner());
		} else {
			// System.err.println("Owner is null");
		}

		if (this.getDescriptorLists() != null) {
			this.getDescriptorLists().size();
		}

		if (this.getDataType() == DataType.CODED || this.getDataType() == DataType.SCALE) {
			if (this.getVocabulary() != null) {
				this.getVocabulary().getId();
				if (this.getVocabulary().getTerms() != null) {
					this.getVocabulary().getTerms().size();
				}
				this.setTerms(null);
			}
			if (this.getTerms() != null) {
				this.getTerms().size();
			}
		} else {
			// clear terms list
			if (this.getTerms() != null) {
				this.getTerms().clear();
			}
		}

		if (this.getImage() != null) {
			this.getImage().getId();
		}
	}

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