Search code examples
javajpaeclipselinkeclipse-virgo

EclipseLink hierarchy error


I'm trying to work with Virgo and EclipseLink and implement a application based on Greenpage project.

I have implemented a hierarchy but i am getting a wierd error (using EclipseLink):

Internal Exception: Exception [EclipseLink-7161] (Eclipse Persistence Services - 2.0.0.v20091127-r5931): org.eclipse.persistence.exceptions.ValidationException
Exception Description: Entity class [class model.Person] has no primary key specified. It should define either an @Id, @EmbeddedId or an @IdClass. If you have defined PK using any of these annotations then make sure that you do not have mixed access-type (both fields and properties annotated) in your entity class hierarchy.

Hierarchy:

Entity <- NamedEntity <- Person <- ... - others classes that extends Person

import java.io.Serializable;

import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.MappedSuperclass;

@MappedSuperclass
public abstract class Entity implements Serializable {

    private static final long serialVersionUID = 1L;

    private Long id;

    @Id
    @GeneratedValue(strategy=GenerationType.AUTO)
    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }
}


import javax.persistence.MappedSuperclass;

@MappedSuperclass
public abstract class NamedEntity extends Entity {
    private static final long serialVersionUID = 1L;

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return name;
    }
}



import javax.persistence.DiscriminatorColumn;
import javax.persistence.Entity;
import javax.persistence.Inheritance;
import javax.persistence.InheritanceType;

import pl.com.mgr.model.NamedEntity;

@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "discriminator")
public abstract class Person extends NamedEntity {
    private static final long serialVersionUID = 1L;

}

EDIT: Answer posted by Kevin Bowersox (many thanks!) helped... partly ;). The problem moves deeper. We are now: Entity <- NamedEntity <- Attribute <- LecturerAttribute

import javax.persistence.Column;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.MappedSuperclass;

import model.NamedEntity;

@MappedSuperclass
public abstract class Attribute<T extends NamedEntity> extends NamedEntity {
    private static final long serialVersionUID = 1L;

    private T entity;
    private String value;

    @ManyToOne
    @JoinColumn(name = "entity", nullable = false)
    public T getEntity() {
        return entity;
    }

    public void setEntity(T entity) {
        this.entity = entity;
    }

    @Column(nullable = false)
    public String getValue() {
        return value;
    }

    public void setValue(String value) {
        this.value = value;
    }


}




import javax.persistence.Entity;

@Entity
public class LecturerAttribute extends Attribute<Lecturer> {
    private static final long serialVersionUID = 1L;

}

And the exception:

Internal Exception: Exception [EclipseLink-7161] (Eclipse Persistence Services - 2.0.0.v20091127-r5931): org.eclipse.persistence.exceptions.ValidationException
Exception Description: Entity class [class model.LecturerAttribute] has no primary key specified. It should define either an @Id, @EmbeddedId or an @IdClass. If you have defined PK using any of these annotations then make sure that you do not have mixed access-type (both fields and properties annotated) in your entity class hierarchy.

Solution

  • Move the annotation from the accessor to the field declaration and change the fields accessibility to protected.

    @MappedSuperclass
    public abstract class Entity implements Serializable {
    
        private static final long serialVersionUID = 1L;
    
        @Id
        @GeneratedValue(strategy=GenerationType.AUTO)
        protected Long id;
    
        public Long getId() {
            return id;
        }
    
        public void setId(Long id) {
            this.id = id;
        }
    }