如何为cats-effect的资源

时间:2019-02-06 07:55:32

标签: scala functional-programming scala-cats cats-effect

我正在尝试使用cats-effect以纯粹的功能方式获取一些基本文件IO(写/读)。遵循this教程之后,下面是我读取文件的结果:

private def readFile(): IO[String] = for {
  lines <-  bufferedReader(new File(filePath)).use(readAllLines)
} yield lines.mkString

def bufferedReader(f: File): Resource[IO, BufferedReader] =
  Resource.make {
    IO(new BufferedReader(new FileReader(f)))
  } { fileReader =>
    IO(fileReader.close()).handleErrorWith(_ => IO.unit)
  }

现在在handleErrorWith函数中,我可以记录发生的任何错误,但是如何为它添加适当的错误处理(例如,返回Resource[IO, Either[CouldNotReadFileError, BufferedReader]])呢?

1 个答案:

答案 0 :(得分:4)

可以通过对返回的IO值使用.attempt来添加正确的错误处理:

import scala.collection.JavaConverters._

val resourceOrError: IO[Either[Throwable, String]] = bufferedReader(new File(""))
  .use(resource => IO(resource.lines().iterator().asScala.mkString))
  .attempt

如果要将其添加到自己的ADT中,可以使用leftMap

import cats.syntax.either._

final case class CouldNotReadError(e: Throwable)

val resourceOrError: IO[Either[CouldNotReadError, String]] =
  bufferedReader(new File(""))
    .use(resource => IO(resource.lines().iterator().asScala.mkString))
    .attempt
    .map(_.leftMap(CouldNotReadError))

此外,您可能对ZIO数据类型感兴趣,该数据类型具有supported cats-effect instances,并且形状与IO[E, A]略有不同,其中E捕获了错误影响类型。