
On Wed, Jun 22, 2011 at 1:25 PM, pipoca
Also, I don't think that the formulation of (:|:) above is sufficient. Suppose:
foo :: Foo -> Bar baz :: Baz -> Quux
foobaz :: [Foo :|: Baz]
-- map foo and baz over foobaz barquux :: [Bar :|: Quux] barquux = map f foobaz how would f be implemented?
It seems to me that you'd need an additional function: either' :: (a -> c) -> (b -> d) -> Either a b -> Either c d
Sure, you would want to treat Either as a bifunctor. Your either' function is usually called bimap. So the call would look like: map (bimap f g) foobaz for some f and g.
However, it still seems to me that that isn't sufficient. Suppose instead:
a :: A -> B :|: C d :: D -> E :|: F
ad :: [A :|: D]
if we want to map a and d over ad using either' to get bcef :: [B :|: C :|: E :|: F] it wouldn't work, we'd get bcef :: [(B :|: C) :|: (E :|: F)] i.e. bcef :: [Either (Either B C) (Either E F)] instead, which is presumably not what we wanted...
You need is a proof that :|: is associative. In other words, we need to pick a "normal form" for expressions like (a :|: (b :|: c)) and ((a :|: b) :|: c) and show that they can all be normalized into the normal form. I use a typeclass: class Monic a b where inject :: a -> b for that kind of work. I always have an instance instance Monic a a where inject = id so I might pick to write: instance Monic ((a :|: b) :|: c) (a :|: (b :|: c)) where inject (Left (Left a)) = Left a inject (Left (Right b)) = (Right (Left b)) inject (Right c) = (Right (Right c)) If you do that, you will soon run into two other problems I can think of (relating to MacLane's coherence conditions, which I think would be fixed by judicious composition with "inject"). You're building up (Either a b) into a monoidal category. There used to be a package called category-extras for this kind of stuff. I think it has been broken up. Does anybody know the status of its replacement(s)?