Skip to content

Latest commit

 

History

History
387 lines (289 loc) · 8.42 KB

File metadata and controls

387 lines (289 loc) · 8.42 KB
layout default

Interpreter

You can try out the language by opening the interpreter and making queries to the language.

$ cosmos -i
> x=1
| x = 1
> x=1 or 2=x
| x = 1
| x = 2

Writing a file

Make a file hello.co with the content,

print('hello world')

The file can be loaded with the -l flag.

$ cosmos -l hello
'hello world'`

Relations

Instead of functions, Cosmos has relations.

They're made using the rel keyword.

//note that the there is no 'return' in the definition
//instead, the parameter y is explicit
//this is typically the 'output' parameter
rel double(x, y)
    y = x*2

double(4,x) //x is 8

Whereas functions have one output, relations may have zero, one or more outputs. You can check this by making queries at the interpreter.

cosmos -i
> x=1 or x=2 //this query has two answers (outputs)
| x = 1
| x = 2

If the system picks one answer and it turns out to be invalid, the system will backtrack and pick the other.

rel p(x)
    x=1 or x=2
    
rel main()
	p(x)
	x!=1
	io.writeln(x) //2

Relations may adopt function syntax. This lets relations be nested.

double(4,x) //logic syntax
x=double(4) //function syntax
print(double(3)) //this will print 6

Logic-wise, double(4,x) is read as a statement: "the double of 4 is x".

double(4) reads as "the double of 4".

First-class relation

Relations are first-class values. It's possible to define a relation within another.

rel p(x)
    rel temp(x)
        x = 2
    temp(x)
    
p(x) //x is 2

Alternatively:

rel p(x)
    temp = rel(x)
        x = 2
    temp(x)

Functors

Functors are composite data.

functor(F, Functor) //declares an object for creating functors
x = F(1, 2) //x is assigned to a functor F composed by the values 1 and 2
x = F(1, a) //uses pattern matching to match F(1, 2) against F(1, a)
print(a) //2

The special relation functor is used to declare F as an object for making functors.

Lists are syntax sugar for the functor Cons. Here are two ways to define a list:

l = [1, 2]
l = Cons(1, Cons(2, Cons))

Relations such as first, map and filter can be used to manipulate lists.

l = [1,2,3]
list.first(l, head) //head is 1
list.rest(l, tail) //tail is [2, 3]
list.map(l, math.inc, l2) //l2 is [2, 3, 4]
list.map(l3, math.inc, l) //l3 is [0, 1, 2]
list.filter(l, rel(x) x!=3;, l4) //l4 is [1, 2]

Immutability

Variables are immutable. Instead of modifying a value we create a new one.

l2 = list.push(l, 55) //instead of modifying l, we create a new variable l2
io.writeln(l)  //[1, 2, 3]
io.writeln(l2) //[1, 2, 3, 55]

Cosmos adopts many principles and features that are common in functional programming languages (although the principles apply to relations rather than functions).

Types

Cosmos manages a balance between strictness and non-strictness. Writing the type of a variable is (almost always) optional.

    Integer n = 7
    Real x = 5.2
    String s = 'abc'
    z = 5 //z is implied to be an Integer
    Functor l = [1, 2, 3]
    functor(F, Functor)
    Functor f = F('apple', 5)

The type system supports composite types.

    Functor String Number f2 = F('apple', 2)

Functor String Number is a composite type that accepts any functor whose first element is a string and second is a number.

    Relation Any Any p = double

Relations get composite types. Relation Any Any is a type that accepts any relation with exactly two arguments.

Tables

Tables (also known as maps, dictionaries, etc.) are structures that map keys to values.

Table t = {x=1 and y=2}
table.set(t, 'a', 1, t2)

print(t) //{'x': 1, 'y': 2}
print(t2) //{'x': 1, 'y': 2, 'a': 1}

Making a module

Cosmos files are typically given the .co extension.

//x.co
rel p(x)
    x=2
	
t = {
	'p' = p
}
export(t)

As we have mentioned, Cosmos is in part inspired by imperative scripting languages--specifically, prototypal ones.

The principle is thus the same. The relation p is inserted into a table, which is then exported.

This could've been written as,

//x.co
t = {
	rel p(x)
		x=2
}
export(t)

We then use the special relation require,

require('x', x)
x.p(2)
print(x) //2

Whitespace

The language is whitespace sensitive.

rel p(x)
    x!=1
    x<5

This could be a single line.

rel p(x) x!=1 and x<5;

It's possible to drop the whitespace semantics by writing the unnecessary characters, although this is not generally advisable.

Note that statements are separated by ands (semicolons are only used to end the indendation).

Booleans

There is no boolean type. Instead, relations themselves are "booleans".

Using cases

As you may have noted, and/or is a huge part of the language. Hence, there are many operators that are shorthand for and/or. One of them is case (alias: cond).

case
	s = 'a'
case
	s = 'b'
	x = 1
case
	x = 2

This is sugar for,

(s = 'a') or (s = 'b' and x = 1) or x = 2

Operators

A similar operator is when. when is simply a more imperative-looking form of case.

Take a simple case-statement,

case
	s = 'a'
	x = 1
case
	s = 'b'

This could be written as,

when(s = 'a')
	x = 1
else
	x = 'b'

As you can see, this reads somewhat like an if-statement. x=1 will be read when s = 'a' is true, and otherwise x = 'b' will be the case.

Note that the condition is kind of redundant. After all, the case-statement does not need a condition. Same for an else-clause. As this is simply a more procedural-looking version way of writing 'case', it's not quite a proper conditional.

That's why the actual if-statement and favoured conditional of the language is as follows.

Conditional

if(s = 'a')
	x = 0
else
	x = 2

This is equivalent to,

(s = 'a' and x = 0) or (not s = 'a' and x = 2)

Or,

(s = 'a' and x = 0) or (s != 'a' and x = 2)

Note that the condition is negated. This is therefore the favoured conditional.

Negation is complicated

When making a conditional, we recommend keeping it to simple conditions like x=1 or 5>x.

Let's see an example of a more complex condition.

if(p(x))
	x=1
else
	x=2

The condition this time is p(x). How do we negate an arbitrary relation like p(x)? All the while keeping the code logically pure?

This can be done, however, it may be somewhat wasteful or experimental. The relation may be called twice, or delayed, so as to ensure the code is sound. This may not be what you want.

A few ways out of this are,

  • Replace if with the similar operator when. This can be done if you do not need to negate the condition (or you want to do so manually).
  • Simply keep it as-is. if is guaranteed to be a logically sound conditional.
  • Switch to a function.
fun main(x)
	if(p(x))
		x=1
	else
		x=2
main(2)

When encased in a function, if behaves as an imperative conditional. It will not do any backtracking or non-determinism.

This feature is currently not checked. It'll work correctly as long as you write a proper function, but this in itself is not checked. We may switch to giving more proper checks in the future.

Impure operators

Our use of functions is meant to be an improvement over Prolog's negation-by-failure. Instead of simply writing impure code, one may encase it in functions-- a much more readable abstraction. This is a smoother way to fall back from logic programming.

We also provide an once keyword.

//this will only select the first answer given by p(x)
once p(x)

Pseudo-imperative Programming

As Cosmos was made to explore declarative programming, where data is immutable, it's not truly imperative. However, procedural-style programming is possible using the pseudo-imperative operator, !. (Like functions, this feature is in-development.)

!x=x+1

This is akin to writing,

x2=x1+1

The operator indicates that the alias x is changed to a different value in the current relation.

This allows us to make for and while statements!

for(x=1;x<=3;!x=x+1;)
	print(x) //1, 2, 3