6

我有一个使用ciris进行配置管理的 http4s 项目。

该项目在 github中

libraryDependencies ++= Seq(
  "is.cir" %% "ciris-cats",
  "is.cir" %% "ciris-cats-effect",
  "is.cir" %% "ciris-core",
  "is.cir" %% "ciris-enumeratum",
  "is.cir" %% "ciris-refined"
).map(_ % "0.12.1")

libraryDependencies ++= Seq(
  "org.http4s" %% "http4s-dsl",
  "org.http4s" %% "http4s-blaze-server"
).map(_ % "0.18.18")

libraryDependencies ++= Seq(
  "com.ovoenergy" %% "ciris-kubernetes" % "0.5",
  "org.typelevel" %% "kittens" % "1.2.0",
  "eu.timepit" %% "refined-cats" % "0.9.3"
)

当我编译我的项目时,我在这里得到以下错误

    [info] Compiling 12 Scala sources to /Users/rajkumar.natarajan/Documents/Coding/OS/ciris-example/target/scala-2.12/classes ...
    [error] /Users/rajkumar.natarajan/Documents/Coding/OS/ciris-example/src/main/scala/is/cir/example/application/Http4sApi.scala:24:68: Cannot find an implicit value for ContextShift[cats.effect.IO]:
    [error] * import ContextShift[cats.effect.IO] from your effects library
    [error] * if using IO, use cats.effect.IOApp or build one with cats.effect.IO.contextShift
    [error]   implicit val ioConcurrentEffect: Concurrent[IO] = cats.effect.IO.ioConcurrentEffect
    [error]                                                                    ^
    [error] /Users/rajkumar.natarajan/Documents/Coding/OS/ciris-example/src/main/scala/is/cir/example/application/Http4sApi.scala:43:69: type mismatch;
    [error]  found   : (ec: scala.concurrent.ExecutionContext, sc: java.util.concurrent.ScheduledExecutorService)cats.effect.Timer[cats.effect.IO] <and> (ec: scala.concurrent.ExecutionContext)cats.effect.Timer[cats.effect.IO]
    [error]  required: cats.effect.Timer[cats.effect.IO]
    [error]         Timeout(finite)(service)(ioConcurrentEffect, cats.effect.IO.timer)
    [error]                                                                     ^

原因是我在这里缺少函数所需的隐式参数 错误告诉我使用use cats.effect.IOApp or build one with cats.effect.IO.contextShift但找不到cats.effect.IOAppcats.effect.IO.contextShift

任何帮助表示赞赏。

4

1 回答 1

6

几种流行的用法,implicit其中之一是传递一些“上下文”信息。您查看的代码是这种用法的经典示例。

当您执行“超时”时,您需要决定两件事:

  • 主作业将在哪里(在哪个线程上)运行
  • 在哪里(在哪个线程上)计时器将被触发,因为在 JVM 世界中(例如与 JavaScript 不同)没有标准计时器。

这些参数的重要之处在于,一方面它们对工作至关重要,但另一方面它们仅支持主要参数。另一件事是,您可能只想拥有一个(或很少)全局对象,以便在任何地方使用这些对象。这就是使它们成为上下文的原因,这就是它们被隐式传递的原因。

现在您可以选择从哪里获得它们:

  • 使它们也成为您的上下文,即强制调用者将它们传递给您(也隐式地
  • 创建您自己的实例

这个选择不是一个微不足道的设计决定,它取决于你将如何使用你的 API。通常,正确的选择是第一个——让它们成为你的上下文。通过这种方式,您允许调用者以它想要的方式设置上下文(例如,应该TimerConcurrent使用相同的线程池还是不同的线程池?)。有时可以创建自己的独立上下文。或者将您从外部收到的另一个上下文包装到您特定的内容中。

假设您想将外部上下文包装在您的边界处Http4sApi,您可以编写如下代码:

final case class Http4sApi()(implicit executionContext: ExecutionContext) extends HttpApiAlg[IO] {
  // create IO-specific context from the executionContext
  private implicit val cs = IO.contextShift(executionContext)
  private implicit val timer = IO.timer(executionContext)

然后你可以写

def withTimeout(timeout: Duration)(service: HttpService[IO]): HttpService[IO] = timeout match {
    case finite: FiniteDuration => Timeout(finite)(service)
    case _                      => service
}

它应该编译。

于 2019-01-07T04:20:43.517 回答