Update: Got the link! The Trivial Monad. Thanks, John. Naturally, all interesting Haskell things eventually lead back to Dan Piponi.
I heard about this from John Wiegley a while ago, but every time I recall it, I can’t remember how it goes, so I thought I’d write it down for myself. I think there’s a paper about it, but I can’t find it. Hopefully I’m recalling it correctly.
The Identity monad trick: Let’s say I want to expose an API that lets you work with a data structure. I want you to be able to keep hold of that data structure and pass it back into my library, and I’ll give it back to you later and we can go back and forth.
But I don’t want you to actually give you the data structure freely so you can go and give it to your friends. So instead I force you into the Identity monad, via a newtype wrapper that only I can unpack.
newtype Secret a = Secret { unSecret :: Identity a }
  deriving (Monad,Functor,Applicative)And I have some function exposing it like:
getSecret :: Foo -> Secret TextHere, use that. What can you do with it? You can’t extract the value out, you can only compose it with more functor or monad stuff:
fmap (map T.toUpper) (getSecret foo)Or:
do text <- getSecret foo
   if all T.isUpper text
      then return (T.reverse text)
      else return textNote that the whole type of this expression is
Secret Text. You still don’t have the secret, you’ve got a
computation over it.
You’ve used the value, but it never escaped1 the actual Identity monad. It’s like I’m giving you the value, but I’m also not giving you the value.
As always, there’s a difference between “secure against your own stupidity” and “secure against attackers.” For the former, this is satisfied.
For the latter, bottom complicates it, so you should force it in the IO monad and catch any exceptions e.g.
extract :: Secret a -> IO (Maybe a)
This prevents people from using
(v >>= \a -> error ("The value is " ++ show a))
To try to get around it.
unsafePerformIO and other non-standard Haskell can get
around it, but if you’re defending against developers, you probably have
control over the environment, so you can just white-list the imports and
extensions and there’s nothing they can do. This is what tryhaskell (via
mueval) does.↩︎