web-dev-qa-db-ja.com

JPAとSpring Dataを使用してストアドプロシージャを実行するにはどうすればよいですか?

データベースでTerminal_GetTicketストアドプロシージャを呼び出そうとしていますが、次の例外が発生し続けます。

PropertyReferenceException: No property getTicket found for type TicketInfo

非常に単純なテストエンティティを使用して構成を相互検証しましたが、すべてが正常に機能しているように見えますが、実際のケースでは、何かが間違っています。

ドメインエンティティ(TicketInfo)は次のとおりです。

@Entity
@NamedStoredProcedureQuery(name = "TicketInfo.getTicket", procedureName = "Terminal_GetTicket", resultClasses = TicketInfo.class, parameters = { 
    @StoredProcedureParameter(mode = ParameterMode.IN, name = "sys_id_game", type = Integer.class)})
public class TicketInfo {

    @Id @GeneratedValue
    private Long id;
    private String idTicket;
    private Integer externalTicketCode;
    private Short sequenseAlert;
    private Integer dlTimeStamp;

すべてのインスタンス変数にはゲッターとセッターが適切に定義されており、ストアドプロシージャにはTicketInfoの属性に一致する合計5つの出力パラメーターがあります。

さらに、ここに私のリポジトリインターフェイスがあります。

public interface TicketInfoRepository extends CrudRepository<TicketInfo, Long> {
    @Transactional(timeout = 5)
    @Procedure
    TicketInfo getTicket(Integer sys_id_game);
}

また、ここに私のcontext.xmlファイルがあります(Spring用):

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:jpa="http://www.springframework.org/schema/data/jpa"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:repository="http://www.springframework.org/schema/data/repository"
    xsi:schemaLocation="http://www.springframework.org/schema/beans 
        http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
        http://www.springframework.org/schema/context
        http://www.springframework.org/schema/context/spring-context-4.1.xsd
        http://www.springframework.org/schema/data/jpa 
        http://www.springframework.org/schema/data/jpa/spring-jpa-1.8.xsd
        http://www.springframework.org/schema/data/repository
        http://www.springframework.org/schema/data/repository/spring-repository-1.5.xsd">

    <context:component-scan base-package="ar.com.boldt.godzilla" />
    <jpa:repositories base-package="xx.xxx.xxx.godzilla.business.dao" />

    <bean id="jpaVendorAdapter"
        class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
        <property name="showSql" value="${dataSource.show.sql}" />
        <property name="generateDdl" value="false" />
        <property name="database" value="SQL_SERVER" />
    </bean>

    <bean id="entityManagerFactory"
        class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="jpaVendorAdapter" ref="jpaVendorAdapter" />
        <!-- spring based scanning for entity classes -->
        <property name="packagesToScan" value="xx.xxx.xxx.godzilla.business.dao" />
    </bean>

    <bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager" />

    <bean id="cacheManager" class="org.springframework.cache.ehcache.EhCacheCacheManager">
        <property name="cacheManager" ref="ehcache" />
    </bean>

    <bean id="ehcache"
        class="org.springframework.cache.ehcache.EhCacheManagerFactoryBean">
        <property name="configLocation" value="classpath:ehcache.xml" />
    </bean>
</beans>

そして最後に、ストアドプロシージャ自体の骨抜きバージョン:

ALTER PROCEDURE [Terminal_GetTicket](
 @arg int 
,@res int output
,@res2 int output
)
as
Declare @error int

select 0, 1, 2

RETURN @error

これで、@Autowiredアノテーションを設定しようとするたびに、上記の例外が発生します。

13
Rys

私はMS SQLストアドプロシージャとspring-data-jpaに苦労していたことを覚えています。これは私が正常に実行できた方法です:

モデル:

@NamedNativeQueries({
    @NamedNativeQuery(
            name = "yourInternalName",
            query = "EXEC [procedure_name] :param1, :param2",
            resultClass = Foo.class
    )
 })
 @Entity
 public class Foo{

     /* Fields, getters, setters*/
}

それは非常に簡単です。ただし、このアプローチは異なります。プロシージャを直接宣言しているわけではありません(これが、RDBSを変更する場合に機能する必要がない理由でもあります)。

次に、リポジトリを拡張する必要があります。

public interface FooRepositoryCustom {

     Foo fancyMethodName(arg1, arg2);
}

そして、それを直接実装します:

public class FooRepositoryImpl implements FooRepositoryCustom {


@PersistenceContext
EntityManager entityManager;

@Override
public Foo fancyMethodName(arg1, arg2) {

    Query query = entityManager.createNamedQuery("yourInternalName");
    query.setParameter("param1", arg1);
    query.setParameter("param2", arg2);
    return query.getResultList();
}

すべてまとめてみましょう。

public interface FooRepository extends CrudRepository<Foo, Long>, FooRepositoryCustom {

}

たとえばFooオブジェクトのリストを返すことにした場合は、カスタムリポジトリの戻り値のみを編集することに注意してください。

7
SirKometa

私はSirKometasのアドバイスに従いましたが、それを機能させることができなかったので、自分に合ったものを思いつき、構文の観点からはより良いと思います。まず、次のようにエンティティクラスを作成します。

@NamedStoredProcedureQueries({//
    @NamedStoredProcedureQuery(//
            name = "MySP"//
            , procedureName = "my_sp"//
            , parameters = { //
                    @StoredProcedureParameter(mode = ParameterMode.IN, name = "arg", type = String.class)}//
            , resultClasses = Foo.class)//})
@Entity
public class Foo {

その場合、リポジトリの実装クラスは次のようになります。

@Component
public class FooRepositoryImpl implements FooCustomRepository {

    @PersistenceContext
    EntityManager entityManager;

    @Override
    public List<Foo> foo(String arg) {
        Query query = entityManager.createNamedStoredProcedureQuery("MySP");
        query.setParameter("arg", arg);
        return query.getResultList();
    }
}

実装の残りは、上記のSirKometaからの答えのようなものです。 アプリケーションでEntityManager Beanを作成する必要がありますこれが機能することも考えてください。

4
Keyhan