Skip to content

Latest commit

 

History

History
109 lines (75 loc) · 5.18 KB

README.md

File metadata and controls

109 lines (75 loc) · 5.18 KB

Primal

Primal

Overview

Primal is a minimalist, lightweight, functional programming language that is designed to be simple and easy to learn. It is inspired by Lambda Calculus, a mathematical formalism that is used to define functions and perform computations.

It is a declarative language, which means that it focuses on what should be done rather than how it should be done. This makes it easier to read and understand, especially for beginners.

It is free and open-source under the MIT license, meaning that you can use it for any purpose, commercial or non-commercial, without any restrictions. You can also modify the source code and distribute it as you see fit.

It is still under development, so there may be bugs and missing features. If you encounter any issues, please report them on GitHub. If you have any suggestions or ideas for improvement, please let us know by sending us feedback using the web form.

Goal

The language is designed primarily for educational purposes. It aims to teach programming concepts without the complexity of more advanced languages. It emphasizes simplicity and minimalist syntax, making it a good choice for beginners who are learning the fundamentals of programming. It is not intended to be used in production environments, but it can be used for small scripts.

Technical envelope

The language has the following characteristics:

  • Lazy evaluated
  • Dynamically typed
  • Single threaded
  • Expression oriented
  • Interpreted
  • Interactive
  • Garbage collected

Philosophy

Everything in Primal is a function, which can be composed to create more complex functions. Functions cannot be overloaded (i.e. only one function with a given name can exist in a program). The full list of functions can be found in the reference page.

Being a functional language, Primal does not have loop constructions. Instead, it uses recursion to perform repetitive tasks, allowing for a more declarative style of programming.

Syntax

The syntax is designed to be simple and easy to read and consists of function declarations that are evaluated to produce a result.

A function declaration is composed of:

  • Name: which must match the regular expression [a-zA-Z][\w\.]*
  • Parameters (optional): a list of identifiers, each matching the regular expression [a-zA-Z][\w\.]
  • Body: an expression which consist of combinations of:
    • Constants: "Hello", 42, true, [1, 2, 3], etc
    • Operations: foo + 3, !foo, foo >= bar, etc
    • Function calls: foo(x), bar(10), etc
    • Conditionals: if (foo) bar else baz

Here is an example of a function with parameters:

cube(n) = pow(n, 3)

And here is an example of a function without parameters:

pi = 3.14159

Main function

There is a special function in the language called main, which serves as the entry point of a program when it is run. The main function is optional and can be declared with any number of parameters, including none. If present, it can call other functions and perform computations to produce a result.

If the main function is not present, the program will still run, but no computations will be performed. In this case, the program will run in interactive mode, where the user can enter expressions and see the results of their evaluation.

Typing system

Primal has a dynamic runtime type checking. This means that users cannot explicitly declare types for parameters or return values in the code. Instead, the system performs type checks during execution to ensure that values match the expected types when passed to functions.

This approach offers flexibility and ease of use but relies on runtime checks to catch type-related issues, which can lead to errors that are only caught during execution.

These are the available types at runtime:

  • Boolean
  • Number
  • String
  • Timestamp
  • List
  • Map
  • Set
  • Stack
  • Queue
  • Vector
  • File
  • Directory
  • Function

Runtime

Primal is an interpreted language. The compiler translates the source code into an intermediate representation that is then executed by the interpreter. The interpreter is responsible for executing the intermediate representation and producing the output of the program.

This allows for fast development and testing cycles since changes to the source code can be quickly compiled and executed without the need for a separate build step.