Как добавить правильную обработку ошибок в Ресурс кошачьего эффекта

Я пытаюсь получить какой-то базовый файл ввода / вывода (запись / чтение) чисто функциональным способом, используя эффект кошки. После изучения этого урока, вот что я закончил для чтения файла:

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 ответ

Решение

Правильная обработка ошибок может быть добавлена ​​с помощью .attempt по возвращенному значению IO:

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, который поддерживает экземпляры с эффектом кошки и имеет немного другую форму IO[E, A] где E фиксирует тип эффекта ошибки.

Другие вопросы по тегам