Modules
Aiken programs are made up of bundles of functions and types called modules. Each module has its own namespace and can export types and values to be used by other modules in the program.
// inside module lib/straw_hats/sunny.ak
fn count_down() {
"3... 2... 1..."
}
fn blast_off() {
"BOOM!"
}
pub fn set_sail() {
[
count_down(),
blast_off(),
]
}
Here we can see a module named straw_hats/sunny
, the name determined by the
filename lib/straw_hats/sunny.ak
. Typically all the modules for one
project would live within a directory with the name of the project, such as
straw_hats
in this example.
The pub
keyword makes this type usable from other modules.
For the functions count_down
and blast_off
we have omitted the pub
keyword, so these functions are private module functions. They can only be
called by other functions within the same module.
Functions, type-aliases and constants can all be exported from a module using
the pub
keyword.
Qualified imports
To use functions or types from another module we need to import them using the
use
keyword.
// inside module src/straw_hats/laugh_tale.ak
use straw_hats/sunny
pub fn find_the_one_piece() {
sunny.set_sail()
}
Custom names
It is also possible to give a module a custom name
when importing it using the as
keyword.
use unix/dog
use animal/dog as kitty
This may be useful to differentiate between multiple modules that would have the same default name when imported.
The definition use straw_hats/sunny
creates a new variable with the name
sunny
and the value of the sunny
module.
In the find_the_one_piece
function we call the imported module's public set_sail
function using the .
operator. If we had attempted to call count_down
it
would result in a compile time error as this function is private to the
sunny
module.
Unqualified import
Values and types can also be imported in an unqualified fashion.
use animal/dog.{Dog, stroke}
pub fn foo() {
let puppy = Dog { name: "Zeus" }
stroke(puppy)
}
This may be useful for values that are used frequently in a module, but generally qualified imports are preferred as it makes it clearer where the value is defined.
You may also combine unqualified imports with custom names as such:
use animal/dog.{Dog, stroke} as kitty
Opaque types
At times it may be useful to create a type and make the constructors and fields private so that users of this type can only use the type through publicly exported functions.
For example we can create a Counter
type which holds an int which can be
incremented. We don't want the user to alter the Int
value other than by
incrementing it, so we can make the type opaque to prevent them from being
able to do this.
// The type is defined with the opaque keyword
pub opaque type Counter {
Counter(value: Int)
}
pub fn new() {
Counter(0)
}
pub fn increment(counter: Counter) {
Counter(counter.value + 1)
}
Because the Counter
type has been marked as opaque
it is not possible for
code in other modules to construct or pattern match on counter values or
access the value
field. Instead other modules have to manipulate the opaque
type using the exported functions from the module, in this case new
and
increment
.
It isn't possible to downcast from Data
into an opaque type because it
isn't generally possibly to know what restrictions applies to that type. For
example, imagine a Rational
opaque type holding a pair of integer values as
numerator and denominator.
By using an opaque type, one can ensure that the denominator remains always
non-negative, and that both numbers remain coprime. Hence, it is in many cases
unsafe to downcast any arbitrary pair of integer into a Rational
since there's
more than just a structural equivalence at play.
For the same reasons, it isn't possible to use an opaque type (or any type holding an opaque type) as a validator's datum and/or redeemer.
There's a special treatment for an opaque type with only one constructor,
and if that constructor has only 1 field. Under the hood, it behaves like a
Haskell's newtype
. This matters when we're dealing with CBOR.
For example:
pub opaque type NewType<a> { field: a }
pub fn new_type(a) -> NewType<a> { NewType(a) }
..
trace new_type(43) // 43
// Compare that to a non-opaque type:
pub type Constr<a> { field: a }
pub fn new_constr(a) -> Constr<a> { Constr(a) }
..
trace new_constr(43) // 121([_ 43])
Well-known modules
The prelude module
There are two modules that are built into the language, the first is the
aiken
prelude module. By default its types and values are automatically
imported into every module you write, but you can still chose to import it the
regular way. This may be useful if you have created a type or value with the
same name as an item from the prelude.
use aiken
/// This definition locally overrides the `Option` type
/// and the `Some` constructor.
pub type Option {
Some
}
/// The original `Option` and `Some` can still be used
pub fn go() -> aiken.Option<Int> {
aiken.Some(1)
}
The content of the Prelude module is documented in aiken-lang/prelude (opens in a new tab)
The builtin module
The second module that comes with the language is for exposing useful builtin functions from Plutus core. Most underlying platform functions are available here using a "snake_case" name. Much of Aiken's syntax ends up compiling to combinations of certain bultins but many aren't "exposed" through the syntax and need to be used directly. The standard library wraps these in a more Aiken-friendly interface so you'll probably never need to use these directly unless you're making your own standard library.
use aiken/builtin
fn eq(a, b) {
builtin.equals_integer(a, b)
}
// is implicitly used when doing:
a == b
The content of the builtin module is documented in aiken-lang/prelude (opens in a new tab).
Conditional modules
Environments
Since v1.1.0
, Aiken supports conditional environment modules. Environment modules follow special rules:
- They must be located in an
env
directory at the root of the project. - When used, at least one of them must be called
default.ak
; it is used by default when no explicit module is provided. - Only one of them is available at the time, in a given compilation pass.
- From within other modules, they are always referred to as
env
, regardless of their actual name.
So for example, imagine the following project structure:
.
├── aiken.toml
├── env
│ ├── default.ak
│ ├── preprod.ak
│ └── preview.ak
└── validators
└── main.ak
From within main
, one can assume the existence of a module env
and import it like any other module use env
. The content of the module depends on the --env
argument passed to the command used to check
or build
the project. When --env
is omitted, it is assumed to be default.ak
. Note that each of those modules must define a similar API. Usually, they will export constants that are used throughout the validator but that depend on the execution environment.
For example:
use env
validator main() {
mint(redeemer, policy_id, self) {
self.signatories
|> list.has(env.administrator)
}
}
pub const administrator = #"0000000000"
The environment names are not restricted (other than the usual restriction for Aiken's module names). They can contain arbitrary logic and definitions and import other modules as well as other environment modules by explicitly importing them as use {env}
(e.g. use preview
). Rules regarding import cycles, however, still apply.
Configurations
Environment modules are quite flexible and expressive. They are useful when needing to wield complex dynamic configurations together which may even leverage other functions. In many cases though, their full power isn't needed and one only needs to define a handful of static constants. For these scenarios, Aiken provides a similar features called conditional configuration values.
Those configurations are defined directly in the aiken.toml
and can represent either integers, booleans, bytearrays or lists / tuples of thoses. Like environment, they expect one configuration to be named default
. All values under it are then injected into a special module config
which can be imported in the usual ways.
Here's a thorough example of a conditional configuration, with their Aiken's equivalent:
name = "aiken-lang/scratchpad"
version = "1.0.0"
[config.default]
price = 1000000 # pub const price: Int = 1_000_000
is_mainnet = true # pub const is_mainnet: Bool = True
network = "mainnet" # pub const network: ByteArray = "mainnet"
quotas = [1, 2, 3] # pub const ratio: List<Int> = [1, 2, 3]
asset = ["HOSKY", 42] # pub const asset: (ByteArray, Int) = ("HOSKY", 42)
[config.default.owner] # pub const owner: ByteArray = #"0000111122223333"
bytes = "0000111122223333"
encoding = "hex"
Then, one can refer to any of the configuration value as they would from a standard module:
use config
fn main() {
if config.is_mainnet {
config.price * 2
} else {
..
}
}
Like conditional environment modules, the --env
option drives the selection of the right configuration values. Yet unlike modules, configuration values cannot refer to other configuration values, and are limited to what the TOML syntax supports.
Documentation
You may add user-facing documentation at the head of modules with a module
documentation comment ////
(quadruple slash!) per line. Markdown is supported
and this text block will be included with the module's entry in generated HTML
documentation.
Hiding module
At times, you may want to hide modules from the generated documentation because they may contains internal functions that are not relevant to expose to any users, but split out for various reasons.
To hide a module, you can start the module documentation with the tag @hidden
.
//// @hidden
//// Some more documentation