Использование составного ключа с JPA

Обучение на ходу с помощью Spring и всех его инструментов. Не уверен, как использовать PK вместо String. tickets = repository.findOne(tk.getSerial()); Хочет первичный ключ, но не может найти правильный путь?

Основной интерфейс:

package com.boot;

import com.boot.model.Tickets;
import com.boot.repository.TicketsRepository;
import com.boot.service.TicketsEditor;
import com.vaadin.annotations.Theme;
import com.vaadin.data.util.BeanContainer;
import com.vaadin.data.util.BeanItemContainer;
import com.vaadin.sass.internal.util.StringUtil;
import com.vaadin.server.FontAwesome;
import com.vaadin.server.VaadinRequest;
import com.vaadin.spring.annotation.SpringUI;
import com.vaadin.ui.*;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.util.StringUtils;

/**
 * Created by Ryan on 7/28/2016.
 */

@SpringUI
@Theme("valo")
public class TicketsUI extends UI {
    private final TicketsRepository repository;
    private final TicketsEditor editor;
    private final Grid grid;
    private final TextField filter;
    private final TextField invoiceFilter;
    private final Button addNewBtn;

    @Autowired
    public TicketsUI(TicketsRepository tkRepo, TicketsEditor tkEditor) {
        this.repository = tkRepo;
        this.editor = tkEditor;
        this.grid = new Grid();
        this.filter = new TextField();
        this.invoiceFilter = new TextField();
        this.addNewBtn = new Button("New game", FontAwesome.PLUS);

    }

    @Override
    protected void init(VaadinRequest request) {


    // build layout
    HorizontalLayout actions = new HorizontalLayout(filter, invoiceFilter, addNewBtn);
    VerticalLayout mainLayout = new VerticalLayout(actions, grid, editor);

    setContent(mainLayout);

    // Configure layouts and components
    actions.setSpacing(true);
    mainLayout.setMargin(true);
    mainLayout.setSpacing(true);

    grid.setHeight(500,Unit.PIXELS);
        grid.setWidth(1800,Unit.PIXELS);
        grid.setColumns("serial", "datePlaced", "dateRemoved","unsoldAmt", "actualGross","actualPrizes",
                "actualNet", "bin", "inplay", "closed", "unsoldTickets", "lastSaleRem", "prizeRem1", "prizeRem2",
                "prizeRem3", "prizeRem4", "prizeRem5","prizeRem6", "prizeRem7", "prizeRem8","prizeRem9", "prizeRem10", "prizeRem11", "prizeRem12", "prizeRem13", "prizeRem14", "prizeRem15",
                "gameTemplatesPartNum", "datePurch", "invoiceNum", "type");
        filter.setInputPrompt("Filter by serial");
        filter.addTextChangeListener(e -> listTickets(e.getText()));

      //  invoiceFilter.setInputPrompt("Filter by invoice");
      //  invoiceFilter.addTextChangeListener(e -> listTickets(e.getText()));
        //grid.setContainerDataSource(new BeanItemContainer(Tickets.class, repository.findByInvoiceNum()));
        // Connect selected Customer to editor or hide if none is selected
        grid.addSelectionListener(e -> {
            if (e.getSelected().isEmpty()) {
                editor.setVisible(false);
            }
            else {
                editor.editTicket((Tickets) grid.getSelectedRow());
            }
        });

        // Instantiate and edit new Customer the new button is clicked
        addNewBtn.addClickListener(e -> editor.editTicket(new Tickets()));

        // Listen changes made by the editor, refresh data from backend
        editor.setChangeHandler(() -> {
            editor.setVisible(false);
            listTickets(filter.getValue());
        });

        // Initialize listing
        listTickets(null);
}
    private void listTickets(String text){
        if (StringUtils.isEmpty(text)) {
            grid.setContainerDataSource(
                    new BeanItemContainer(Tickets.class, repository.findAll()));
        }else
        {
            grid.setContainerDataSource(new BeanItemContainer(Tickets.class, repository.findBySerial(text)));


        }
    }
}

Билет

    package com.boot.model;

import javax.persistence.*;
import java.sql.Date;

/**
 * Created by Ryan on 7/26/2016.
 */
@Entity
@IdClass(TicketsPK.class)
public class Tickets {
    private String serial;
    private Date datePlaced;
    private Date dateRemoved;
    private Integer unsoldAmt;
    private Integer actualGross;
    private Integer actualPrizes;
    private Integer actualNet;
    private Integer bin;
    private Byte inplay;
    private Integer closed;
    private Integer unsoldTickets;
    private Byte lastSaleRem;
    private Integer prizeRem1;
    private Integer prizeRem2;
    private Integer prizeRem3;
    private Integer prizeRem4;
    private Integer prizeRem5;
    private Integer prizeRem6;
    private Integer prizeRem7;
    private Integer prizeRem8;
    private Integer prizeRem9;
    private Integer prizeRem10;
    private Integer prizeRem11;
    private Integer prizeRem12;
    private Integer prizeRem13;
    private Integer prizeRem14;
    private Integer prizeRem15;
    private String gameTemplatesPartNum;
    private Date datePurch;
    private String invoiceNum;
    private Integer type;

    @Id
    @Column(name = "Serial")
    public String getSerial() {
        return serial;
    }

    public void setSerial(String serial) {
        this.serial = serial;
    }

    @Basic
    @Column(name = "Date_placed")
    public Date getDatePlaced() {
        return datePlaced;
    }

    public void setDatePlaced(Date datePlaced) {
        this.datePlaced = datePlaced;
    }

    @Basic
    @Column(name = "Date_removed")
    public Date getDateRemoved() {
        return dateRemoved;
    }

    public void setDateRemoved(Date dateRemoved) {
        this.dateRemoved = dateRemoved;
    }

    @Basic
    @Column(name = "Unsold_amt")
    public Integer getUnsoldAmt() {
        return unsoldAmt;
    }

    public void setUnsoldAmt(Integer unsoldAmt) {
        this.unsoldAmt = unsoldAmt;
    }

    @Basic
    @Column(name = "Actual_gross")
    public Integer getActualGross() {
        return actualGross;
    }

    public void setActualGross(Integer actualGross) {
        this.actualGross = actualGross;
    }

    @Basic
    @Column(name = "Actual_prizes")
    public Integer getActualPrizes() {
        return actualPrizes;
    }

    public void setActualPrizes(Integer actualPrizes) {
        this.actualPrizes = actualPrizes;
    }

    @Basic
    @Column(name = "Actual_net")
    public Integer getActualNet() {
        return actualNet;
    }

    public void setActualNet(Integer actualNet) {
        this.actualNet = actualNet;
    }

    @Basic
    @Column(name = "Bin")
    public Integer getBin() {
        return bin;
    }

    public void setBin(Integer bin) {
        this.bin = bin;
    }

    @Basic
    @Column(name = "Inplay")
    public Byte getInplay() {
        return inplay;
    }

    public void setInplay(Byte inplay) {
        this.inplay = inplay;
    }

    @Basic
    @Column(name = "Closed")
    public Integer getClosed() {
        return closed;
    }

    public void setClosed(Integer closed) {
        this.closed = closed;
    }

    @Basic
    @Column(name = "Unsold_tickets")
    public Integer getUnsoldTickets() {
        return unsoldTickets;
    }

    public void setUnsoldTickets(Integer unsoldTickets) {
        this.unsoldTickets = unsoldTickets;
    }

    @Basic
    @Column(name = "Last_sale_rem")
    public Byte getLastSaleRem() {
        return lastSaleRem;
    }

    public void setLastSaleRem(Byte lastSaleRem) {
        this.lastSaleRem = lastSaleRem;
    }

    @Basic
    @Column(name = "Prize_rem1")
    public Integer getPrizeRem1() {
        return prizeRem1;
    }

    public void setPrizeRem1(Integer prizeRem1) {
        this.prizeRem1 = prizeRem1;
    }

    @Basic
    @Column(name = "Prize_rem2")
    public Integer getPrizeRem2() {
        return prizeRem2;
    }

    public void setPrizeRem2(Integer prizeRem2) {
        this.prizeRem2 = prizeRem2;
    }

    @Basic
    @Column(name = "Prize_rem3")
    public Integer getPrizeRem3() {
        return prizeRem3;
    }

    public void setPrizeRem3(Integer prizeRem3) {
        this.prizeRem3 = prizeRem3;
    }

    @Basic
    @Column(name = "Prize_rem4")
    public Integer getPrizeRem4() {
        return prizeRem4;
    }

    public void setPrizeRem4(Integer prizeRem4) {
        this.prizeRem4 = prizeRem4;
    }

    @Basic
    @Column(name = "Prize_rem5")
    public Integer getPrizeRem5() {
        return prizeRem5;
    }

    public void setPrizeRem5(Integer prizeRem5) {
        this.prizeRem5 = prizeRem5;
    }

    @Basic
    @Column(name = "Prize_rem6")
    public Integer getPrizeRem6() {
        return prizeRem6;
    }

    public void setPrizeRem6(Integer prizeRem6) {
        this.prizeRem6 = prizeRem6;
    }

    @Basic
    @Column(name = "Prize_rem7")
    public Integer getPrizeRem7() {
        return prizeRem7;
    }

    public void setPrizeRem7(Integer prizeRem7) {
        this.prizeRem7 = prizeRem7;
    }

    @Basic
    @Column(name = "Prize_rem8")
    public Integer getPrizeRem8() {
        return prizeRem8;
    }

    public void setPrizeRem8(Integer prizeRem8) {
        this.prizeRem8 = prizeRem8;
    }

    @Basic
    @Column(name = "Prize_rem9")
    public Integer getPrizeRem9() {
        return prizeRem9;
    }

    public void setPrizeRem9(Integer prizeRem9) {
        this.prizeRem9 = prizeRem9;
    }

    @Basic
    @Column(name = "Prize_rem10")
    public Integer getPrizeRem10() {
        return prizeRem10;
    }

    public void setPrizeRem10(Integer prizeRem10) {
        this.prizeRem10 = prizeRem10;
    }

    @Basic
    @Column(name = "Prize_rem11")
    public Integer getPrizeRem11() {
        return prizeRem11;
    }

    public void setPrizeRem11(Integer prizeRem11) {
        this.prizeRem11 = prizeRem11;
    }

    @Basic
    @Column(name = "Prize_rem12")
    public Integer getPrizeRem12() {
        return prizeRem12;
    }

    public void setPrizeRem12(Integer prizeRem12) {
        this.prizeRem12 = prizeRem12;
    }

    @Basic
    @Column(name = "Prize_rem13")
    public Integer getPrizeRem13() {
        return prizeRem13;
    }

    public void setPrizeRem13(Integer prizeRem13) {
        this.prizeRem13 = prizeRem13;
    }

    @Basic
    @Column(name = "Prize_rem14")
    public Integer getPrizeRem14() {
        return prizeRem14;
    }

    public void setPrizeRem14(Integer prizeRem14) {
        this.prizeRem14 = prizeRem14;
    }

    @Basic
    @Column(name = "Prize_rem15")
    public Integer getPrizeRem15() {
        return prizeRem15;
    }

    public void setPrizeRem15(Integer prizeRem15) {
        this.prizeRem15 = prizeRem15;
    }

    @Id
    @Column(name = "game_templates_part_num")
    public String getGameTemplatesPartNum() {
        return gameTemplatesPartNum;
    }

    public void setGameTemplatesPartNum(String gameTemplatesPartNum) {
        this.gameTemplatesPartNum = gameTemplatesPartNum;
    }

    @Basic
    @Column(name = "date_purch")
    public Date getDatePurch() {
        return datePurch;
    }

    public void setDatePurch(Date datePurch) {
        this.datePurch = datePurch;
    }

    @Basic
    @Column(name = "invoice_num")
    public String getInvoiceNum() {
        return invoiceNum;
    }

    public void setInvoiceNum(String invoiceNum) {
        this.invoiceNum = invoiceNum;
    }

    @Basic
    @Column(name = "type")
    public Integer getType() {
        return type;
    }

    public void setType(Integer type) {
        this.type = type;
    }


}

TicketsPK

package com.boot.model;

import javax.persistence.Column;
import javax.persistence.Id;
import java.io.Serializable;

/**
 * Created by Ryan on 7/26/2016.
 */
public class TicketsPK implements Serializable {
    private String serial;
    private String gameTemplatesPartNum;

    @Column(name = "Serial")
    @Id
    public String getSerial() {
        return serial;
    }

    public void setSerial(String serial) {
        this.serial = serial;
    }

    @Column(name = "game_templates_part_num")
    @Id
    public String getGameTemplatesPartNum() {
        return gameTemplatesPartNum;
    }

    public void setGameTemplatesPartNum(String gameTemplatesPartNum) {
        this.gameTemplatesPartNum = gameTemplatesPartNum;
    }

}

Repo

package com.boot.repository;

import com.boot.model.Tickets;
import com.boot.model.TicketsPK;
import org.springframework.data.jpa.repository.JpaRepository;

import java.util.List;

/**
 * Created by Ryan on 7/28/2016.
 */
public interface TicketsRepository extends JpaRepository<Tickets, String>{

    List<Tickets> findBySerial (String serial);

}

Обрезка стека трассировки:

org.springframework.dao.InvalidDataAccessApiUsageException: Provided id of the wrong type for class com.boot.model.Tickets. Expected: class com.boot.model.TicketsPK, got class java.lang.String; nested exception is java.lang.IllegalArgumentException: Provided id of the wrong type for class com.boot.model.Tickets. Expected: class com.boot.model.TicketsPK, got class java.lang.String
    at org.springframework.orm.jpa.EntityManagerFactoryUtils.convertJpaAccessExceptionIfPossible(EntityManagerFactoryUtils.java:384) ~[spring-orm-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.orm.jpa.vendor.HibernateJpaDialect.translateExceptionIfPossible(HibernateJpaDialect.java:227) ~[spring-orm-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean.translateExceptionIfPossible(AbstractEntityManagerFactoryBean.java:417) ~[spring-orm-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.dao.support.ChainedPersistenceExceptionTranslator.translateExceptionIfPossible(ChainedPersistenceExceptionTranslator.java:59) ~[spring-tx-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.dao.support.DataAccessUtils.translateIfNecessary(DataAccessUtils.java:213) ~[spring-tx-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.dao.support.PersistenceExceptionTranslationInterceptor.invoke(PersistenceExceptionTranslationInterceptor.java:147) ~[spring-tx-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179) ~[spring-aop-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.data.jpa.repository.support.CrudMethodMetadataPostProcessor$CrudMethodMetadataPopulatingMethodInterceptor.invoke(CrudMethodMetadataPostProcessor.java:131) ~[spring-data-jpa-1.9.2.RELEASE.jar:na]
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179) ~[spring-aop-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.aop.interceptor.ExposeInvocationInterceptor.invoke(ExposeInvocationInterceptor.java:92) ~[spring-aop-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179) ~[spring-aop-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:208) ~[spring-aop-4.2.4.RELEASE.jar:4.2.4.RELEASE]
    at com.sun.proxy.$Proxy80.findOne(Unknown Source) ~[na:na]
    at com.boot.service.TicketsEditor.editTicket(TicketsEditor.java:88) ~[classes/:na]
    at com.boot.TicketsUI.lambda$init$ca60a5c0$1(TicketsUI.java:75) ~[classes/:na]
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) ~[na:1.8.0_91]
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) ~[na:1.8.0_91]
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) ~[na:1.8.0_91]
    at java.lang.reflect.Method.invoke(Method.java:498) ~[na:1.8.0_91]
at com.boot.TicketsUI.lambda$init$ca60a5c0$1(TicketsUI.java:75) ~[classes/:na]
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) ~[na:1.8.0_91]
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) ~[na:1.8.0_91]
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) ~[na:1.8.0_91]
at java.lang.reflect.Method.invoke(Method.java:498) ~[na:1.8.0_91]

TicketEditor

package com.boot.service;

import com.boot.model.Tickets;
import com.boot.model.TicketsPK;
import com.boot.repository.TicketsRepository;
import com.vaadin.data.fieldgroup.BeanFieldGroup;
import com.vaadin.event.ShortcutAction;
import com.vaadin.server.FontAwesome;
import com.vaadin.spring.annotation.SpringComponent;
import com.vaadin.spring.annotation.UIScope;
import com.vaadin.ui.*;
import com.vaadin.ui.components.calendar.CalendarComponentEvents;
import com.vaadin.ui.themes.ValoTheme;
import org.springframework.beans.factory.annotation.Autowired;

/**
 * Created by Ryan on 7/28/2016.
 */
@SpringComponent
@UIScope
public class TicketsEditor extends VerticalLayout{

    private final TicketsRepository repository;
    private Tickets tickets;

    //TextField gameName = new TextField("Game Name");
    TextField serial = new TextField("Serial");
    DateField datePlaced = new DateField("Date placed");
    DateField dateRemoved = new DateField("Date Removed");
    TextField unsoldAmt = new TextField("Unsold Amount");
    TextField actualGross = new TextField("Actual Gross");
    TextField actualPrizes = new TextField("Actual Prizes");
    TextField actualNet = new TextField("Actual Net");
    TextField bin = new TextField("Bin");
    TextField inplay = new TextField("Inplay");
    TextField closed = new TextField("Closed");
    TextField unsoldTickets = new TextField("Unsold Tickets");
    TextField lastSaleRem = new TextField("Last Sale Remaining");
    TextField prizeRem1 = new TextField("Prize Remaining 1");
    TextField prizeRem2 = new TextField("Prize Remaining 2");
    TextField prizeRem3 = new TextField("Prize Remaining 3");
    TextField prizeRem4 = new TextField("Prize Remaining 4");
    TextField prizeRem5 = new TextField("Prize Remaining 5");
    TextField prizeRem6 = new TextField("Prize Remaining 6");
    TextField prizeRem7 = new TextField("Prize Remaining 7");
    TextField prizeRem8 = new TextField("Prize Remaining 8");
    TextField prizeRem9 = new TextField("Prize Remaining 9");
    TextField prizeRem10 = new TextField("Prize Remaining 10");
    TextField prizeRem11 = new TextField("Prize Remaining 11");
    TextField prizeRem12 = new TextField("Prize Remaining 12");
    TextField prizeRem13 = new TextField("Prize Remaining 13");
    TextField prizeRem14 = new TextField("Prize Remaining 14");
    TextField prizeRem15 = new TextField("Prize Remaining 15");
    TextField gameTemplatesPartNum = new TextField("Part Number");
    DateField datePurchased = new DateField("Date Purchased");
    TextField invoice = new TextField("Invoice");
    TextField type = new TextField("Type");

    /* Action buttons */
    Button save = new Button("Save", FontAwesome.SAVE);
    Button cancel = new Button("Cancel");
    Button delete = new Button("Delete", FontAwesome.TRASH_O);
    CssLayout actions = new CssLayout(save, cancel, delete);

    @Autowired
    public TicketsEditor(TicketsRepository repository){
        this.repository = repository;
        addComponents(serial,datePlaced,dateRemoved, unsoldAmt, actualGross, actualPrizes, actualNet,bin,inplay,
                closed, unsoldTickets, lastSaleRem, prizeRem1, prizeRem2, prizeRem3, prizeRem4, prizeRem5, prizeRem6,
                prizeRem7, prizeRem8, prizeRem9, prizeRem10, prizeRem11, prizeRem12, prizeRem13, prizeRem14, prizeRem15,
                actions);

        setSpacing(true);
        actions.setStyleName(ValoTheme.LAYOUT_COMPONENT_GROUP);
        save.setStyleName(ValoTheme.BUTTON_PRIMARY);
        save.setClickShortcut(ShortcutAction.KeyCode.ENTER);

        save.addClickListener(e -> repository.saveAndFlush(tickets));
        delete.addClickListener(e -> repository.delete(tickets));
        cancel.addClickListener(e -> editTicket(tickets));
        setVisible(false);
    }
    public interface ChangeHandler{
        void onChange();
    }
    public final void editTicket(Tickets tk){
        final boolean persisted = tk.getSerial() != null;
        if(persisted){
            tickets = repository.findOne(tk.getSerial());
        }else{
            tickets = tk;
        }

        cancel.setVisible(persisted);
        BeanFieldGroup.bindFieldsUnbuffered(tickets, this);
        setVisible(true);
        save.focus();
        serial.selectAll();
    }
    public void setChangeHandler(ChangeHandler h){
        save.addClickListener(e -> h.onChange());
        delete.addClickListener(e -> h.onChange());
    }
}

1 ответ

em.find принимает в качестве примера IdClass, Так что в вашем случае вам нужно передать в экземпляр TicketsPK, не один компонент этого.

Другие вопросы по тегам