Sometimes I feel the need to selectively allow or disallow alternatives

This seems like the kind of lightweight property that Liquid Haskell (a GHC plugin) is especially suited for. Here's a small example of "at some places in the program we want the type system to enforce that only the constructor LeftFoo can be used."

data Foo = LeftFoo !Int | RightFoo !Char

{-@ measure isLeftFoo @-}
isLeftFoo :: Foo -> Bool
isLeftFoo LeftFoo{} = True
isLeftFoo _ = False

{-@ takesOnlyLeftFoo :: {x:Foo | isLeftFoo x} -> Int @-}
takesOnlyLeftFoo :: Foo -> Int
takesOnlyLeftFoo (LeftFoo i) = i


I've taken Olaf's hypothetical Foo and swapped in Int for A and Char for B, defined a function isLeftFoo that returns True for LeftFoo-constructed Foos, and I call isLeftFoo in the type of a function that is meant to only be called with LeftFoo values.

This syntax in the mushroom-at-symbol comments is a refinement type. It has the same structure as the haskell type `Foo -> Int` but adds a precondition on the input that isLeftFoo returns True. When you compile your project with Liquid Haskell, the precondition will be checked at compile time by checking that callers establish the fact on any value they pass into takesOnlyLeftFoo.