如何将 List[zio.Task[List[A]]] 转换为 zio.Task[[List[A]]

How to convert List[zio.Task[List[A]]] to zio.Task[[List[A]]

我需要处理一组 ID,return 结果为 zio.Task[List[RelevantReadingRow]]

def getBaselinesForRequestIds(baseLineReqIds: Set[String]): Task[List[RelevantReadingRow]] =
  dynamoConnection
    .run(
      table.getAll("baseline_req_id" in baseLineReqIds)
    )
    .flatMap(_.toList.separate match {
      case (err :: _, _)           => ZIO.fail(new Throwable(describe(err)))
      case (Nil, relevantReadings) => ZIO.succeed(relevantReadings)
    })

上面的代码有效,但我需要将它们作为 mutch 以 25 个元素为一组进行处理。 我试过了,但后来我得到了 zio.Task

的列表
def getBaselinesForRequestIds(baseLineReqIds: Set[String]): Task[List[RelevantReadingRow]] = {
    val subSet = baseLineReqIds.grouped(25).toList
    val res = for {
      rows <- subSet.map(reqIds => dynamoConnection
        .run(
          table.getAll("baseline_req_id" in reqIds)
        ).flatMap(e => e.toList.separate match {
        case (err :: _, _) => ZIO.fail(new Throwable(describe(err)))
        case (Nil, relevantReadings) => ZIO.succeed(relevantReadings)
      }))
    } yield rows
    res // this is List[zio.Task[List[RelevantReadingRow]]]
  } 

我不知道如何转换回 zio.Task[List[RelevantReadingRow]]

有什么建议吗?

你可以用ZIO.collectAll把List[Task]转换成Task[List],我还以为是ZIO.sequence...,可能我跟猫搞混了...

以下示例适用于 Zio2


package sample

import zio._

object App extends ZIOAppDefault {
  case class Result(value: Int) extends AnyVal

  val data: List[Task[List[Result]]] = List(
    Task { List(Result(1), Result(2)) },
    Task { List(Result(3)) }
  )

  val flattenValues: Task[List[Result]] = for {
    values <- ZIO.collectAll { data }
  } yield values.flatten

  val app = for {
    values <- flattenValues
    _      <- Console.putStrLn { values }
  } yield()

  def run = app
}

特别是你的样品... 并假设 'separate' 它只是收集一些错误的扩展方法(returns 错误列表和结果列表的元组),并忽略方法 'describe' 将错误转换为可抛出的

https://scastie.scala-lang.org/jgoday/XKxVP2ECSFOv4chgSFCckg/7


package sample

import zio._


object App extends ZIOAppDefault {
  class DynamoMock {
    def run: Task[List[RelevantReadingRow]] = Task {
      List(
        RelevantReadingRow(1),
        RelevantReadingRow(2),
      )
    }
  }

  case class RelevantReadingRow(value: Int) extends AnyVal

  implicit class ListSeparate(list: List[RelevantReadingRow]) {
    def separate: (List[String], List[RelevantReadingRow]) =
      (Nil, list)
  }

  def getBaselinesForRequestIds(baseLineReqIds: Set[String]): Task[List[RelevantReadingRow]] = {
    val dynamoConnection = new DynamoMock()
    val subSet = baseLineReqIds.grouped(25).toList
    val res: List[Task[List[RelevantReadingRow]]] = for {
      rows <- subSet.map(reqIds => dynamoConnection
        .run.flatMap(e => e.toList.separate match {
          case (err :: _, _) => ZIO.fail(new Throwable(err))
          case (Nil, relevantReadings) => ZIO.succeed(relevantReadings)
        }))
    } yield rows

    for {
      rows <- ZIO.collectAll(res)
    } yield rows.flatten
  }

  val app = for {
    values <- getBaselinesForRequestIds(Set("id1", "id2"))
    _      <- Console.putStrLn { values }
  } yield()

  def run = app
}

所以,这是一个基于@jgoday 回答的替代解决方案

def getBaselinesForRequestIds(baseLineReqIds: Set[String]): Task[List[RelevantReadingRow]] =
    for {
      values <- ZIO.foreachPar(baseLineReqIds.grouped(25).toList) {
        reqIds =>
          dynamoConnection
            .run(
              table.getAll("baseline_req_id" in reqIds)
            ).flatMap(e => e.toList.separate match {
            case (err :: _, _) => ZIO.fail(new Throwable(describe(err)))
            case (Nil, relevantReadings) => ZIO.succeed(relevantReadings)
          })
      }
    } yield values.flatten