I am studying Haskell currently and try to understand a project that uses Haskell to implement cryptographic algorithms. After reading Learn You a Haskell for Great Good online, I begin to understand the code in that project. Then I found I am stuck at the following code with the "@" symbol:
-- | Generate an @n@-dimensional secret key over @rq@.
genKey :: forall rq rnd n . (MonadRandom rnd, Random rq, Reflects n Int)
=> rnd (PRFKey n rq)
genKey = fmap Key $ randomMtx 1 $ value @n
Here the randomMtx is defined as follows:
-- | A random matrix having a given number of rows and columns.
randomMtx :: (MonadRandom rnd, Random a) => Int -> Int -> rnd (Matrix a)
randomMtx r c = M.fromList r c <$> replicateM (r*c) getRandom
And PRFKey is defined below:
-- | A PRF secret key of dimension @n@ over ring @a@.
newtype PRFKey n a = Key { key :: Matrix a }
All information sources I can find say that @ is the as-pattern, but this piece of code is apparently not that case. I have checked the online tutorial, blogs and even the Haskell 2010 language report at https://www.haskell.org/definition/haskell2010.pdf. There is simply no answer to this question.
More code snippets can be found in this project using @ in this way too:
-- | Generate public parameters (\( \mathbf{A}_0 \) and \(
-- \mathbf{A}_1 \)) for @n@-dimensional secret keys over a ring @rq@
-- for gadget indicated by @gad@.
genParams :: forall gad rq rnd n .
(MonadRandom rnd, Random rq, Reflects n Int, Gadget gad rq)
=> rnd (PRFParams n gad rq)
genParams = let len = length $ gadget @gad @rq
n = value @n
in Params <$> (randomMtx n (n*len)) <*> (randomMtx n (n*len))
I deeply appreciate any help on this.
That
@n
is an advanced feature of modern Haskell, which is usually not covered by tutorials like LYAH, nor can be found the the Report.It's called a type application and is a GHC language extension. To understand it, consider this simple polymorphic function
Intuitively calling
dup
works as follows:a
x
of the previously chosen typea
dup
then answers with a value of type(a,a)
In a sense,
dup
takes two arguments: the typea
and the valuex :: a
. However, GHC is usually able to infer the typea
(e.g. fromx
, or from the context where we are usingdup
), so we usually pass only one argument todup
, namelyx
. For instance, we haveNow, what if we want to pass
a
explicitly? Well, in that case we can turn on theTypeApplications
extension, and writeNote the
@...
arguments carrying types (not values). Those are something that exists at compile time, only -- at runtime the argument does not exist.Why do we want that? Well, sometimes there is no
x
around, and we want to prod the compiler to choose the righta
. E.g.Type applications are often useful in combination with some other extensions which make type inference unfeasible for GHC, like ambiguous types or type families. I won't discuss those, but you can simply understand that sometimes you really need to help the compiler, especially when using powerful type-level features.
Now, about your specific case. I don't have all the details, I don't know the library, but it's very likely that your
n
represents a kind of natural-number value at the type level. Here we are diving in rather advanced extensions, like the above-mentioned ones plusDataKinds
, maybeGADTs
, and some typeclass machinery. While I can't explain everything, hopefully I can provide some basic insight. Intuitively,takes as argument
@n
, a kind-of compile-time natural, which is not passed at runtime. Instead,takes
@n
(compile-time), together with a proof thatn
satisfies constraintC n
. The latter is a run-time argument, which might expose the actual value ofn
. Indeed, in your case, I guess you have something vaguely resemblingwhich essentially allows the code to bring the type-level natural to the term-level, essentially accessing the "type" as a "value". (The above type is considered an "ambiguous" one, by the way -- you really need
@n
to disambiguate.)Finally: why should one want to pass
n
at the type level if we then later on convert that to the term level? Wouldn't be easier to simply write out functions likeinstead of the more cumbersome
The honest answer is: yes, it would be easier. However, having
n
at the type level allows the compiler to perform more static checks. For instance, you might want a type to represent "integers modulon
", and allow adding those. Havingworks, but there is no check that
x
andy
are of the same modulus. We might add apples and oranges, if we are not careful. We could instead writewhich is better, but still allows to call
foo 5 x y
even whenn
is not5
. Not good. Instead,prevents things to go wrong. The compiler statically checks everything. The code is harder to use, yes, but in a sense making it harder to use is the whole point: we want to make it impossible for the user to try adding something of the wrong modulus.
Concluding: these are very advanced extensions. If you're a beginner, you will need to slowly progress towards these techniques. Don't be discouraged if you can't grasp them after only a short study, it does take some time. Make a small step at a time, solve some exercises for each feature to understand the point of it. And you'll always have StackOverflow when you are stuck :-)