mirror of
https://github.com/idris-lang/Idris2.git
synced 2025-01-04 14:18:26 +03:00
Validated
data structure was added (#1098)
This commit is contained in:
parent
4144510bb3
commit
cf981d4c68
155
libs/contrib/Data/Validated.idr
Normal file
155
libs/contrib/Data/Validated.idr
Normal file
@ -0,0 +1,155 @@
|
||||
module Data.Validated
|
||||
|
||||
import Data.List1
|
||||
|
||||
import Decidable.Equality
|
||||
|
||||
%default total
|
||||
|
||||
||| `Validated` is like an `Either` but accumulates all errors with semigroup operation.
|
||||
public export
|
||||
data Validated e a = Valid a | Invalid e
|
||||
|
||||
--- Instances of standard interfaces ---
|
||||
|
||||
public export
|
||||
(Eq e, Eq a) => Eq (Validated e a) where
|
||||
Valid x == Valid y = x == y
|
||||
Invalid e == Invalid f = e == f
|
||||
_ == _ = False
|
||||
|
||||
export
|
||||
(Show e, Show a) => Show (Validated e a) where
|
||||
showPrec d $ Valid x = showCon d "Valid" $ showArg x
|
||||
showPrec d $ Invalid e = showCon d "Invalid" $ showArg e
|
||||
|
||||
public export
|
||||
Functor (Validated e) where
|
||||
map f $ Valid x = Valid $ f x
|
||||
map _ $ Invalid e = Invalid e
|
||||
|
||||
public export
|
||||
Bifunctor Validated where
|
||||
bimap _ s $ Valid x = Valid $ s x
|
||||
bimap f _ $ Invalid e = Invalid $ f e
|
||||
|
||||
||| Applicative composition preserves invalidity sequentially accumulating all errors.
|
||||
public export
|
||||
Semigroup e => Applicative (Validated e) where
|
||||
pure = Valid
|
||||
|
||||
Valid f <*> Valid x = Valid $ f x
|
||||
Invalid e1 <*> Invalid e2 = Invalid $ e1 <+> e2
|
||||
Invalid e <*> Valid _ = Invalid e
|
||||
Valid _ <*> Invalid e = Invalid e
|
||||
|
||||
-- There is no `Monad` implementation because it can't be coherent with the accumulating `Applicative` one.
|
||||
|
||||
||| Semigroup operation selects the leftmost valid value.
|
||||
||| If both sides are invalid, errors are accumulated.
|
||||
public export
|
||||
Semigroup e => Semigroup (Validated e a) where
|
||||
l@(Valid _) <+> _ = l
|
||||
_ <+> r@(Valid _) = r
|
||||
Invalid e1 <+> Invalid e2 = Invalid $ e1 <+> e2
|
||||
|
||||
public export
|
||||
Monoid e => Monoid (Validated e a) where
|
||||
neutral = Invalid neutral
|
||||
|
||||
||| Alternative composition preserves validity selecting the leftmost valid value.
|
||||
||| If both sides are invalid, errors are accumulated.
|
||||
public export
|
||||
Monoid e => Alternative (Validated e) where
|
||||
empty = neutral
|
||||
(<|>) = (<+>)
|
||||
|
||||
public export
|
||||
Foldable (Validated e) where
|
||||
foldr op init $ Valid x = x `op` init
|
||||
foldr _ init $ Invalid _ = init
|
||||
|
||||
foldl op init $ Valid x = init `op` x
|
||||
foldl _ init $ Invalid _ = init
|
||||
|
||||
null $ Valid _ = False
|
||||
null $ Invalid _ = True
|
||||
|
||||
public export
|
||||
Traversable (Validated e) where
|
||||
traverse f $ Valid x = Valid <$> f x
|
||||
traverse _ $ Invalid e = pure $ Invalid e
|
||||
|
||||
public export
|
||||
Uninhabited (Valid x = Invalid e) where
|
||||
uninhabited Refl impossible
|
||||
|
||||
public export
|
||||
Uninhabited (Invalid e = Valid x) where
|
||||
uninhabited Refl impossible
|
||||
|
||||
public export
|
||||
(DecEq e, DecEq a) => DecEq (Validated e a) where
|
||||
decEq (Valid x) (Invalid y) = No uninhabited
|
||||
decEq (Invalid x) (Valid y) = No uninhabited
|
||||
decEq (Valid x) (Valid y) with (decEq x y)
|
||||
decEq (Valid _) (Valid _) | Yes p = rewrite p in Yes Refl
|
||||
decEq (Valid _) (Valid _) | No up = No \case Refl => up Refl
|
||||
decEq (Invalid x) (Invalid y) with (decEq x y)
|
||||
decEq (Invalid _) (Invalid _) | Yes p = rewrite p in Yes Refl
|
||||
decEq (Invalid _) (Invalid _) | No up = No \case Refl => up Refl
|
||||
|
||||
--- Convenience representations ---
|
||||
|
||||
||| Special case of `Validated` with a `List` as an error accumulator.
|
||||
public export %inline
|
||||
ValidatedL : Type -> Type -> Type
|
||||
ValidatedL e a = Validated (List1 e) a
|
||||
|
||||
public export %inline
|
||||
oneInvalid : e -> Applicative f => Validated (f e) a
|
||||
oneInvalid x = Invalid $ pure x
|
||||
|
||||
--- Conversions to and from `Either` ---
|
||||
|
||||
public export %inline
|
||||
fromEither : Either e a -> Validated e a
|
||||
fromEither $ Right x = Valid x
|
||||
fromEither $ Left e = Invalid e
|
||||
|
||||
public export %inline
|
||||
fromEitherL : Either e a -> ValidatedL e a
|
||||
fromEitherL $ Right x = Valid x
|
||||
fromEitherL $ Left e = oneInvalid e
|
||||
|
||||
public export %inline
|
||||
toEither : Validated e a -> Either e a
|
||||
toEither $ Valid x = Right x
|
||||
toEither $ Invalid e = Left e
|
||||
|
||||
--- Conversions to and from `Maybe` ---
|
||||
|
||||
public export %inline
|
||||
fromMaybe : Monoid e => Maybe a -> Validated e a
|
||||
fromMaybe $ Just x = Valid x
|
||||
fromMaybe $ Nothing = empty
|
||||
|
||||
public export %inline
|
||||
toMaybe : Validated e a -> Maybe a
|
||||
toMaybe $ Valid x = Just x
|
||||
toMaybe $ Invalid _ = Nothing
|
||||
|
||||
--- Property of being valid ---
|
||||
|
||||
public export
|
||||
data IsValid : Validated e a -> Type where
|
||||
ItIsValid : IsValid $ Valid x
|
||||
|
||||
export
|
||||
Uninhabited (IsValid $ Invalid e) where
|
||||
uninhabited ItIsValid impossible
|
||||
|
||||
public export
|
||||
isItValid : (v : Validated e a) -> Dec (IsValid v)
|
||||
isItValid $ Valid _ = Yes ItIsValid
|
||||
isItValid $ Invalid _ = No absurd
|
@ -97,6 +97,8 @@ modules = Control.ANSI,
|
||||
|
||||
Data.Tree.Perfect,
|
||||
|
||||
Data.Validated,
|
||||
|
||||
Data.Vect.Binary,
|
||||
|
||||
Data.Vect.Sort,
|
||||
|
Loading…
Reference in New Issue
Block a user