When this guide uses the term "Plutarch Type" we explicitly talk about a type of kind PType
. We will refer to " types of kind PType
" simply as PType
s. We explicitly qualify when referring to the kind PType
.
Note to beginners: Plutarch uses a language extension called
DataKinds
. This means that there are kinds beyondType
(aka*
). We refer the read to [3] for an extended beginner-level introduction to these concepts if desired.
PType
is defined as type PType = S -> Type
; that is, it is a kind synonym for S -> Type
(where S
and Type
are themselves kinds). This synonym is important to keep in mind because when querying the kind of something like PBool
in, say, GHCi, we will not see PType
as the kind. Instead, we get
ghci> :k PBool
PBool :: S -> Type
Thus, any time we see the kind S -> Type
, we should mentally substitute its kind synonym PType
. We reiterate: types of kind PType
, should be considered as tags on computation. They do not represent types of values in the same way as standard Haskell types.
The kind of basic types such as Integer
in Haskell has the kind: Type
; the corresponding "basic" kind in Plutarch is simply PType
. Higher-kinded types in Haskell, such as Maybe
, will kinds such as Type -> Type
. In Plutarch, the corresponding kind is:
ghci> :k PMaybe
PMaybe :: PType -> S -> Type
Since the kind arrow ->
is right-associative, we first read this as PMaybe :: PType -> (S -> Type)
; and since we know that that PType
and S -> Type
and synonyms, we read this as PMaybe :: PType -> PType
, which should agree without intuition.
The kind S -> Type
is mysterious at first, but we recall that PType
s are tags on (unexecuted) computations indicating their result type. The S
kind represents the computational context; thus, a PType
expects to receive a computational context represented by a value s
whose type has kind S
that it will tag to produce a Type
. Note that end-users never instantiate the value s
with a concrete value; it is simply a type-level mechanism to maintain functional purity.
The above notion is essential to understanding why not all PType
s have data constructors; the data constructors are irrelevant, except insofar as they enable the implementation to keep track of Haskell-level and UPLC-level representations. PInteger
is one such case; it is impossible to construct a constant y
where y :: PInteger s
. Other PType
s, such as PMaybe
, do have data constructors (specifically PJust
and PNothing
), but still do not carry data from the viewpoint of UPLC. A value such as PNothing
merely facilitates convenient term construction and deconstruction. When pcon
sees PNothing
, it knows it should build a UPLC constant that is morally equivalent to the concept of Nothing :: Maybe a
.
In general, the concrete UPLC representations are connected to Plutarch types through their PlutusType
implementation.
Also see: Figuring out the representation of a Plutarch type.