scala - 如何使用 doobie 在 Scala 中对 PostgreSQL 数据库执行字符串 SQL 语句列表?

标签 scala doobie

我将以下 10 行 Python 代码移植到 Scala:

import psycopg2

def execute(user, password, database, host, port, *queries):
    connection = psycopg2.connect(user=user, password=password, host=host, port=port, database=database)
    cursor = connection.cursor()
    for sql in queries:
        print(sql)
        cursor.execute(sql)
    connection.commit()
    cursor.close()
    connection.close()

我有以下等效的 Scala 代码:

def execute(user: String, password: String, database: String, host: String, port: Int, queries: String*): Unit = {
  ???
}    

我想在一个单个事务中针对数据库(假设它是 Postgres)执行(并打印)一堆 SQL 语句并完成。

如何使用 doobie 来做到这一点?

注意:

  1. 我无法更改 execute() 的接口(interface)(包括我无法添加类型或隐式参数)。它必须接受字符串用户、密码等以及查询的可变参数:String*,从而保持界面与Python界面相同。

  2. 另请注明所需的所有进口

最佳答案

您可以使用 for 理解在 doobie 的一个事务中运行多个查询,例如:

val query = for {
   _  <- sql"insert into person (name, age) values ($name, $age)".update.run
   id <- sql"select lastval()".query[Long].unique
} yield p

但是这个解决方案不适用于您的情况,因为您有一个动态查询列表。幸运的是,我们可以使用traverse来自猫:

import cats.effect.ContextShift
import doobie._
import doobie.implicits._
import cats.effect._
import scala.concurrent.ExecutionContext
import cats.implicits._
import cats._
import cats.data._

def execute(user: String, password: String, database: String, host: String, port: Int, queries: String*): Unit = {

     //you can use other executor if you want
     //it would be better to pass context shift as implicit argument to method
    implicit val cs: ContextShift[IO] = IO.contextShift(ExecutionContext.global) 

    //let's create transactor  
    val xa = Transactor.fromDriverManager[IO](
      "org.postgresql.Driver",
       s"jdbc:postgresql://$host:$port/$database", //remember to change url or make it dynamic, if you run it agains another database
       user,
       password
    )

    val batch = queries
        .toList //we need to change String* to list, since String* doesn't have necessary typeclass for Aplicative
        .traverse(query => Update0(query, None).run) //we lift strings to Query0 and then run them, then we change List[ConnectionIO[Int]] to ConnectionIO[List[Int]]
        //above can be done in two steps using map and sequence

    batch  //now we've got single ConnectionIO which will run in one transaction
      .transact(xa) //let's make it IO[Int]
      .unsafeRunSync() //we need to block since your method returns Unit

  }

您的 IDE 可能会显示此代码无效,但它是正确的。 IDE 无法处理 Scala 的魔力。

您还可以考虑使用 unsafeRunTimed 而不是 unsafeRunSync 来添加时间限制。

另外,记得添加postgresql driver for jdbccats到您的build.sbt。 Doobie 在底层使用了猫,但我认为显式依赖可能是必要的。

关于scala - 如何使用 doobie 在 Scala 中对 PostgreSQL 数据库执行字符串 SQL 语句列表?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55601408/

相关文章:

scala - 为什么 Doobie 使用免费 monad?

scala - 如何使用 Doobie 在插入语句中获得可选结果?

generics - 在 Scala 参数化(泛型)类型错误中使用 Guava 的简单缓存管理器

Scala、PlayFramework、Mockito、ExecutionContext null

scala - 如何将父类(super class)型的Sea分离成子类型的Seqs

scala - 使用泛型进行 Doobie 批量更新

scala - 位于 src_management 文件夹中时未找到对象

scala - 如何更改 Scalatra 应用程序的 "webapp"目录的位置?

scala - 使用 Doobie 在单个事务中进行多个查询?