org.sakaiproject.assignment.api.model.Assignment Maven / Gradle / Ivy
/**********************************************************************************
* $URL$
* $Id$
***********************************************************************************
*
* Copyright (c) 2003, 2004, 2005, 2006, 2007, 2008, 2009 The Sakai Foundation
*
* Licensed under the Educational Community 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.opensource.org/licenses/ECL-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.sakaiproject.assignment.api.model;
import java.time.Instant;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import javax.persistence.CascadeType;
import javax.persistence.CollectionTable;
import javax.persistence.Column;
import javax.persistence.ElementCollection;
import javax.persistence.Entity;
import javax.persistence.EnumType;
import javax.persistence.Enumerated;
import javax.persistence.GeneratedValue;
import javax.persistence.Id;
import javax.persistence.Index;
import javax.persistence.JoinColumn;
import javax.persistence.Lob;
import javax.persistence.MapKeyColumn;
import javax.persistence.OneToMany;
import javax.persistence.Table;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.NoArgsConstructor;
import lombok.ToString;
import org.hibernate.annotations.Cache;
import org.hibernate.annotations.CacheConcurrencyStrategy;
import org.hibernate.annotations.Fetch;
import org.hibernate.annotations.FetchMode;
import org.hibernate.annotations.GenericGenerator;
import org.hibernate.annotations.Type;
import com.fasterxml.jackson.annotation.JsonIdentityInfo;
import com.fasterxml.jackson.annotation.JsonManagedReference;
import com.fasterxml.jackson.annotation.ObjectIdGenerators;
/**
* Assignment represents a specific assignment for a specific section or class.
*
*
Important notes about Java java.util.Date timezone persistence
*
* * MySQL DATETIME vs TIMESTAMP
* - DATETIME is a date and time and has a much wider range than TIMESTAMP. MySQL's date operations
* can be used to perform date calculations. Probably the most notable trait is that the value that
* is stored will be the same when retrieved regardless of timezone info. This shifts the
* responsibility to the application to handling timezone info. Changing the server/jvm timezone
* would result in different times if not handled.
* It's best canonicalized with DATETIME is not affected by timezones.
* - TIMESTAMP is a specific point in time (uses 4 bytes making it more efficient in mysql).
* Values are converted from the current timezone to UTC for storage and then converted back
* from UTC to the current timezone. The current timezone is usually the mysql servers timezone.
* Changing the server/jvm timezone wouldn't matter.
* - Connector/J useLegacyDatetimeCode is used for backwards compatibility, handling timezone info
* the way it always has. Newer versions will probably change this to false. The general rule of thumb
* is to use whatever your database was running with (i.e. true), and new databases with new data to use false.
* - Connector/J 5.1.x useLegacyDatetimeCode=true and > 5.1 is false
* - MariaDB Connector/J 1.1.7+ useLegacyDatetimeCode=true
* * Hibernate
* - JPA @Temporal creates a TIMESTAMP type in MYSQL by default
* - java 8 time is supported natively in Hibernate 5
* - where @Temporal is implicit and therefore not needed
* - LocalDate implicitly @Temporal(TemporalType.Date)
* - LocalTime implicitly @Temporal(TemporalType.Time)
* - LocalDateTime implicitly @Temporal(TemporalType.Timestamp)
* - java 8 time Instant is not a supported Type in Hibernate < 5
* - So we use a custom type called org.sakaiproject.springframework.orm.hibernate.type.InstantType,
* which stores the time consistent with the use of Instant in UTC in a DATETIME field.
* This can be removed after upgrading to Hibernate 5.
*
*/
@Entity
@Cache(usage = CacheConcurrencyStrategy.READ_WRITE)
@Table(name = "ASN_ASSIGNMENT", indexes = {
@Index(name = "IDX_ASN_ASSIGNMENT_CONTEXT", columnList = "CONTEXT")
})
@Data
@NoArgsConstructor
@ToString(exclude = {"authors", "submissions", "groups", "properties", "attachments"})
@EqualsAndHashCode(of = "id")
@JsonIdentityInfo(generator = ObjectIdGenerators.PropertyGenerator.class, property = "id")
public class Assignment {
@Id
@Column(name = "ASSIGNMENT_ID", length = 36, nullable = false)
@GeneratedValue(generator = "uuid")
@GenericGenerator(name = "uuid", strategy = "uuid2")
private String id;
@Column(name = "TITLE")
private String title;
@Lob
@Column(name = "INSTRUCTIONS", length = 65535)
private String instructions;
@Column(name = "CONTEXT", length = 99, nullable = false)
private String context;
@Column(name = "SECTION")
private String section;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "CREATED_DATE", nullable = false)
private Instant dateCreated;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "MODIFIED_DATE")
private Instant dateModified;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "VISIBLE_DATE")
private Instant visibleDate;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "OPEN_DATE")
private Instant openDate;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "DUE_DATE")
private Instant dueDate;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "CLOSE_DATE")
private Instant closeDate;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "DROP_DEAD_DATE")
private Instant dropDeadDate;
@Column(name = "MODIFIER", length = 99)
private String modifier;
@Column(name = "AUTHOR", length = 99)
private String author;
@Column(name = "DRAFT", nullable = false)
private Boolean draft = Boolean.FALSE;
@Column(name = "DELETED")
private Boolean deleted = Boolean.FALSE;
@Column(name = "HIDE_DUE_DATE")
private Boolean hideDueDate = Boolean.FALSE;
@Column(name = "IS_GROUP")
private Boolean isGroup = Boolean.FALSE;
@Column(name = "POSITION")
private Integer position;
@OneToMany(mappedBy = "assignment", cascade = CascadeType.ALL, orphanRemoval = true)
@JsonManagedReference
private Set submissions = new HashSet<>();
@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE)
@ElementCollection
@MapKeyColumn(name = "NAME")
@Lob
@Column(name = "VALUE", length = 65535)
@CollectionTable(name = "ASN_ASSIGNMENT_PROPERTIES", joinColumns = @JoinColumn(name = "ASSIGNMENT_ID"))
@Fetch(FetchMode.SUBSELECT)
private Map properties = new HashMap<>();
@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE)
@ElementCollection
@CollectionTable(name = "ASN_ASSIGNMENT_GROUPS", joinColumns = @JoinColumn(name = "ASSIGNMENT_ID"), indexes = @Index(columnList = "ASSIGNMENT_ID"))
@Fetch(FetchMode.SUBSELECT)
@Column(name = "GROUP_ID")
private Set groups = new HashSet<>();
@Cache(usage = CacheConcurrencyStrategy.NONSTRICT_READ_WRITE)
@ElementCollection
@CollectionTable(name = "ASN_ASSIGNMENT_ATTACHMENTS", joinColumns = @JoinColumn(name = "ASSIGNMENT_ID"), indexes = @Index(columnList = "ASSIGNMENT_ID"))
@Fetch(FetchMode.SUBSELECT)
@Column(name = "ATTACHMENT", length = 1024)
private Set attachments = new HashSet<>();
@Enumerated(value = EnumType.STRING)
@Column(name = "ACCESS_TYPE", nullable = false)
private Access typeOfAccess = Access.SITE;
@Column(name = "HONOR_PLEDGE")
private Boolean honorPledge = Boolean.FALSE;
@Enumerated
@Column(name = "SUBMISSION_TYPE")
private SubmissionType typeOfSubmission = SubmissionType.ASSIGNMENT_SUBMISSION_TYPE_NONE;
@Enumerated
@Column(name = "GRADE_TYPE")
private GradeType typeOfGrade = GradeType.GRADE_TYPE_NONE;
@Column(name = "MAX_GRADE_POINT")
private Integer maxGradePoint;
@Column(name = "SCALE_FACTOR")
private Integer scaleFactor;
@Column(name = "INDIVIDUALLY_GRADED")
private Boolean individuallyGraded;
@Column(name = "RELEASE_GRADES")
private Boolean releaseGrades = Boolean.FALSE;
@Column(name = "ALLOW_ATTACHMENTS")
private Boolean allowAttachments;
@Column(name = "ALLOW_PEER_ASSESSMENT")
private Boolean allowPeerAssessment = Boolean.FALSE;
@Type(type = "org.sakaiproject.springframework.orm.hibernate.type.InstantType")
@Column(name = "PEER_ASSESSMENT_PERIOD_DATE")
private Instant peerAssessmentPeriodDate;
@Column(name = "PEER_ASSESSMENT_ANON_EVAL")
private Boolean peerAssessmentAnonEval;
@Column(name = "PEER_ASSESSMENT_STUDENT_REVIEW")
private Boolean peerAssessmentStudentReview = Boolean.FALSE;
@Column(name = "PEER_ASSESSMENT_NUMBER_REVIEW")
private Integer peerAssessmentNumberReviews;
@Lob
@Column(name = "PEER_ASSESSMENT_INSTRUCTIONS", length = 65535)
private String peerAssessmentInstructions;
@Column(name = "CONTENT_REVIEW")
private Boolean contentReview = Boolean.FALSE;
public enum Access {
SITE,
GROUP
}
public enum SubmissionType {
ASSIGNMENT_SUBMISSION_TYPE_NONE, // 0
TEXT_ONLY_ASSIGNMENT_SUBMISSION, // 1
ATTACHMENT_ONLY_ASSIGNMENT_SUBMISSION, // 2
TEXT_AND_ATTACHMENT_ASSIGNMENT_SUBMISSION, // 3
NON_ELECTRONIC_ASSIGNMENT_SUBMISSION, // 4
SINGLE_ATTACHMENT_SUBMISSION // 5
}
public enum GradeType {
GRADE_TYPE_NONE, // 0
UNGRADED_GRADE_TYPE, // 1
LETTER_GRADE_TYPE, // 2
SCORE_GRADE_TYPE, // 3
PASS_FAIL_GRADE_TYPE, // 4
CHECK_GRADE_TYPE // 5
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy