https://github.com/flofriday/moose
π A new fun programming language
https://github.com/flofriday/moose
programming-language
Last synced: 6 months ago
JSON representation
π A new fun programming language
- Host: GitHub
- URL: https://github.com/flofriday/moose
- Owner: flofriday
- License: mit
- Created: 2022-05-17T12:02:04.000Z (over 3 years ago)
- Default Branch: main
- Last Pushed: 2022-10-12T15:06:26.000Z (about 3 years ago)
- Last Synced: 2025-03-30T22:51:13.559Z (7 months ago)
- Topics: programming-language
- Language: Swift
- Homepage:
- Size: 626 KB
- Stars: 5
- Watchers: 2
- Forks: 1
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# π Moose
**Moose** is a new programming language. This repo also includes an interpreter
written in [Swift](https://www.swift.org/).## Try it out
**Note:** We only support macOS, iOS and Linux. On Windows you can use WSL.
You need to [install swift](https://www.swift.org/download/), after which you can run:
```bash
swift run Moose
```Or if you want to have an optimized build you can build and run with:
```bash
swift build -c release
./.build/release/Moose
```The interpreter also accepts a single file:
```bash
./.build/release/Moose Examples/ticktacktoe.moose
```## Status
At the moment we have already implemented all features we wanted. However,
please consider that this was mainly a learning experience. We might continue
working on it and playing with things that are interesting to us.## Getting Started
The rest of this file is and introduction to the language which tries to
introduce the core ideas.Furthermore, there are some examples in the
[Examples](https://github.com/flofriday/Moose/tree/main/Examples) folder and
there is a full documentation of the quite small standard library in
[Stdlib.md](https://github.com/flofriday/Moose/blob/main/Stdlib.md)## Types
There are only 4 built in types in Moose: `Int, Float, String, Bool`
## Variables
Moose provides two types of variables: immutable and mutable variables.
### Mutable Variables
Mutable variables are variables that can be reassigned over and over again. To declare a mutable variable, the keyword `mut` must be used.
```dart
mut a = 2
a = 4
a +: 1mut b: Int
b = 2
```### Immutable Variables
While mutable variables need to be declared using the `mut` keyword, immutable variables donβt have such a keyword. They also have to be assigned directly on the declaration, since the declaration without explicit initialisation is initialised with `nil`.
```dart
a = 3;
a = 2; // Error, cannot reassign immutable variable
``````dart
b: Int;
b; // nil
b = 3; // Error, cannot reassign immutable variable
``````dart
b: Int = 3;
b; // 3
```## Nil
Every object can be `nil`, and since even basic types are objects they can be `nil` too.
```dart
cookie: Int = nil; // Basic types can be nil
strudel: Int; // Implizitly without assigning a value a variable is automatically nil.
```To make working with null more ergonomic, we introduced the double questionmark operator `??`
```dart
breakfast: String = nil;
println(breakfast ?? "Melage and semmel");// This code is just syntactic sugar and is equivalent to
println(breakfast != nil ? breakfast : "Melage and semmel");
```## Lists
Lists are arrays that can grow and shrink dynamically. Moreover there is a special syntax to create them, with square brackets. Lists also only contain objects of a single type.
```dart
wishlist: [String] = ["Computer", "Bicycle", "Teddybear"];println(wishlist[0]); // "Computer"
println(wishlist[-1]); // "Teddybear"wishlist.append("car");
wishlist.append(["Aircraftcarrier", "Worlddomination"]);
```## Dicts
Dictionaries (aka Hashtables or Hashmaps) are also build in.
```dart
ages: {String:Int} = {
"Flo": 23,
"Paul": 24,
"Clemens": 7,
}println(ages["Paul"]) // 24
println(ages.contains("Paul")) // true
ages.remove("Paul")
println(ages.contains("Paul")) // false
```## If Else
Like go and rust we donβt require Parenteses around the condition but do require braces around the body.
```dart
age = 23
if age > 18 {
println("Please enter")
} else if age == 18 {
println("Finally you can enter")
} else {
println("I am sorry come back in a year")
}// Sometimes you need to discriminate against Pauls
name = "Paul"
if (age > 12) && (name != "Paul") {
println("Welcome to the waterpark")
}```
## For
`for` is the only loop in the language and is used as a foreach, C-Style for loop or like a while loop.
```dart
// For-each style
for i in range(10) {
println("I was here")
}// C-style loop
for i = 0; i < 100; i +: 3 {
println(i)
}// while style loop
for true {
println("I will never terminate")
}
```## Comments
Comments are written with `//`, which comments out the rest of the line.
```dart
a = 3; // this is a comment
```## Functions and Operators
### Classic Functions
```dart
func myFunction(age: Int, name: String) > String {
return "Hi I am " + name + age.toString()
}println(myFunction(42, "Alan Turing"))
```### Operator Functions
Beside classic functions, Moose provides the possibility to write own operators for own types. In fact, all operators have a function implementation. There are 3 types of operator functions: `prefix`, `infix` and `postfix`
```dart
// What if JS was right with adding numbers to strings?
infix + (a: String, b: Int) > String {
return a + b.toString()
}// A prefix + makes numbers positive
prefix + (a: Int) > Int {
if a < 0 {
return a * -1
}
return a
}a = "4"
b = 12c = a + b
println(c) // 412mut d = -2
println(+d) // 2
```### Print Objects
Each object has a default implementation of `represent()` that returns a string with all the fields and their values of the object. By assigning the object to a `String` variable or a parameter, the `represent()` function is called internally.
```dart
p = Person("Alan Turing", 41)println(p)
// equivalent to
println(p.represent())
// Person: {name: "Alan Turing", age: 41}
```## Classes and Objects
```dart
class Person {
name: String
age: Intfunc hello() {
// Instead of this, Moose uses me
println("Hey I am " + me.name)
}
}// Classes have a default constructor in which all all fields have to be filled out
anna = Person("Anna", 74)
anna.hello()
```## Inheritance
Moose supports single class inheritance.
```dart
class Employee < Person {
work: Stringfunc hello() > String {
print("Hi I am " + me.name + " and I work at " + me.work)
}
}catrin = Employee("Google", "Catrin", 56 )
catrin.hello()
```## Extending Classes
You extend existing classes by using the `extend` construct.
```dart
extend Person {
func wasBorn() > Int {
return 2022 - me.age
}
}j = Person("Johannes", 23)
println(j.wasBorn())
```## Tuple
In Moose, tuples are a standard data type, like Int, String and others. It also allows programmers to return multiple values from a function and offers the possibility of unpacking.
```dart
person = ("luis", 17, false)// Without unpacking
name1 = person.0
age1 = person.1
married1 = person.2// With unpacking
(name2, age2, married2) = personprintln(name1 == name2)
```### Unpacking Objects
Unpacking is not only possible for tuples themselves, but also for objects, where the tuple contains the first n object fields.
```dart
p = Person("Alan Turing", 41)
(name, age) = p
```## Indexing
Accessing data of an object is not only possible for `List` s, but for all datatypes that implement the `getItem(Int)` method. Similarly the `setItem(Int, Any)` allows to write at an index.
```dart
class Person {
name: Stringfunc getItem(i: Int) > (String) {
return name[i]
}
}p = Person("Alan Turing")
println(p[0]) // "A"
```## Error handling and Panics
Moose has two types of error handling. Unlike exceptions panics are unrecoverable
errors, which will cause the interpreter stop execution and exit with a
non-zero exit code.```dart
a: Int = nil
b = a + 3 // NIlUsagePanicl = [1, 2, 3]
x = l[9000] // OutOfBoundsPanic
```For recoverable errors the convention is to return a tuple, where the last
element is a string with an error message. If the error is nil everything
succeeded, whereas if an error occurred it will be a message describing the
reason.```dart
(mail, err) = open("mail.txt")
if err != nil {
println("Oh no I cannot read the mail because: " + err)
exit(1)
}println(mail)
```