繁体   English   中英

映射并缩小/折叠scalaz.Validation的HList

[英]Map and reduce/fold over HList of scalaz.Validation

我开始时有这样的事情:

def nonEmpty[A] = (msg: String) => (a: Option[A]) => a.toSuccess(msg)

val postal: Option[String] = request.param("postal")
val country: Option[String] = request.param("country")

val params =
  (postal  |> nonEmpty[String]("no postal" )).toValidationNel |@|
  (country |> nonEmpty[String]("no country")).toValidationNel

params { (postal, country) => ... }

现在我认为减少样板以获得更好的可读性并且不必向更多的初级团队成员解释什么.toValidateNel|@| 意思。 第一个想法是List但最后一行将停止工作,我不得不放弃一些静态安全。 所以我看向无形:

import shapeless._; import poly._; import syntax.std.tuple._

val params = (
  postal  |> nonEmpty[String]("no postal"),
  country |> nonEmpty[String]("no country")
)

params.map(_.toValidatioNel).reduce(_ |@| _)

但是,我甚至无法通过.map(...)位。 我按照#scalaz的建议尝试过:

type Va[+A] = Validation[String, A]
type VaNel[+A] = ValidationNel[String, A]

params.map(new (Va ~> VaNel) { def apply[T](x: Va[T]) = x.toValidationNel })

......无济于事

我曾经在#scalaz上寻求过帮助,但似乎并不是人们只有开箱即用的答案。 但是,我真的很想学习如何在实际和学习目的上解决这个问题。

PS实际上我的验证包含在Kleisli[Va, A, B]因此我可以使用>=>组成单独的验证步骤,但这似乎与问题正交,因为.map(...)是达到之后,所有Kleisli都将“减少”到Validation[String, A]

以下是shapeless-contribtraverse

import scalaz._, Scalaz._
import shapeless._, contrib.scalaz._, syntax.std.tuple._

def nonEmpty[A] = (msg: String) => (a: Option[A]) => a.toSuccess(msg)

val postal: Option[String] = Some("00000")
val country: Option[String] = Some("us")

val params = (
  postal  |> nonEmpty[String]("no postal"),
  country |> nonEmpty[String]("no country")
)

接着:

object ToVNS extends Poly1 {
  implicit def validation[T] = at[Validation[String, T]](_.toValidationNel)
}

val result = traverse(params.productElements)(ToVNS).map(_.tupled)

现在result是一个ValidationNel[String, (String, String)] ,你可以用它来做任何事情,你可以通过使用|@|减少你可以得到的糟糕的ApplicativeBuilder |@|

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM