0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

Micronaut + Kotlin のプロジェクトにてMyBatisを使ってデータベースにデータ登録してみる

Posted at

前回

Micronaut + Kotlin のプロジェクトにてMyBatisを使ってデータベースからデータ取得してみる

前回のプロジェクトを拡張し、データベースにデータ登録してみたいと思います。
実際に作成したリポジトリはこちらになります。

Mapper

Interfaceの作成

BookMapper.ktに次のメソッドの宣言を追加します。

  • insert:データの登録(Insert文の実行)を行います。
  • update:データの更新(Update文の実行)を行います。
server/src/main/kotlin/example/mapper/BookMapper.kt
package example.mapper

import example.model.Book
import org.apache.ibatis.annotations.Mapper

@Mapper
interface BookMapper {
    fun findAll(): List<Book>
    fun findById(id: Int): Book
    fun insert(book: Book)
    fun update(book: Book)
}

SQLの作成

実行するSQL本体を定義します。
今回のサンプルではBookMapper.xmlに定義します。

server/src/main/resources/example/mapper/BookMapper.xml
    <insert id="insert" parameterType="example.model.Book">
        INSERT INTO book (
            name,
            publisher,
            publication_date,
            created_at,
            updated_at
        ) VALUES (
            #{name},
            #{publisher},
            #{publicationDate},
            current_timestamp,
            current_timestamp
        );
    </insert>

    <update id="update" parameterType="example.model.Book">
        UPDATE book
        SET
            name = #{name},
            publisher = #{publisher},
            publication_date = #{publicationDate},
            updated_at = current_timestamp
        WHERE
            id = #{id}
    </update>

実装部分の作成

実装部分としてServiceクラスにもメソッドを追加します。

  • insert(book: Book)
  • update(book: Book)
server/src/main/kotlin/example/service/BookService.kt
package example.service

import example.mapper.BookMapper
import example.model.Book
import org.apache.ibatis.session.SqlSessionFactory
import javax.inject.Singleton

@Singleton
class BookService(private val sqlSessionFactory: SqlSessionFactory) : BookMapper {

    override fun findAll(): List<Book> {
        sqlSessionFactory.openSession().use { session ->
            val bookMapper = session.getMapper(BookMapper::class.java)
            return bookMapper.findAll()
        }
    }

    override fun findById(id: Int): Book {
        sqlSessionFactory.openSession().use { session ->
            val bookMapper = session.getMapper(BookMapper::class.java)
            return bookMapper.findById(id)
        }
    }

    override fun insert(book: Book) {
        sqlSessionFactory.openSession().use { session ->
            val bookMapper = session.getMapper(BookMapper::class.java)
            bookMapper.insert(book)
            session.commit()
        }
    }

    override fun update(book: Book) {
        sqlSessionFactory.openSession().use { session ->
            val bookMapper = session.getMapper(BookMapper::class.java)
            bookMapper.update(book)
            session.commit()
        }
    }

}

Controller

データベースにアクセスするAPIを作成します。
Mapperと同様に登録と更新用の口を作成します。

  • create(book: Book):登録用のAPI。/bookにPOSTでリクエストすることで実行。
  • update(book: Book):更新用のAPI。/bookにPUTでリクエストすることで実行。
server/src/main/kotlin/example/controller/BookController.kt
package example.controller

import example.mapper.BookMapper
import example.model.Book
import io.micronaut.http.HttpResponse
import io.micronaut.http.annotation.Controller
import io.micronaut.http.annotation.Get
import io.micronaut.http.annotation.Post
import io.micronaut.http.annotation.Put


@Controller()
class BookController(private val bookMapper: BookMapper) {

    /**
     * 書籍情報の全取得。
     *
     * @return HttpResponse
     */
    @Get("/book")
    fun read(): HttpResponse<List<Book>> {
        return HttpResponse.ok(bookMapper.findAll())
    }

    /**
     * 書籍情報の取得。
     * ID指定
     *
     * @param id 書籍ID
     *
     * @return HttpResponse
     */
    @Get("/book/{id}")
    fun readById(id: Int): HttpResponse<Book> {
        return HttpResponse.ok(bookMapper.findById(id))
    }

    /**
     * 書籍情報の登録。
     */
    @Post("/book")
    fun create(book: Book): HttpResponse<String> {
        bookMapper.insert(book)
        return HttpResponse.ok()
    }

    /**
     * 書籍情報の更新。
     */
    @Put("/book")
    fun update(book: Book): HttpResponse<String> {
        bookMapper.update(book)
        return HttpResponse.ok()
    }
}

Model

モデルも若干修正します。
前回のままだとJSONが一部の項目で正しくマッピングできなかったので、JsonPropertyのアノテーションを追加します。

server/src/main/kotlin/example/model/Book.kt
package example.model

import com.fasterxml.jackson.annotation.JsonProperty
import io.micronaut.core.annotation.Introspected
import java.sql.Date


/**
 * 書籍データクラス
 */
@Introspected
data class Book(
        @JsonProperty("id")
        var id: Int?,

        @JsonProperty("name")
        var name: String,

        @JsonProperty("publisher")
        var publisher: String?,

        @JsonProperty("publication_date")
        var publicationDate: Date?,
)

ひとまず完成

ここまでで書籍情報の登録、更新するAPIが作成されました。
試してみましょう。

サーバを実行します。

$ cd server 
$ ./gradlew run

今回、APIへのリクエストにはPOSTMANを使います。

まず登録してみます。
image.png

ドン!
image.png

登録できましたね!

次、更新を試してみます。
nametest書籍3にしてみます。
image.png
ドン!
image.png
更新できました!

おわりに

今回追加したinsertupdateBookService内にて各SQL実行後にsession.commit()をしています。
参考にしたガイドでも同じように各SQLの実行後にsession.commit()を実行しトランザクションを確定させています。
ただ、個人的にはメソッド単位でトランザクションを確定させるのは使い勝手が悪いようにも感じます。
例えば、実業務で開発するサービスでは1リクエストの中で複数のテーブル更新もすることがあるかと思います。
できればAPI単位でトランザクションを持ちたい所です。
Spring Bootだと記事が見つかるけど、Micronautだとどうやるのだろうと調査中。。

次にやりたいこと

  • DB周りの自動テストを作りたい。何を使ってやるのがデファクトなんでしょうか…?
  • React勉強し始めたので、フロント作ってみようかな。

参考

0
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?