2016-11-11 59 views
1

我有一些實體,存儲庫和一個謂詞來處理動態查詢生成器。當我運行一個簡單的查詢時,我得到一個異常。爲什麼Spring Data會給我這個反射異常?

  1. 爲什麼會出現反射錯誤?
  2. 靠近軌跡的底部,它表示它不能將整數設置爲字符串。我不知道爲什麼那裏或如果它的相關。無論如何,所有的值都是Integer,所以我甚至不知道String來自哪裏。
  3. 在調用findAll()內發生異常。我假設Spring Data工作正常,並且我配置了錯誤的東西。我只是不知道它是什麼。

堆棧跟蹤(編輯,以減少交大小)

2016-11-11 11:52:05,405 [http-bio-8080-exec-13] ERROR com.etisoftware.lib.spring.web.controller.GlobalExceptionHandlerAdvice:34 - An uncaught exception occurred 
    org.springframework.orm.jpa.JpaSystemException: Error accessing field [private java.lang.Integer com.etisoftware.workorderprinting.beans.cborg.WorkOrderOwner.number] by reflection for persistent property [com.etisoftware.workorderprinting.beans.cborg.WorkOrderOwner#number] : 1; nested exception is org.hibernate.property.access.spi.PropertyAccessException: Error accessing field [private java.lang.Integer com.etisoftware.workorderprinting.beans.cborg.WorkOrderOwner.number] by reflection for persistent property [com.etisoftware.workorderprinting.beans.cborg.WorkOrderOwner#number] : 1 
     at org.springframework.orm.jpa.vendor.HibernateJpaDialect.convertHibernateAccessException(HibernateJpaDialect.java:333) 
     at org.springframework.orm.jpa.vendor.HibernateJpaDialect.translateExceptionIfPossible(HibernateJpaDialect.java:244) 
     at org.springframework.orm.jpa.AbstractEntityManagerFactoryBean.translateExceptionIfPossible(AbstractEntityManagerFactoryBean.java:491) 
     at org.springframework.dao.support.ChainedPersistenceExceptionTranslator.translateExceptionIfPossible(ChainedPersistenceExceptionTranslator.java:59) 
     at org.springframework.dao.support.DataAccessUtils.translateIfNecessary(DataAccessUtils.java:213) 

    Caused by: org.hibernate.property.access.spi.PropertyAccessException: Error accessing field [private java.lang.Integer com.etisoftware.workorderprinting.beans.cborg.WorkOrderOwner.number] by reflection for persistent property [com.etisoftware.workorderprinting.beans.cborg.WorkOrderOwner#number] : 1 
     at org.hibernate.property.access.spi.GetterFieldImpl.get(GetterFieldImpl.java:71) 
     at org.hibernate.tuple.entity.AbstractEntityTuplizer.getIdentifier(AbstractEntityTuplizer.java:224) 

     ... 89 more 
    Caused by: java.lang.IllegalArgumentException: Can not set java.lang.Integer field com.etisoftware.workorderprinting.beans.cborg.WorkOrderOwner.number to java.lang.String 
     at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:167) 
     at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:171) 
     at sun.reflect.UnsafeFieldAccessorImpl.ensureObj(UnsafeFieldAccessorImpl.java:58) 
     at sun.reflect.UnsafeObjectFieldAccessorImpl.get(UnsafeObjectFieldAccessorImpl.java:36) 
     at java.lang.reflect.Field.get(Field.java:393) 
     at org.hibernate.property.access.spi.GetterFieldImpl.get(GetterFieldImpl.java:67) 
     ... 133 more   

工作單實體

@Entity 
@Table(name = "workorders") 
public class WorkOrder 
{ 
    @Id 
    @GeneratedValue(strategy = GenerationType.IDENTITY) 
    @Column(name = "workorder_nbr") 
    private Integer   number; 

    @ManyToOne(fetch = FetchType.LAZY) 
    @JoinColumn(name = "subscriber_nbr") 
    private Subscriber  subscriber; 

    @ManyToOne(fetch = FetchType.LAZY) 
    @JoinColumn(name = "wo_type") 
    private WorkOrderType  type; 

    @ManyToOne(fetch = FetchType.LAZY) 
    @JoinColumn(name = "pool_nbr") 
    private WorkerPool  pool; 

    @JoinColumn(name = "wo_status_nbr") 
    @ManyToOne(fetch = FetchType.LAZY) 
    private WorkOrderStatus status; 

    @JoinColumn(name = "wo_owner_nbr") 
    @ManyToOne(fetch = FetchType.LAZY) 
    private WorkOrderOwner owner; 

    @Column(name = "sched_date") 
    private LocalDate   scheduleDate; 

    @ManyToOne(fetch = FetchType.LAZY) 
    @JoinColumn(name = "slot_nbr") 
    private TimeSlot slot; 

    @Column(name = "auto_complete_date") 
    private LocalDate   autoCompleteDate; 

    @Column(name = "override") 
    private Boolean   override; 

    @JoinColumn(name = "wo_location_nbr") 
    @ManyToOne(fetch = FetchType.LAZY) 
    private Location   location; 

    /* 
    * Fetch the notes later in the service using this number because connecting the notes table is a pain 
    */ 
    @Column(name = "wo_note_nbr") 
    private Integer   noteNumber; 

WorkOrderOwner實體

@Entity 
@Table(name = "wo_owners") 
public class WorkOrderOwner 
{ 
    @Id 
    @GeneratedValue(strategy = GenerationType.IDENTITY) 
    @Column(name = "wo_owner_nbr") 
    private Integer number; 

    @Column(name = "owner_id") 
    private String name; 

    @Column(name = "email_address") 
    private String email_address; 

WorkOrderPredicates

@Component 
public class WorkOrderPredicates 
{ 
    private static DateTimeFormatter dateFormatter  = DateTimeFormatter.ofPattern("MM/dd/yyyy"); 
    private static DateTimeFormatter timeFormatter  = DateTimeFormatter.ofPattern("hh:mm:ss"); 
    private static DateTimeFormatter dateTimeFormatter = DateTimeFormatter.ofPattern("MM/dd/yyyy hh:mm:ss"); 

    public Predicate buildQueryFromSearchCriteria(PrintWorkOrderFormDto queryDto) 
    { 

    BooleanBuilder builder = new BooleanBuilder(); 
    List<Predicate> predicates = new ArrayList<>(); 
    /* 
    * TODO I think think this string "entity" which is a variable name is going to give us a problem. 
    */ 
    PathBuilder<WorkOrder> entityPath = new PathBuilder<>(WorkOrder.class, "workOrder"); 

    for (SelectedCriteria criteria : queryDto.getCriteria().getRules()) 
    { 
     assignValueAndOperator(predicates, entityPath, criteria); 
    } 

    /* 
    * Per Chris, the queries will only have one group. That is, there will be a single condition [AND | OR] and then a 
    * list of criteria. 
    */ 
    if (queryDto.getCriteria().getCondition().equals(SelectionFilterCondition.AND)) 
    { 
     for (Predicate p : predicates) 
     builder.and(p); 
    } 
    else if (queryDto.getCriteria().getCondition().equals(SelectionFilterCondition.OR)) 
    { 
     for (Predicate p : predicates) 
     builder.or(p); 
    } 

    return builder; 
    } 

    @SuppressWarnings("unchecked") 
    protected static void assignValueAndOperator(List<Predicate> predicates, PathBuilder<?> entityPath, SelectedCriteria criteria) 
    { 
    switch (criteria.getOperator()) 
    { 
     case BETWEEN: 
     List<LocalDate> values = (List<LocalDate>) criteria.getValue(); 
     LocalDate from = values.get(0); 
     LocalDate to = values.get(1); 
     // TODO how do we handle Time separately from Date. 
     // TODO how do we handle DateTime 
     predicates.add(entityPath.getDate(criteria.getField(), LocalDate.class).between(from, to)); 
     break; 
     case BEGINS_WITH: 
     predicates.add(entityPath.getString(criteria.getField()).like(criteria.getValue() + "%")); 
     break; 
     case NOT_BEGINS_WITH: 
     predicates.add(entityPath.getString(criteria.getField()).notLike(criteria.getValue() + "%")); 
     break; 
     case CONTAINS: 
     predicates.add(entityPath.getString(criteria.getField()).like("%" + criteria.getValue() + "%")); 
     break; 
     case NOT_CONTAINS: 
     predicates.add(entityPath.getString(criteria.getField()).notLike("%" + criteria.getValue() + "%")); 
     break; 
     case ENDS_WITH: 
     predicates.add(entityPath.getString(criteria.getField()).like("%" + criteria.getValue())); 
     break; 
     case NOT_ENDS_WITH: 
     predicates.add(entityPath.getString(criteria.getField()).notLike("%" + criteria.getValue())); 
     break; 
     case EQUAL: 
     predicates.add(entityPath.get(criteria.getField()).eq(extractAndCreateValueOfType(criteria.getValue(), criteria.getType()))); 
     break; 
     case NOT_EQUAL: 
     predicates.add(entityPath.get(criteria.getField()).ne(criteria.getValue())); 
     break; 
     case GREATER: 
     // TODO this casting might not be right 
     predicates.add(entityPath.getNumber(criteria.getField(), Integer.class).gt((Integer) criteria.getValue())); 
     break; 
     case LESS: 
     // TODO this casting might not be right 
     predicates.add(entityPath.getNumber(criteria.getField(), Integer.class).lt((Integer) criteria.getValue())); 
     break; 
     case LESS_OR_EQUAL: 
     // TODO this casting might not be right 
     predicates.add(entityPath.getNumber(criteria.getField(), Integer.class).loe((Integer) criteria.getValue())); 
     break; 
     case GREATER_OR_EQUAL: 
     // TODO this casting might not be right 
     predicates.add(entityPath.getNumber(criteria.getField(), Integer.class).goe((Integer) criteria.getValue())); 
     break; 
     case IN: 
     /* 
     * TODO this is causing a hibernate error because its trying to map an integer to a string. This type setting 
     * below needs to be dynamic based on the selectedcriteria type 
     */ 
     List<String> inValues = (List<String>) criteria.getValue(); 
     predicates.add(entityPath.get(criteria.getField(), String.class).in(inValues)); 
     break; 
     case NOT_IN: 
     List<String> notInValues = (List<String>) criteria.getValue(); 
     predicates.add(entityPath.get(criteria.getField(), String.class).in(notInValues).not()); 
     break; 
     case IS_EMPTY: 
     predicates.add(entityPath.getList(criteria.getField(), List.class).isEmpty()); 
     break; 
     case IS_NOT_EMPTY: 
     predicates.add(entityPath.getList(criteria.getField(), List.class).isNotEmpty()); 
     break; 
     case IS_NOT_NULL: 
     predicates.add(entityPath.get(criteria.getField()).isNotNull()); 
     break; 
     case IS_NULL: 
     predicates.add(entityPath.get(criteria.getField()).isNull()); 
     break; 
     default: 
     // TODO implement the default case 
     break; 
    } 

    } 

    @SuppressWarnings("cast") 
    private static Object extractAndCreateValueOfType(Object valueObject, SelectionDataType selectionDataType) 
    { 
    Object value = null; 
    switch (selectionDataType) 
    { 
     case BOOLEAN: 
     // TODO 
     break; 
     case DATE: 
     value = LocalDate.parse((String) valueObject, dateFormatter); 
     break; 
     case TIME: 
     value = LocalDateTime.parse((String) valueObject, timeFormatter); 
     break; 
     case DATETIME: 
     value = LocalDateTime.parse((String) valueObject, dateTimeFormatter); 
     break; 
     case DOUBLE: 
     value = (Double) valueObject; 
     break; 
     case INTEGER: 
     value = (Integer) valueObject; 
     break; 
     case STRING: 
     value = (String) valueObject; 
     break; 
     default: 
     break; 
    } 
    return value; 
    } 

WorkOrderService fetch方法

@Transactional(transactionManager = PersistenceConfigCbOrgAbstract.txMgrName, readOnly = true) 
    protected List<WorkOrderDto> fetchWorkOrders(PrintWorkOrderFormDto queryParams, PrintStatus printStatus) 
    { 
    List<WorkOrderDto> dtos = new ArrayList<>(); 
    printStatus.setMessage("Fetching work orders"); 
    logger.debug("Query Params: " + queryParams); 

    for (WorkOrder bean : workOrderRepository.findAll(workOrderPredicates.buildQueryFromSearchCriteria(queryParams))) 
    { 
     logger.debug(ToStringBuilder.reflectionToString(bean)); 
     WorkOrderDto dto = modelMapper.map(bean, WorkOrderDto.class); 
     fetchAndAssignNotes(dto, bean.getNoteNumber()); 
     fetchAndAssignCreateDate(dto, bean.getNumber()); 
     fetchAndAssignPendingDeviceDtos(dto, bean.getNumber()); 
     fetchAndAssignPendingServiceDtos(dto, bean.getNumber()); 
     fetchAndAssignBilling(dto, bean.getNumber()); 

     try 
     { 
     dto.setCustomer(new CustomerDto()); 
     fetchAndAssignCustomer(dto); 
     } 
     catch (JsonParseException e) 
     { 
     logger.error("Could not parse customer information", e); 
     printStatus.setMessage("Cannot fetch work orders. Please check the application log"); 
     } 
     catch (JsonMappingException e) 
     { 
     logger.error("Could not map customer information", e); 
     printStatus.setMessage("Cannot fetch work orders. Please check the application log"); 
     } 
     catch (IOException e) 
     { 
     logger.error("Got an IO exception while trying to map customer information", e); 
     printStatus.setMessage("Cannot fetch work orders. Please check the application log"); 
     } 

     logger.debug(ToStringBuilder.reflectionToString(dto, ToStringStyle.MULTI_LINE_STYLE, Boolean.TRUE)); 
     printStatus.setRecordCount(printStatus.getRecordCount() + 1); 
     dtos.add(dto); 
    } 

    return dtos; 
    } 

wo_owners表

CREATE TABLE wo_owners (
    wo_owner_nbr serial NOT NULL, 
    owner_id nchar(20) NOT NULL, 
    email_address nchar(80), 
    PRIMARY KEY (wo_owner_nbr) 
); 

工作訂單

CREATE TABLE cborg2001:workorders (
    workorder_nbr serial NOT NULL, 
    subscriber_nbr int, 
    wo_type int, 
    pool_nbr int, 
    wo_status_nbr int, 
    wo_owner_nbr int, 
    wo_note_nbr int, 
    sched_date date, 
    slot_nbr int, 
    auto_complete_date date, 
    override int DEFAULT 0, 
    wo_location_nbr int, 
    PRIMARY KEY (workorder_nbr) 
); 

QueryDSL查詢

com.querydsl.jpa.impl.JPAQuery:233 - select workOrder from WorkOrder workOrder where workOrder.scheduleDate = ?1 

生成的查詢

select 
    workorder0_.workorder_nbr as workorde1_28_, 
    workorder0_.auto_complete_date as auto_com2_28_, 
    workorder0_.wo_location_nbr as wo_locat6_28_, 
    workorder0_.wo_note_nbr as wo_note_3_28_, 
    workorder0_.override as override4_28_, 
    workorder0_.wo_owner_nbr as wo_owner7_28_, 
    workorder0_.pool_nbr as pool_nbr8_28_, 
    workorder0_.sched_date as sched_da5_28_, 
    workorder0_.slot_nbr as slot_nbr9_28_, 
    workorder0_.wo_status_nbr as wo_stat10_28_, 
    workorder0_.subscriber_nbr as subscri11_28_, 
    workorder0_.wo_type as wo_type12_28_ 
from 
    workorders workorder0_ 
where 
    workorder0_.sched_date=? 

我用下面的查詢來驗證列的數據類型。

SELECT ST.tabname, SC.colname, SC.coltype 
FROM systables ST 
JOIN syscolumns SC ON SC.tabid = ST.tabid 
JOIN sys 
WHERE ST.tabname="wo_owners"; 

它返回的結果:

tabname  wo_owner_nbr coltype 
wo_owners wo_owner_nbr 262 
wo_owners owner_id 271 

的Informix coltypes列出here

262 - 256 = 6 = SERIAL 
271 - 256 = 15 = NCHAR 

注*在DB-Access中,256的偏移值總是被添加到這些coltype代碼,因爲DB-Access將SERIAL,SERIAL8和BIGSERIAL列NOT NULL。source

「了Informix®支持SERIAL,SERIAL8和BIGSERIAL數據類型,以生產自動整數序列。SERIAL基於整數(32位)」 source

+1

的是你的朋友在這裏造成的。請注意堆棧跟蹤中的最後一個原因。該字段在您的實體中設置爲整數,但它似乎來自數據庫作爲字符串。這顯然不好。 – rmlan

+0

我現在看到你在#2中指出了我的上述發現。這是絕對相關的,並且意味着在你的數據庫中,'workorder_nbr'列是varchar或text類型或其他字符串類型。檢查你的數據庫表定義。 – rmlan

+0

workorders中的wo_owner_nbr是一個int。 wo_owners中的wo_owner_nbr是串行的。 我已使用此信息編輯原始帖子。 – RhythmicDevil

回答

0

確切的問題與Hibernate或Spring Data無關,它與我從UI中選擇元素派生ArrayLists的方式有關。在WorkOrderPredicates我這樣做:

List<String> inValues = (List<String>) criteria.getValue(); 

這使得值的字符串,它也是理所當然不能被分配到一個整數。

更糟糕的是我後來改成了是這樣的:

if (criteria.getType().equals(SelectionDataType.INTEGER)) 
    { 
    ArrayList<Integer> values = criteria.getValue(); 
    predicates.add(entityPath.get(criteria.getField()).in((values))); 
    } 
    else if (criteria.getType().equals((SelectionDataType.STRING))) 
    { 
    ArrayList<String> values = criteria.getValue(); 
    predicates.add(entityPath.get(criteria.getField()).in((evalues))); 
    } 

這裏,由於仿製藥在運行時擦除的ArrayList中仍是一個ArrayList <字符串>甚至可以通過它看起來像我鑄造它。

我的解決辦法是建立新的陣列是這樣的:

if (criteria.getType().equals(SelectionDataType.INTEGER)) 
    { 
    predicates.add(entityPath.get(criteria.getField()).in((extractArrayListInteger(criteria)))); 
    } 
    else if (criteria.getType().equals((SelectionDataType.STRING))) 
    { 
    predicates.add(entityPath.get(criteria.getField()).in((extractArrayListString(criteria)))); 
    } 

方法來創建新的ArrayList

@SuppressWarnings("unchecked") 
    private static ArrayList<Integer> extractArrayListInteger(SelectedCriteria criteria) 
    { 
    ArrayList<Integer> values = new ArrayList<>(); 
    for (Object v : (ArrayList<Object>) criteria.getValue()) 
    { 
     values.add(Integer.valueOf(v.toString())); 
    } 
    return values; 
    } 

    @SuppressWarnings("unchecked") 
    private static ArrayList<String> extractArrayListString(SelectedCriteria criteria) 
    { 
    ArrayList<String> values = new ArrayList<>(); 
    for (String v : (ArrayList<String>) criteria.getValue()) 
    { 
     values.add(v); 
    } 
    return values; 
    } 
0

從我們看到它的確看起來像有正在添加字符串從數據庫中,但您確保該列被定義爲整數數據類型。我只是查找串行語法,它似乎在postgres中定義和整數列加上一個序列。

因此我沒有看到直接的答案,但想提出一些事情來追蹤這個問題。

  1. use this to see the actual datatype該列已在數據庫中。

  2. 負荷由ID所涉及的每JPA實體,以驗證該問題實際上是關於WorkOrderOwner

  3. 移除問題的所有屬性無關

  4. 變化number屬性字符串,看看問題消失

  5. 確認你沒有getter和setter或者說,他們有適當的數據類型

  6. 讓我們知道您的發現

+0

我將第1點的發現添加到原始帖子中。 – RhythmicDevil

+0

2.我可以使用集成測試通過ID加載每個JPA實體。 – RhythmicDevil

1

@Transactional註釋必須在一個公共的方法。您可以將註釋移動到調用您的代碼片段中的受保護方法的公共方法,也可以將您的受保護方法更改爲公共方法。

+0

謝謝,我應該抓住這一點。我已經修好了,它實際上解決了我遇到的另一個問題,但主要問題依然存在。 – RhythmicDevil

相關問題