Reference Manual

Here you will find everything you need to know about the core syntax and semantics of Lento in a quick and easy to read format, from the basics to the advanced features.

Table of Contents

1. Introduction

Lento is a general-purpose, strong, and statically-typed programming language designed for simplicity, performance, and safety. The language is heavily inspired by Rust, F#, Haskell, Swift, and Python, and aims to combine the best features of each language into a single coherent and expressive programming model based on modern functional paradigm principles and compilation techniques.

While Lento is a new language, it draws on decades of experience in language design and implementation, and incorporates many modern language features and best practices.

It is designed to be easy to learn and use, while still providing powerful features for building complex, high-performance applications in a wide range of domains. Lento is intended to be a general-purpose language, suitable for a wide range of applications, from system programming to web development, and more.

The Lento language is designed around the following core principles:

2. Lexical Structure

Lento source code is written using the Unicode character set, and is case-sensitive. The following lexical elements are recognized by the Lento compiler:

3. Types

Lento is a statically-typed language, which means that the type of every expression is known at compile time. The following types are supported by the Lento language:

3.1. Primitive Types

Coming soon…

3.2. Composite Types

Coming soon…

3.2.1. Sum Types

Coming soon…

3.2.2. Product Types

Coming soon…

3.3. Function Types

Coming soon…

3.4. Type Aliases

Coming soon…

3.5. Type Variables

Coming soon…

3.6. Annotations

Coming soon…

3.7. Conversion

Coming soon…

3.8. Coercion

Coming soon…

3.9. Compatibility

Coming soon…

3.10. Equivalence

Coming soon…

3.11. Invariance

Coming soon…

3.12. Variance

Coming soon…

3.13. Bounds

Coming soon…

3.14. Constraints

Coming soon…

3.15. Erasure

Coming soon…

3.16. Reflection

Coming soon…

3.17. About Type Inference

Coming soon…

3.18. About Type Checking

Coming soon…

4. Expressions

Expressions are the building blocks of Lento programs, representing values, operations, and computations. The following sections describe the various kinds of expressions that can be used in Lento programs.

4.1. Literals

Literals are fixed values that are written directly in the source code. A literal can represent elements such as, numbers, strings, characters, booleans, etc. The following table lists the different kinds of literals supported by the Lento language:

Literal TypeExampleDescription
Integer42A whole number without a fractional part.
Float3.14A number with a fractional part.
String"foo"A sequence of characters enclosed in double quotes.
Char'a'A single Unicode character enclosed in single quotes.
BooleantrueA value representing true or false.
Unit()A value representing the absence of a value.
List[1, 2, 3]A sequence of values enclosed in square brackets.
Tuple(1, 2, 3)A fixed-size collection of values enclosed in parentheses.
Dictionary{ foo: 42, bar: 3.14 }A collection of key-value pairs enclosed in curly braces.
Set{ 1, 2, 3 }A collection of unique values enclosed in curly braces.

4.2. Operators

Operators are symbols that represent computations on values. Lento supports a wide range of operators for arithmetic, comparison, logical, and other operations. Operators can be used in expressions to combine, transform, and manipulate values.

Lento also allow users to define custom operators using operator functions to extend the language with new operator symbols and keywords that can be used in expressions.

The following sections describe the various kinds of operators supported by the Lento language.

4.2.1. Built-in Operators

Lento provides a set of built-in operators for performing common operations on values. The following table lists the different kinds of built-in operators supported by the Lento language sorted by precedence:

OperatorDescriptionTypesExample
.Member accessStructs, Tuplespoint.x
[]IndexingLists, Tuples, Dictionarieslist[0]
()Function callFunctionsf(x)
^ExponentiationInteger, Float2 ^ 3
!Logical NOTBoolean!true
-NegationInteger, Float-42
*MultiplicationInteger, Float2 * 3
/DivisionInteger, Float6 / 2
% or modModulusInteger7 % 3, 7 mod 3
+AdditionInteger, Float, String2 + 3, "foo" + "bar"
-SubtractionInteger, Float5 - 3
isType checkAnyx is Int, y is String("foo" + bar + "baz")
andLogical ANDBooleantrue and false
orLogical ORBooleantrue or false
,Tuple and Product TypeAny(1, 2, "Hi"), Int, Float
|Sum TypeSum TypesInt | Float
==EqualityInteger, Float, String, Boolean2 == 3
!=InequalityInteger, Float, String, Boolean2 != 3
<Less thanInteger, Float, String2 < 3
<=Less than or equalInteger, Float, String2 <= 3
>Greater thanInteger, Float, String2 > 3
>=Greater than or equalInteger, Float, String2 >= 3

4.2.2. Custom Operators

Lento allows users to define custom operators using operator functions. An operator function is linked to a regular function and a special operator symbol defined in a special compile-time step. Operator functions can support both unary and binary operators, and the operator can be used in expressions just like built-in operators.

It is strongly recommended to use custom operators sparingly and only when they provide a clear and intuitive syntax for a specific operation.

The following example demonstrates how to define a custom operator function in Lento:

// Define a function
increment_by_one(x: Int) -> Int = x + 1

// Register the custom operator in the compiler
operator {
    symbol: "++",
    position: "prefix",
    precedence: 800,
    associativity: "left",
    function: increment_by_one,
    signature: Int -> Int
}

// Use the custom operator in an expression
result = 42++  // result = 43

Notice that the signature should match a variant of increment_by_one function. The position can be either prefix, infix or ternary (coming soon), and the associativity can be either left or right. The precedence should be a number between 0 and 1000, where 0 is the lowest precedence and 1000 is the highest precedence.

4.2.3. Operator Precedence

Operators in Lento have a precedence level that determines the order in which they are evaluated in an expression. Operators with higher precedence are evaluated before operators with lower precedence. The following table lists the different precedence levels of operators in Lento, from highest to lowest:

PrecedenceClassExample Operators
1100Postfix. [] ()
1000Prefix! -
900Exponential^
800Multiplicative* / %
700Additive+ -
610Tuple and Product Type,
600Sum Type|
500Relational== != < <= > >=
400Logical ANDand
300Logical ORor
200Conditionalif else match cond
100Assignment=

4.2.4. Operator Associativity

Operators in Lento have an associativity that determines the order in which they are evaluated when they have the same precedence level in an expression. Operators can be left-associative, right-associative, or (special to Lento) multi-associative. The following table lists the different associativity types of operators in Lento:

AssociativityDescriptionExample Operators
LeftOperators are evaluated from left to right+ - * / and
RightOperators are evaluated from right to left^ or
MultiOperators are evaluated from left to right, but can be chained in sequence, |

4.2.5. Operator Position

Operators in Lento can be prefix, infix, postfix, or ternary. The position of an operator determines how it is used in an expression:

PositionDescriptionExample OperatorsExample Expressions
Prefixbefore its 1 operand! - ++!true -42 ++x
Infixbetween its 2 operands+ - * / ==2 + 3 5 - 3 6 * 2 7 / 3 2 == 3
Postfixafter its 1 operand++ --x++ x--
Ternarybetween its 3 operandsif-elsex if y else z

4.2.6. Operator Overloading

Lento allows users to define multiple versions of an operator function with different argument types. This feature is known as operator overloading and allows users to define custom behavior for operators based on the types of their operands.

It is strongly recommended to use operator overloading sparingly and only when it provides a clear and intuitive syntax for a specific operation.

The following example demonstrates how to define multiple versions of an operator function in Lento:

// Define a new type
type Car = {
    brand: String,
    model: String,
    year: Int
    speed: Float
}

// Define a function
add_cars(car1: Car, car2: Car) -> Car =
    Car(
        brand: car1.brand,
        model: car2.model,
        year: car1.year,
        speed: car2.speed
    )

// Register the custom operator
operator {
    symbol: "+",
    position: "infix",
    precedence: 700,
    associativity: "left",
    function: add_cars,
    signature: Car, Car -> Car
}

// Use the custom operator in an expression
car1 = Car("Toyota", "Corolla", 2020, 120.0)
car2 = Car("Honda", "Civic", 2021, 130.0)
result = car1 + car2
// result = Car("Toyota", "Civic", 2020, 130.0)

4.2.7. Operator Symbols and Keywords

When defining custom operators, it is important to choose operator symbols and keywords that are clear, concise, and consistent with the existing syntax of the language. Lento defines operators as either symbols or keywords.

When defining custom operators, it is recommended to use symbols for arithmetic, bitwise, and comparison operations, and keywords for logical and relational operations.

Sometimes operator symbols may include characters used in other operators, such as + and ++, or = and ==. In such cases, it is recommended to in these ambiguous cases to either, use the operator function directly, such as increment instead of ++, or equals instead of ==, or use whitespace to separate the operator symbols from other tokens in the expression to avoid confusion, such as x + ++y instead of x+++y, which can be ambiguous in prefix and postfix operators ++ and infix +.

So if one wants to define a custom operator for incrementing a value, it is recommended to use a different symbol, such as ++ and +=, or use a keyword, such as increment and increment_by.

4.3. Variables

Coming soon…

4.4. Functions

Coming soon…

4.5. Control Flow

Coming soon…

4.5.1. Conditional Statements

Coming soon…

4.5.2. Pattern Matching

Coming soon…

4.5.3. Loops

Coming soon…

4.6. Structs

Coming soon…

4.7. Classes

Coming soon…

4.8. Lists

Coming soon…

4.9. Tuples

Coming soon…

4.10. Dictionaries

Coming soon…

4.11. Sets

Coming soon…

4.12. Generators

Coming soon…

4.13. Comprehensions

Coming soon…

4.14. Slices

Coming soon…

4.15. Evaluation Order

Coming soon…

5. Error Handling

Coming soon…

6. Modules

Coming soon…

7. Standard Library

Coming soon…

8. Appendix

Coming soon…


REST OF THE DOCUMENTATION IS COMING SOON…