From e831d999e80130ec90627d0aff5c08a0e788f8ac Mon Sep 17 00:00:00 2001 From: Jonathan Cubides Date: Tue, 14 Jan 2025 13:15:29 +0100 Subject: [PATCH] Review essential tutorial content (#135) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Markdown formatting improvements and a few text refinements. --------- Co-authored-by: Ɓukasz Czajka <62751+lukaszcz@users.noreply.github.com> Co-authored-by: Lukasz Czajka --- .../juvix.lock.yaml | 11 + docs/tutorials/essential.juvix.md | 515 +++++++++++++----- 2 files changed, 400 insertions(+), 126 deletions(-) create mode 100644 docs/blog/posts/strictly-positive-data-types/juvix.lock.yaml diff --git a/docs/blog/posts/strictly-positive-data-types/juvix.lock.yaml b/docs/blog/posts/strictly-positive-data-types/juvix.lock.yaml new file mode 100644 index 00000000..6d69b34e --- /dev/null +++ b/docs/blog/posts/strictly-positive-data-types/juvix.lock.yaml @@ -0,0 +1,11 @@ +# This file was autogenerated by Juvix version 0.6.9. +# Do not edit this file manually. + +version: 2 +checksum: 8d026c805b15debbae84bf1db2278b0fe5c4708e1ae049a204c385259c437d5c +dependencies: +- git: + name: anoma_juvix-stdlib + ref: 01ff19f1135048be3402e094f2fc89406a44a995 + url: https://github.com/anoma/juvix-stdlib + dependencies: [] diff --git a/docs/tutorials/essential.juvix.md b/docs/tutorials/essential.juvix.md index 38f657a3..7cd59d5e 100644 --- a/docs/tutorials/essential.juvix.md +++ b/docs/tutorials/essential.juvix.md @@ -2,7 +2,7 @@ icon: material/notebook-heart comments: true search: - boost: 2 + boost: 3 tags: - tutorial - beginner @@ -18,7 +18,11 @@ import Stdlib.Prelude open; ![tara-teaching](./../assets/images/tara-teaching.svg){ align=left width="280" } -Welcome to the essential Juvix tutorial! This brief guide will introduce you to basic Juvix language features and allow you to quickly get up to speed with programming in Juvix. After working through this tutorial, you should be able to write simple Juvix programs and have a basic understanding of common patterns and language concepts. Let's get started on your Juvix journey! +Welcome to the essential Juvix tutorial! This brief guide will introduce you to +basic Juvix language features and allow you to quickly get up to speed with +programming in Juvix. After working through this tutorial, you should be able to +write simple Juvix programs and have a basic understanding of common patterns +and language concepts. Let's get started on your Juvix journey! ## Juvix REPL @@ -87,7 +91,11 @@ Stdlib.Prelude> (1, 2) Stdlib.Prelude> [1; 2; 3] 1 :: 2 :: 3 :: nil ``` -The parentheses around pairs, as in `(1, 2)`, are in fact optional when no ambiguity arises. The notation `[a; b; c]` is an abbreviation for `a :: b :: c :: nil`, where `::` is a list "cons" operator and `nil` (also `[]`) denotes the empty list. + +The parentheses around pairs, as in `(1, 2)`, are in fact optional when no +ambiguity arises. The notation `[a; b; c]` is an abbreviation for `a :: b :: c +:: nil`, where `::` is a list "cons" operator and `nil` (also `[]`) denotes the +empty list. In fact, you can use all functions and types from the [Stdlib.Prelude](https://anoma.github.io/juvix-stdlib/Stdlib.Prelude.html) @@ -118,7 +126,9 @@ Stdlib.Prelude> :type (1, "A") Pair Nat String ``` -The type `List Nat` is the type of lists whose elements have type `Nat`. The type `Pair Nat String` is the type of pairs whose first component has type `Nat` and second component has type `String`. +The type `List Nat` is the type of lists whose elements have type `Nat`. The +type `Pair Nat String` is the type of pairs whose first component has type `Nat` +and second component has type `String`. ## Files, modules and compilation @@ -126,7 +136,8 @@ Currently, the REPL does not support adding new definitions. To define new functions or data types, you need to put them in a separate file and either load the file in the REPL with `:load file.juvix`, evaluate it with the shell command `juvix eval file.juvix`, or compile it to a binary executable with `juvix -compile native file.juvix`. To only type-check a file without evaluating it, use `juvix typecheck file.juvix`. +compile native file.juvix`. To only type-check a file without evaluating it, use +`juvix typecheck file.juvix`. To conveniently edit Juvix files, an [Emacs mode](./emacs.juvix.md) and a [VSCode extension](./vscode.html) are available. @@ -166,9 +177,17 @@ Function definitions in Juvix look like this: add (x y : Nat) : Nat := x + y; ``` -In contrast to many other languages, no function definition keyword like `def` or `fun` is required. Just the name of the function needs to be provided, followed by its arguments and the result type. The function body is specified after the assignment symbol (`:=`). The definition needs to be terminated with a semicolon. +In contrast to many other languages, no function definition keyword like `def` +or `fun` is required. Just the name of the function needs to be provided, +followed by its arguments and the result type. The function body is specified +after the assignment symbol (`:=`). The definition needs to be terminated with a +semicolon. -Juvix is a statically typed language, which means that the compiler enforces a strict typing discipline with type checking done at compilation time. It is mandatory to specify the types of the function arguments and of the result. When the arguments have different types, they need to be given in separate parenthesized argument groups: +Juvix is a statically typed language, which means that the compiler enforces a +strict typing discipline with type checking done at compilation time. It is +mandatory to specify the types of the function arguments and of the result. When +the arguments have different types, they need to be given in separate +parenthesized argument groups: ```juvix addOrMul (shouldMul : Bool) (x y : Nat) : Nat := @@ -177,9 +196,16 @@ addOrMul (shouldMul : Bool) (x y : Nat) : Nat := | else := add x y; ``` -As you see above, functions are called by listing their space-separated arguments after the function name: `add x y` calls `add` with two arguments `x` and `y`. This syntax is different than in mainstream imperative languages, but common in functional languages like OCaml, Haskell or Lean. In Juvix, the expression `add (x, y)` is a call to the function `add` with a single argument `(x, y)` which is a pair of `x` and `y`. Using `add (x, y)` instead of `add x y` in `addOrMul` would result in a type error. +As you see above, functions are called by listing their space-separated +arguments after the function name: `add x y` calls `add` with two arguments `x` +and `y`. This syntax is different than in mainstream imperative languages, but +common in functional languages like OCaml, Haskell or Lean. In Juvix, the +expression `add (x, y)` is a call to the function `add` with a single argument +`(x, y)` which is a pair of `x` and `y`. Using `add (x, y)` instead of `add x y` +in `addOrMul` would result in a type error. -A "function" can have zero arguments. In this way, definitions of functions and constants / variables follow the same syntax: +A "function" can have zero arguments. In this way, definitions of functions and +constants / variables follow the same syntax: ```juvix extract-module-statements module MyValue; @@ -187,7 +213,11 @@ myValue : Nat := 42; end; ``` -Juvix functions can be parameterized by types, similar to generics in Java, Rust or C++. In functional programming terminology, such functions are called _polymorphic_. For example, here is a polymorphic identity function which takes an argument of an arbitrary type `A` and returns it: +Juvix functions can be parameterized by types, similar to generics in Java, +Rust, and C++. In functional programming terminology, such functions are called +_polymorphic_ - they act uniformly on all types, without relying on +type-specific implementations. For example, here is a polymorphic identity +function which takes an argument of an arbitrary type `A` and returns it: ```juvix extract-module-statements module IdPolymorphic; @@ -195,13 +225,19 @@ id {A} (x : A) : A := x; end; ``` -When calling `id`, the right type to be substituted for `A` is inferred automatically by the type checker. For example, in the call `id 3` the type checker infers that `A` must be `Nat`. +When calling `id`, the right type to be substituted for `A` is inferred +automatically by the type checker. For example, in +the call `id 3` the type checker infers that `A` refers to the type `Nat`. + +!!! note -Note that `id` already exists in the standard library, so redefining it in your file may lead to name clashes. + The function `id` already exists in the standard library, so redefining it + in your file may lead to name clashes. ## Local definitions -Local definitions visible only inside a function body are introduced with the `let .. in ..` syntax: +Local definitions visible only inside a function body are introduced with the +`let .. in ..` syntax: ```juvix foo (pair : Pair Nat Nat) : Nat := @@ -212,28 +248,55 @@ foo (pair : Pair Nat Nat) : Nat := in bang (add x bar); ``` -The identifiers `x`, `y`, `bar` and `bang` are visible only inside the `foo` function after they are defined. The definitions in a `let`-block follow the same syntax as top-level definitions. In particular, it is possible to define local functions, like `bang` above. Type annotations for variables or data structure patterns, like for `bar` or `(x, y)` above, can be omitted and automatically inferred by the type checker. +The identifiers `x`, `y`, `bar`, and `bang` are only accessible within the `foo` +function after their declaration. The definitions in a `let`-block follow the +same syntax as top-level definitions. In particular, it is possible to define +local functions, like `bang` above. Type annotations for variables or data +structure patterns, like for `bar` or `(x, y)` above, can be omitted and +automatically inferred by the type checker. ## Functional programming -Juvix is a purely functional language, which means that functions have no side effects and all variables are immutable. Once a variable is assigned a value it cannot be modified - destructive updates are simply absent from the language. Instead of mutating existing variables or data structures, a common functional programming pattern is to "update" them by creating new ones. +Juvix is a purely functional language, which means that functions have no side +effects and all variables are immutable. Once a variable is assigned a value it +cannot be modified - destructive updates are simply absent from the language. +Instead of mutating existing variables or data structures, a common functional +programming pattern is to "update" them by creating new ones. -For example, a polymorphic function that swaps the components of a pair could be implemented as follows: +For example, a polymorphic function that swaps the components of a pair could be +implemented as follows: ```juvix swap {A B} (pair : Pair A B) : Pair B A := (snd pair, fst pair); ``` -Instead of modifying the pair in place, `swap` returns a new pair with the components swapped. The standard library functions `fst` and `snd` return the first and second components of a pair, respectively. - -Purely functional programming may at first seem exotic to a mainstream imperative programmer, but once assimilated it offers unique advantages. Purity guarantees that all functions are _referentially transparent_, meaning that they always return the same result for the same arguments. This is not the case in imperative programs, where the result of a function may depend on the implicit global state or the call may have side effects, which is a common source of errors. Functional programs are often more succinct, reliable, and easier to reason about. In particular, [formal verification](https://ethereum.org/en/developers/docs/smart-contracts/formal-verification/) is more manageable for functional than for imperative programs. - -Functional programming does require a certain shift in the developer's mindset. Nonetheless, the learning curve of a well-designed user-focused purely functional language should not be too steep. Juvix aims to deliver on this promise. +Instead of modifying the pair in place, `swap` returns a new pair with the +components swapped. The standard library functions `fst` and `snd` return the +first and second components of a pair, respectively. + +Purely functional programming may at first seem exotic to a mainstream +imperative programmer, but once assimilated it offers unique advantages. Purity +guarantees that all functions are _referentially transparent_, meaning that they +always return the same result for the same arguments. This is not the case in +imperative programs, where the result of a function may depend on the implicit +global state or the call may have side effects, which is a common source of +errors. Functional programs are often more succinct, reliable, and easier to +reason about. In particular, [formal +verification](https://ethereum.org/en/developers/docs/smart-contracts/formal-verification/) +is more manageable for functional than for imperative programs. + +Functional programming does require a certain shift in the developer's mindset. +Nonetheless, the learning curve of a well-designed user-focused purely +functional language should not be too steep. Juvix aims to deliver on this +promise. ## Records -A statically typed programming language would not be very useful without the ability to define new data types. Records with named fields of specified types are among the most common data types. In Juvix, a record type can be defined as follows. +A statically typed programming language would not be very useful without the +ability to define new data types. Records with named fields of specified types +are among the most common data types. In Juvix, a record type can be defined as +follows. ```juvix type Resource := mkResource@{ @@ -242,7 +305,9 @@ type Resource := mkResource@{ }; ``` -The above defines a record type `Resource` with two fields `quantity` and `price`, both of type `Nat`. A record of type `Resource` can be created using the _record constructor_ `mkResource`: +The above defines a record type `Resource` with two fields `quantity` and +`price`, both of type `Nat`. A record of type `Resource` can be created using +the _record constructor_ `mkResource`: ```juvix myResource : Resource := mkResource@{ @@ -251,7 +316,7 @@ myResource : Resource := mkResource@{ }; ``` -The fields of a record can be accessed with _record projections_ (`Resource.quantity` and `Resource.price` below). +The fields of a record can be accessed via their _record projections_, as demonstrated in the following function with `Resource.quantity` and `Resource.price`. ```juvix totalCost (r : Resource) : Nat := @@ -292,9 +357,12 @@ type Ordering := end; ``` -The above defines a type `Ordering` with three possible values: `LessThan`, `Equal`, `GreaterThan`. The values of an enumeration type are its _constructors_. +The above defines a type `Ordering` with three possible values: `LessThan`, +`Equal`, and `GreaterThan`. The values of an enumeration type are its +_constructors_. -Distinguishing between different constructors can be achieved using `case`-expressions: +Distinguishing between different constructors can be achieved using +`case`-expressions: ```juvix orderingToInt (ord : Ordering) : Int := @@ -304,7 +372,12 @@ orderingToInt (ord : Ordering) : Int := | GreaterThan := 1; ``` -Records and enumerations are special cases of _inductive types_ specified by a number of constructors with arguments. Here is an example of an inductive type with two constructors `Created` and `Consumed`, each with one argument of type `Nat`. +Records and enumerations are special cases of _inductive types_ specified by a +number of constructors with arguments. Here is an example of an inductive type +with two constructors: + +- `Created` with one argument `commitment` of type `Nat`, +- `Consumed` with one argument `nullifier` of type `Nat`. ```juvix extract-module-statements module Tag0; @@ -318,7 +391,10 @@ type Tag := end; ``` -Actually, it is not required to name the arguments of a constructor - their types can be specified in a space-separated sequence instead. The names are often omitted when the constructor has only one argument with the argument being a record and/or the argument name being insignificant. +Actually, it is not required to name the arguments of a constructor - their +types can be specified in a space-separated sequence instead. The names are +often omitted when the constructor has only one argument with the argument being +a record and/or the argument name being insignificant. ```juvix type Commitment := mkCommitment@{ @@ -332,11 +408,16 @@ type Tag := | Consumed Nullifier; ``` -For most types with multiple constructors, it is considered good practice to wrap constructor arguments into records. This makes type information more explicit and allows to pass all constructor arguments together to a helper function. +For most types with multiple constructors, it is considered good practice to +wrap constructor arguments into records. This makes type information more +explicit and allows to pass all constructor arguments together to a helper +function. ## Optional values -The polymorphic inductive type `Maybe` from the standard library is commonly used to represent optional or nullable values in a type-safe manner. It is analogous to `Option` in Rust, `option` in OCaml or `Maybe` in Haskell. +The polymorphic inductive type `Maybe` from the standard library is commonly +used to represent optional or nullable values in a type-safe manner. It is +analogous to `Option` in Rust, `option` in OCaml or `Maybe` in Haskell. ```juvix extract-module-statements module myMaybe; @@ -347,6 +428,7 @@ end; ``` Here are two standard library functions commonly used with the `Maybe` type: + ```juvix extract-module-statements module myMaybeFunctions; isJust {A} (maybeValue : Maybe A) : Bool := @@ -361,7 +443,10 @@ fromMaybe {A} (default : A) (maybeValue : Maybe A) : A := end; ``` -Juvix requires all functions to be total, i.e., to return a result for all possible argument values. One way of handling partial functions is to use the `Maybe` type for the result with `nothing` returned when the result is undefined. +In Juvix, all functions must be total, meaning they must return a result for +every possible input. To handle cases where a function might not produce a result +for certain inputs, you can use the `Maybe` type. When the result is undefined, +the function can return `nothing`. ```juvix getCommitment (tag : Tag) : Maybe Commitment := @@ -373,11 +458,16 @@ getCommitmentD (default : Commitment) (tag : Tag) : Commitment := fromMaybe default (getCommitment tag); ``` -The function `getCommitmentD` returns its first argument `default` if the `tag` does not contain a commitment. +The function `getCommitmentD` returns its first argument `default` if the `tag` +does not contain a commitment. ## Lists and iteration -In the Juvix standard library, the list type is defined as a polymorphic inductive type with two constructors `nil` (empty list `[]`) and `::` ("cons" - a non-empty list with a head and a tail). +In the Juvix standard library, the list type is defined as a polymorphic +inductive type with two constructors: + +- `nil` - empty list `[]`, +- `::` - "cons" - a non-empty list with a head and a tail. ```juvix extract-module-statements module ListDefinition; @@ -387,13 +477,28 @@ type List A := end; ``` -Recall that constructor arguments can be specified without naming them by listing their types after the constructor name, like above for the arguments of `::`. The first argument of `::` is the _head_ (first list element) and the second argument is the _tail_ (remaining list elements). - -The "cons" constructor `::` can be used in right-associative infix notation, e.g., `1 :: 2 :: 3 :: nil` is the same as `1 :: (2 :: (3 :: nil))` which is the same as `(::) 1 ((::) 2 ((::) 3 nil))`. So if `lst` is a list of `Nat`s, then `1 :: lst` is the list with head `1` and tail `lst`, i.e., the first element of `1 :: lst` is `1` and the remaining elements come from `lst`. When specifying all list elements at once, a move convenient notation `[1; 2; 3]` can be used. So `[1; 2; 3]` is the same as `1 :: [2; 3]`, `1 :: 2 :: [3]`, `1 :: 2 :: 3 :: []` and `1 :: 2 :: 3 :: nil`. - -Lists are fundamental data structures in functional programming which represent ordered sequences of elements. Lists are often used where an imperative program would use arrays, but these are not always directly interchangeable. To use lists, imperative array-based code needs to be reformulated to process elements sequentially. - -In Juvix, list processing is often performed with the `for` iterator. The expression +Recall that constructor arguments can be specified without naming them by +listing their types after the constructor name, like above for the arguments of +`::`. The first argument of `::` is the _head_ (first list element) and the +second argument is the _tail_ (remaining list elements). + +The "cons" constructor `::` can be used in right-associative infix notation, +e.g., `1 :: 2 :: 3 :: nil` is the same as `1 :: (2 :: (3 :: nil))` which is the +same as `(::) 1 ((::) 2 ((::) 3 nil))`. So if `lst` is a list of `Nat`s, then `1 +:: lst` is the list with head `1` and tail `lst`, i.e., the first element of `1 +:: lst` is `1` and the remaining elements come from `lst`. When specifying all +list elements at once, a move convenient notation `[1; 2; 3]` can be used. So +`[1; 2; 3]` is the same as `1 :: [2; 3]`, `1 :: 2 :: [3]`, `1 :: 2 :: 3 :: []` +and `1 :: 2 :: 3 :: nil`. + +Lists are fundamental data structures in functional programming which represent +ordered sequences of elements. Lists are often used where an imperative program +would use arrays, but these are not always directly interchangeable. To use +lists, imperative array-based code needs to be reformulated to process elements +sequentially. + +In Juvix, list processing is often performed with the `for` iterator. The +expression ``` for (acc := initialValue) (x in list) { @@ -401,7 +506,14 @@ for (acc := initialValue) (x in list) { } ``` -is evaluated as follows. First, the _accumulator_ `acc` is assigned its initial value. Then each list element is processed sequentially in the order from left to right (from beginning to end of list). At each step, `BODY` is evaluated with the current value of `acc` and the current element `x`. The result of evaluating `BODY` becomes the new value of `acc`, and the iteration proceeds with the next element. The final value of the accumulator `acc` becomes the value of the whole `for`-expression. +is evaluated as follows. First, the _accumulator_ `acc` is assigned its initial +value. Then each list +element is processed sequentially in the order from left to right (from +beginning to end of list). At each step, `BODY` is evaluated with the current +value of `acc` and the current element `x`. The result of evaluating `BODY` +becomes the new value of `acc`, and the iteration proceeds with the next +element. The final value of the accumulator `acc` becomes the value of the whole +`for`-expression. For example, here is a function which sums the elements of a list of natural numbers: @@ -412,9 +524,14 @@ sum (lst : List Nat) : Nat := }; ``` -First, `acc` is initialized to 0. Going through the list from left to right, at each step the current element `x` is added to `acc` giving the new value of `acc`. At the end, `acc` is equal to the sum of all elements in the list. +First, the accumulator `acc` is initialized to 0. As the list `lst` is +traversed from left to right, each element `x` is added to `acc`, updating its +value. By the end of the iteration, `acc` holds the sum of all elements in the +list. -As another example, consider a function computing the total cost of all resources priced at more than 100 (recall `Resource` and `totalCost` defined earlier in this tutorial). +As another example, consider a function computing the total cost of all +resources priced at more than 100 (recall `Resource` and `totalCost` defined +earlier in this tutorial). ```juvix totalCostOfValuableResources (lst : List Resource) : Nat := @@ -425,7 +542,10 @@ totalCostOfValuableResources (lst : List Resource) : Nat := }; ``` -The next example demonstrates how to perform iteration with multiple accumulators. The following function `minmax` computes the minimum and the maximum in a list of natural numbers. The functions `min` and `max` are defined in the standard library. +The next example demonstrates how to perform iteration with multiple +accumulators. The following function `minmax` computes the minimum and the +maximum values in a list of natural numbers. The functions `min` and `max` are +defined in the standard library. ```juvix minmax (lst : List Nat) : Pair Nat Nat := @@ -434,7 +554,10 @@ minmax (lst : List Nat) : Pair Nat Nat := }; ``` -You can iterate over multiple lists simultaneously with the help of the `zip` function, which combines two lists into a list of pairs. For example, `zip [1; 2; 3] [4; 5; 6]` evaluates to `[(1, 4); (2, 5); (3, 6)]`. As an illustration, here is an implementation of the dot product for two lists of the same length. +You can iterate over multiple lists simultaneously with the help of the `zip` +function, which combines two lists into a list of pairs. For example, `zip [1; +2; 3] [4; 5; 6]` evaluates to `[(1, 4); (2, 5); (3, 6)]`. As an illustration, +here is an implementation of the dot product for two lists of the same length. ```juvix dotProduct (lst1 lst2 : List Nat) : Nat := @@ -443,7 +566,8 @@ dotProduct (lst1 lst2 : List Nat) : Nat := }; ``` -The dot product is the sum of products of elements at corresponding positions, e.g., +The dot product is the sum of products of elements at corresponding positions, +e.g., ``` dotProduct [1; 2; 3] [4; 5; 6] @@ -463,6 +587,7 @@ sumAllProducts (lst1 lst2 : List Nat) : Nat := ``` For example: + ``` sumAllProducts [1; 2; 3] [4; 5; 6] = 1 * 4 + 1 * 5 + 1 * 6 + @@ -471,7 +596,9 @@ sumAllProducts [1; 2; 3] [4; 5; 6] = 90 ``` -The `for` iterator is suitable for sequentially accumulating values from a list. Often, we want to transform a list into a new list. The `map` and `filter` iterators are commonly used for this purpose. +The `for` iterator is suitable for sequentially accumulating values from a list. +Often, we want to transform a list into a new list. The `map` and `filter` +iterators are commonly used for this purpose. The expression @@ -479,7 +606,9 @@ The expression map (x in lst) {f x} ``` -evaluates to a list obtained from `lst` by replacing each element `x` with `f x`. For example, the following function increases the prices of all resources by `n`. +evaluates to a list obtained from `lst` by replacing each element `x` with `f +x`. For example, the following function increases the prices of all resources by +`n`. ```juvix increaseAllPrices (n : Nat) (lst : List Resource) : List Resource := @@ -490,7 +619,9 @@ increaseAllPrices (n : Nat) (lst : List Resource) : List Resource := }; ``` -The `filter` iterator picks out elements of a list that satisfy a given condition. For example, the following function picks resources with price greater than `price`. The order of the elements is preserved. +The `filter` iterator selects elements of a list that satisfy a given +condition. For example, the following function picks resources with price +greater than `price`. The order of the elements is preserved. ```juvix pickValuable (price : Nat) (lst : List Resource) : List Resource := @@ -499,13 +630,31 @@ pickValuable (price : Nat) (lst : List Resource) : List Resource := }; ``` -Lists do not allow for random access to their elements by index. The Juvix standard library intentionally does not provide a function to access the nth element of a list. Such a function could easily be implemented, but it would not be efficient. Lists are *not* arrays. If you find yourself wanting to access list elements by numerical index, you are most likely doing something very wrong: trying to incongruently fit imperative array programming patterns into a functional language instead of using more elegant functional techniques. If you are used to array-based programming, the shift away from "low-level" index-based array manipulations in favour of "high-level" list iterators and functions may be challenging at first. The section [Common techniques](#common-techniques) at the end of this tutorial collects some methods for solving common programming tasks in a purely functional manner. +Lists do not allow for random access to their elements by index. The Juvix +standard library intentionally does not provide a function to access the nth +element of a list. Such a function could easily be implemented, but it would not +be efficient. Lists are *not* arrays. If you find yourself wanting to access +list elements by numerical index, you are most likely doing something wrong: +trying to incongruently fit imperative array programming patterns into a +functional language instead of using more elegant functional techniques. If you +are used to array-based programming, the shift away from "low-level" index-based +array manipulations in favour of "high-level" list iterators and functions may +be challenging at first. The section [Common techniques](#common-techniques) at +the end of this tutorial collects some methods for solving common programming +tasks in a purely functional manner. ## Pipes -With the pipe `|>` operator, the last argument of a function can be moved to the front: `z |> f x y` is the same as `f x y z`. This is useful for chaining function applications which perform some processing steps in sequence: `x |> doStep1 |> doStep2 |> doStep3 |> doStep4` is the same as `doStep4 (doStep3 (doStep2 (doStep1 x)))`. Such processing pipelines are favored over loops with complex bodies, as they result in cleaner code, better separation of concerns across steps, and improved modularity. +With the pipe `|>` operator, the last argument of a function can be moved to the +front: `z |> f x y` is the same as `f x y z`. This is useful for chaining +function applications which perform some processing steps in sequence: `x |> +doStep1 |> doStep2 |> doStep3 |> doStep4` is the same as `doStep4 (doStep3 +(doStep2 (doStep1 x)))`. Such processing pipelines are favored over loops with +complex bodies, as they result in cleaner code, better separation of concerns +across steps, and improved modularity. -For example, recall the function `totalCostOfValuableResources` from the previous section. +For example, recall the function `totalCostOfValuableResources` from the +previous section. ```juvix extract-module-statements module pipes0; @@ -518,7 +667,11 @@ totalCostOfValuableResources (lst : List Resource) : Nat := end; ``` -The body of the `for`-expression is somewhat complex, performing three distinct operations: checking if the price of `r` exceeds a threshold, computing the total cost for `r`, and computing the sum. The `for`-expression can be rewritten into a pipeline, resulting in increased readability and cleaner separation between the performed operations. +The body of the `for`-expression is somewhat complex, performing three distinct +operations: checking if the price of `r` exceeds a threshold, computing the +total cost for `r`, and computing the sum. The `for`-expression can be rewritten +into a pipeline, resulting in increased readability and cleaner separation +between the performed operations. ```juvix extract-module-statements module pipes1; @@ -530,9 +683,17 @@ totalCostOfValuableResources (lst : List Resource) : Nat := end; ``` -The iterators `map` and `filter` can be used as functions like above, with the body becoming the first argument and the list becoming the second argument. So `map doIt lst` is the same as `map (x in lst) {doIt x}`, and `filter doIt lst` is the same as `filter (x in lst) {doIt x}`. +The iterators `map` and `filter` can be used as functions like above, with the +body becoming the first argument and the list becoming the second argument. So +`map doIt lst` is the same as `map (x in lst) {doIt x}`, and `filter doIt lst` +is the same as `filter (x in lst) {doIt x}`. -Now suppose we would like to modify `totalCostOfValuableResources` to take into account only resources with quantity greater than 10. In the first version, we would need to modify the `for`-expression body by inserting an extra check, which would complicate it even further. In the second version, we just need to add an extra step to the pipeline. There is no need to modify existing pipeline steps. +Now suppose we would like to modify `totalCostOfValuableResources` to take into +account only resources with quantity greater than 10. In the first version, we +would need to modify the `for`-expression body by inserting an extra check, +which would complicate it even further. In the second version, we just need to +add an extra step to the pipeline. There is no need to modify existing pipeline +steps. ```juvix extract-module-statements module pipes2; @@ -547,15 +708,20 @@ end; ## Sets -Lists represent ordered sequences of elements with possible duplicates. They are intended for sequential processing and do not support efficient membership checks. If you need to check for membership, the order is not significant and duplicates not allowed, then a `Set` is an appropriate data structure. +Lists represent ordered sequences of elements with possible duplicates. They are +intended for sequential processing and do not support efficient membership +checks. If you need to check for membership, the order is not significant and +duplicates not allowed, then a `Set` is an appropriate data structure. -Sets are not in the standard library prelude, so you need to import them separately. +Sets are not in the standard library prelude, so you need to import them +separately. ```juvix import Stdlib.Data.Set as Set open using {Set}; ``` -The above statement makes set functions available qualified with `Set.` and the `Set` type available unqualified. +The above statement makes set functions available qualified with `Set.` and the +`Set` type available unqualified. The following functions are available for sets. @@ -564,7 +730,8 @@ The following functions are available for sets. - `Set.insert elem set` inserts `elem` into `set`, returning the updated set. - `Set.delete elem set` removes `elem` from `set`. -As an example, here is a function which removes duplicates from a list, preserving element order and keeping the first occurrence of each value. +As an example, here is a function which removes duplicates from a list, +preserving element order and keeping the first occurrence of each value. ```juvix removeDuplicates (lst : List Nat) : List Nat := @@ -577,13 +744,21 @@ removeDuplicates (lst : List Nat) : List Nat := |> reverse; ``` -The function uses an auxiliary set `seen` to check if an element was already encountered. The result of the `for` iteration is a pair of `(acc, seen)`, so to get the result list we need to extract the first component `acc` with `fst`. Since the `for` iterator goes through the list from beginning to end, the order of the accumulated result list is reversed. The standard library function `reverse` reverses the result list back to the original order. +The function uses an auxiliary set `seen` to check if an element was already +encountered. The result of the `for` iteration is a pair `(acc, seen)`, so to +get the result list we need to extract the first component `acc` using `fst`. +Since the `for` iterator goes through the list from beginning to end, the order +of the accumulated result list is reversed. The standard library function +`reverse`, as its name indicates, reverses the result list back to the original +order. ## Maps -Maps are data structures which represent associations from keys to values. Each key may be associated with only one value. +A *map* is a data structure that represents associations from keys to values. Each +key can be associated with only one value. + +In Juvix, the `Map` type of maps needs to be imported with the following statement: -In Juvix, the `Map` type needs to be imported with the following statement: ```juvix import Stdlib.Data.Map as Map open using {Map}; ``` @@ -591,13 +766,22 @@ import Stdlib.Data.Map as Map open using {Map}; The following functions are supported for maps. - `Map.empty` is the empty map. -- `Map.lookup key map` evaluates to `just value` if `key` is associated with `value` in `map`, or to `nothing` if `key` has no association in `map`. -- `Map.insert key value map` associates `key` with `value` in `map`, overriding any previous association if present. + +- `Map.lookup key map` evaluates to `just value` if `key` is associated with +`value` in `map`, or to `nothing` if `key` has no association in `map`. + +- `Map.insert key value map` associates `key` with `value` in `map`, overriding +any previous association if present. + - `Map.delete key map` removes the association for `key` from the `map`. + - `Map.keys map` returns the list of keys present in `map`. + - `Map.values map` returns the list of values associated with some key in `map`. -As an example of `Map` usage, consider the following function which groups resources by their price and adds up the quantities to create one resource for each price. The order of elements in the result list is not preserved. +As an example of `Map` usage, consider the following function which groups +resources by their price and adds up the quantities to create one resource for +each price. The order of elements in the result list is not preserved. ```juvix groupResourcesByPrice (lst : List Resource) : List Resource := @@ -611,7 +795,10 @@ groupResourcesByPrice (lst : List Resource) : List Resource := |> Map.values; ``` -In fact, `groupResourcesByPrice` could be written more succinctly using `Map.insertWith`, which does not replace the value when the key is already present, but instead combines the new and the old values using a provided function. +In fact, `groupResourcesByPrice` could be written more succinctly using +`Map.insertWith`, which does not replace the value when the key is already +present, but instead combines the new and the old values using a provided +function. ```juvix extract-module-statements module map1; @@ -629,9 +816,14 @@ end; ## Traits -In Juvix, traits provide a way to define shared behaviour for types, similarly to traits in Rust, type classes in Haskell or interfaces in Java. A trait specifies a set functions that must be implemented in an instance for a given type. Traits allow to write generic, reusable code by specifying constraints on types without committing to a specific implementation. +In Juvix, traits provide a way to define shared behaviour for types, similarly +to traits in Rust, type classes in Haskell, and interfaces in Java. A trait +specifies a set of functions that must be implemented in an instance for a given +type. Traits allow you to write generic, reusable code by specifying constraints on +types without committing to a specific implementation. -For example, the `Eq` trait from the standard library specifies the equality function `Eq.eq`. +For example, the `Eq` trait from the standard library specifies the equality +function `Eq.eq`. ```juvix extract-module-statements module EqTrait; @@ -643,7 +835,8 @@ type Eq A := end; ``` -An instance of Eq for a given type can be defined by implementing `Eq.eq` for this type. Here is an `Eq` instance definition for the `Resource` type. +An instance of `Eq` for a given type can be defined by implementing `Eq.eq` for +this type. Here is an `Eq` instance definition for the `Resource` type. ```juvix instance @@ -655,17 +848,30 @@ eqResourceI : Eq Resource := }; ``` -A polymorphic function that needs an equality operation for its type parameter can be defined generically by requiring an instance of the `Eq` trait, rather than relying on a specific equality implementation. Then the function can be used with any type for which an instance of `Eq` is available. The appropriate instance is chosen at compilation time based on the type. The corresponding concrete equality implementation from the instance is then used. +A polymorphic function that needs an equality operation for its type parameter +can be defined generically by requiring an instance of the `Eq` trait, rather +than relying on a specific equality implementation. Then, the function can be +used with any type for which an instance of `Eq` is available. The appropriate +instance is chosen at compilation time based on the type. The corresponding +concrete equality implementation from the instance is then used. + +As a trivial example, the standard library actually defines the infix equality +operator `==` in terms of the `Eq` trait. -As a trivial example, the standard library actually defines the infix equality operator `==` in terms of the `Eq` trait. ```juvix extract-module-statements module traits1; == {A} {{Eq A}} (x y : A) : Bool := Eq.eq x y; end; ``` -The implicit instance argument `{{Eq A}}` specifies that wherever the function `==` is called, an instance of `Eq` is required for the type parameter `A`. The specific instance is automatically inferred by the type checker, separately for each function call. -Because we have defined an instance of `Eq` for `Resource`, we can now use `==` with resources. The type checker automatically choses `eqResourceI` as the required instance and uses the corresponding equality implementation. +The implicit instance argument `{{Eq A}}` specifies that wherever the function +`==` is called, an instance of `Eq` is required for the type parameter `A`. The +specific instance is automatically inferred by the type checker, separately for +each function call. + +Because we have defined an instance of `Eq` for `Resource`, we can now use `==` +with resources. The type checker automatically choses `eqResourceI` as the +required instance and uses the corresponding equality implementation. ```juvix countResource (resource : Resource) (lst : List Resource) : Nat := @@ -676,7 +882,9 @@ countResource (resource : Resource) (lst : List Resource) : Nat := }; ``` -The above function does not depend on the details of the `Resource` type. It only requires that equality be available for the list elements. Hence, the function can be generalized to use the `Eq` trait. +The above function does not depend on the details of the `Resource` type. It +only requires that equality be available for the list elements. Hence, the +function can be generalized to use the `Eq` trait. ```juvix countValue {A} {{Eq A}} (value : A) (lst : List A) : Nat := @@ -687,7 +895,11 @@ countValue {A} {{Eq A}} (value : A) (lst : List A) : Nat := }; ``` -To use the polymorphic equality operator `==`, a type must have an instance of the `Eq` trait. It is quite tedious to manually implement instances of `Eq` for each user-defined type. Fortunately, an `Eq` instance may be derived automatically if there already exist `Eq` instances for the types of all record fields / constructor arguments. +To use the polymorphic equality operator `==`, a type must have an instance of +the `Eq` trait. It is quite tedious to manually implement instances of `Eq` for +each user-defined type. Fortunately, an `Eq` instance may be derived +automatically if `Eq` instances already exist for the types of all record fields +or constructor arguments. ``` deriving instance @@ -696,9 +908,13 @@ eqResourceI : Eq Resource; ## Debugging -Juvix does not currently have a debugger. A common way of debugging Juvix programs is to make use of the REPL. Once you load your file into the REPL (with `juvix repl file.juvix`, or via Emacs or VSCode), you can evaluate any function from the file with the desired arguments and inspect the result. +Juvix does not currently have a debugger. A common way of debugging Juvix +programs is to make use of the REPL. Once you load your file into the REPL (with +`juvix repl file.juvix`, or via Emacs or VSCode), you can evaluate any function +from the file with the desired arguments and inspect the result. -Another technique is to use the `trace` function which prints its argument and returns it. +Another technique is to use the `trace` function which prints its argument and +returns it. ```juvix import Stdlib.Debug.Trace open; @@ -710,9 +926,14 @@ combineResources (r1 r2 : Resource) : Resource := }; ``` -The function `combineResources` first prints `r1`, then prints `r2`, then returns updated `r1`. The sequencing operator `>->` first evaluates the expression on the left, ignores the result, then evaluates the expression on the right and returns it. The import above the function is necessary, because `trace` is not in the standard library prelude. +The function `combineResources` first prints `r1`, then prints `r2`, then +returns updated `r1`. The sequencing operator `>->` first evaluates the +expression on the left, ignores the result, then evaluates the expression on the +right and returns it. The import above the function is necessary, because +`trace` is not in the standard library prelude. -The `failwith` function may also be useful to immediately crash with an error message. +The `failwith` function may also be useful to immediately crash with an error +message. ```juvix import Stdlib.Debug.Fail open; @@ -728,7 +949,7 @@ giveAllAway (lst : List Resource) : List Resource := | else := trace "Hurray!" >-> lst'; ``` -Finally, `assert` allows to specify assumptions at different points in the program. +Finally, `assert` allows you to specify assumptions at different points in the program. ```juvix dividePrice (n : Nat) (r : Resource) : Resource := @@ -738,26 +959,30 @@ dividePrice (n : Nat) (r : Resource) : Resource := }; ``` - ## Common techniques -This section lists some common programming tasks and explains how to solve them in a purely functional manner. +This section lists some common programming tasks and explains how to solve them +in a purely functional manner. ### Accumulate list elements from left to right - - Solution: use `for`. - - Example: - ```juvix extract-module-statements - module ForExample; - reverse {A} (lst : List A) : List A := - for (acc := []) (x in lst) { - x :: acc - }; - end; - ``` + +- Solution: use `for`. + +- Example: + ```juvix extract-module-statements + module ForExample; + reverse {A} (lst : List A) : List A := + for (acc := []) (x in lst) { + x :: acc + }; + end; + ``` ### Accumulate list elements from right to left - - Solution: use `rfor`. - - Example: + +- Solution: use `rfor`. + +- Example: ```juvix duplicate (lst : List Nat) : List Nat := rfor (acc := []) (x in lst) { @@ -766,14 +991,20 @@ This section lists some common programming tasks and explains how to solve them ``` ### Check if a list is empty - - Solution: use `isEmpty` - - Anti-pattern: do _not_ use `length lst == 0` - The `length` function requires computation time proportional to the length of its argument - it needs to traverse the entire list to compute the length. The `isEmpty` function runs in constant time. +- Solution: use `isEmpty`. + +- Anti-pattern: do _not_ use `length lst == 0`. + +The `length` function requires computation time proportional to the length of +its argument - it needs to traverse the entire list to compute the length. The +`isEmpty` function runs in constant time. ### Check if a condition holds for all list elements - - Solution: use `all`. - - Example: + +- Solution: use `all`. + +- Example: ```juvix allDivisible (n : Nat) (lst : List Nat) : Bool := all (x in lst) { @@ -782,8 +1013,10 @@ This section lists some common programming tasks and explains how to solve them ``` ### Check if a condition holds for any list element - - Solution: use `any`. - - Example: + +- Solution: use `any`. + +- Example: ```juvix anyDivisible (n : Nat) (lst : List Nat) : Bool := any (x in lst) { @@ -792,8 +1025,10 @@ This section lists some common programming tasks and explains how to solve them ``` ### Keep a state when accumulating list elements - - Solution: use an extra accumulator. - - Example: + +- Solution: use an extra accumulator. + +- Example: ```juvix listToMap {A} {{Ord A}} (lst : List A) : Map Nat A := for (acc, i := [], 0) (x in lst) { @@ -804,12 +1039,18 @@ This section lists some common programming tasks and explains how to solve them ``` ### Replicate an element into a list - - Solution: use `replicate`. The call `replicate n a` evaluates to `[a; a; ..; a]` with `a` repeated `n` times. + +- Solution: use `replicate`. The call `replicate n a` evaluates to +`[a; a; ..; a]` with `a` repeated `n` times. ### Concatenate two lists - - Solution: use `++`. - - Warning: `lst1 ++ lst2` takes time proportional to the length of `lst1`. When used inside a loop, care must be taken to avoid excessive running time. - - Example: + +- Solution: use `++`. + +- Warning: `lst1 ++ lst2` takes time proportional to the length of `lst1`. +When used inside a loop, care must be taken to avoid excessive running time. + +- Example: ```juvix extract-module-statements module myFlatten1; flatten {A} (listOfLists : List (List A)) : List A := @@ -818,21 +1059,33 @@ This section lists some common programming tasks and explains how to solve them }; end; ``` - In each step, `++` takes time proportional to the length of `lst`, so the total running time of `flatten` is proportional to the length of the result. + In each step, `++` takes time proportional to the length of `lst`, so the + total running time of `flatten` is proportional to the length of the result. - - Anti-example: +- Anti-example: ```juvix flattenWRONG {A} (listOfLists : List (List A)) : List A := for (acc := []) (lst in listOfLists) { acc ++ lst }; ``` - In each step, `++` takes time proportional to the current length of `acc`, which gets longer with every step. The total running time of `flattenWRONG` is proportional to the square of the length of the result. + In each step, `++` takes time proportional to the current length of `acc`, + which gets longer with every step. The total running time of `flattenWRONG` is + proportional to the square of the length of the result. + ### Add an element at the back of a list - - Solution: don't do it. Lists are designed to allow adding elements efficiently only at the front. You should _never_ accumulate list elements by adding them at the back. Change the direction of your iteration (from `for` to `rfor` or vice versa) or use `reverse`. - - If you _really_ need to do it: use `lst ++ [x]`, but be aware of the inefficiency. The only legitimate use-cases are when the length of `lst` is a known constant, or when used carefully like list concatenation. - - Legitimate use-case examples: + +- Solution: don't do it. Lists are designed to allow adding elements efficiently +only at the front. You should _never_ accumulate list elements by adding them at +the back. Change the direction of your iteration (from `for` to `rfor` or vice +versa) or use `reverse`. + +- If you _really_ need to do it: use `lst ++ [x]`, but be aware of the +inefficiency. The only legitimate use-cases are when the length of `lst` is a +known constant, or when used carefully like list concatenation. + +- Legitimate use-case examples: ```juvix key32Bytes : List Nat := replicate 31 0x0 ++ [0x1]; @@ -842,7 +1095,8 @@ This section lists some common programming tasks and explains how to solve them lst ++ [sep] ++ acc }; ``` - - Anti-example: + +- Anti-example: ```juvix tagsToPairWRONG (tags : List Tag) : Pair (List Nullifier) (List Commitment) := for (nfs, cms := [], []) (tag in tags) { @@ -851,7 +1105,8 @@ This section lists some common programming tasks and explains how to solve them | Created cm := nfs, cms ++ [cm] }; ``` - - Correction: + +- Correction: ```juvix tagsToPair (tags : List Tag) : Pair (List Nullifier) (List Commitment) := rfor (nfs, cms := [], []) (tag in tags) { @@ -862,24 +1117,29 @@ This section lists some common programming tasks and explains how to solve them ``` ### Iterate over a range of numbers - - Solution: use `a to b` or `a to b step k`. - - Examples: + +- Solution: use `a to b` or `a to b step k`. + +- Examples: ```juvix + -- list of numbers up to `n` listUpTo (n : Nat) : List Nat := rfor (acc := []) (x in 1 to n) { x :: acc }; + -- divisors of a number divisors (n : Nat) : List Nat := if - | n == 0 := [] - | else := - rfor (acc := []) (x in 1 to n) { - if - | mod n x == 0 := x :: acc - | else := acc - }; + | n == 0 := [] + | else := + rfor (acc := []) (x in 1 to n) { + if + | mod n x == 0 := x :: acc + | else := acc + }; + -- sum of even numbers up to `n` sumEvenUpTo (n : Nat) : Nat := for (acc := 0) (x in 2 to n step 2) { acc + x @@ -887,11 +1147,14 @@ This section lists some common programming tasks and explains how to solve them ``` ### Repeat n times - - Solution: use `iterate`. - - Example: + +- Solution: use `iterate`. + +- Example: ```juvix random (seed : Nat) : Nat := mod (1103515245 * seed + 12345) 2147483648; + randoms (n initialSeed : Nat) : Pair Nat (List Nat) := let update (acc : Pair Nat (List Nat)) : Pair Nat (List Nat) :=