# Tomo is a statically typed, garbage collected imperative language with # emphasis on simplicity, safety, and speed. Tomo code cross compiles to C, # which is compiled to a binary using your C compiler of choice. # To begin with, let's define a main function: func main(): # This function's code will run if you run this file. # Print to the console say("Hello world!") # You can also use !! as a shorthand: !! This is the same as using say(), but a bit easier to type # Declare a variable with ':=' (the type is inferred to be integer) my_variable := 123 # Assign a new value my_variable = 99 # Floating point numbers are similar, but require a decimal point: my_num := 2.0 # Strings can use interpolation with the dollar sign $: say("My variable is $my_variable and this is a sum: $(1 + 2)") say(" Multiline strings begin with a " at the end of a line and continue in an indented region below. You can have leading spaces after the first line and they'll be preserved. The multiline string won't include a leading or trailing newline. ") # Docstring tests use ">>" and when the program runs, they will print # their source code to the console on stderr. >> 1 + 2 # If there is a following line with "=", you can perform a check that # the output matches what was expected. >> 2 + 3 = 5 # If there is a mismatch, the program will halt and print a useful # error message. # Booleans use "yes" and "no" instead of "true" and "false" my_bool := yes # Conditionals: if my_bool: say("It worked!") else if my_variable == 99: say("else if") else: say("else") # Arrays: my_numbers := [10, 20, 30] # Empty arrays require specifying the type: empty_array := [:Int] >> empty_array.length = 0 # Arrays are 1-indexed, so the first element is at index 1: >> my_numbers[1] = 10 # Negative indices can be used to get items from the back of the array: >> my_numbers[-1] = 30 # If an invalid index outside the array's bounds is used (e.g. # my_numbers[999]), an error message will be printed and the program will # exit. # Iteration: for num in my_numbers: >> num # Optionally, you can use an iteration index as well: for index, num in my_numbers: pass # Pass means "do nothing" # Arrays can be created with array comprehensions, which are loops: >> [x*10 for x in my_numbers] = [100, 200, 300] >> [x*10 for x in my_numbers if x != 20] = [100, 300] # Loop control flow uses "skip" and "stop" for x in my_numbers: for y in my_numbers: if x == y: skip # For readability, you can also use postfix conditionals: skip if x == y if x + y == 60: # Skip or stop can specify a loop variable if you want to # affect an enclosing loop: stop x # Tables are efficient hash maps table := {"one"=1, "two"=2} >> table["two"] = 2? # The value returned is optional because none will be returned if the key # is not in the table: >> table["xxx"] = none : Int # Optional values can be converted to regular values using `!` (which will # create a runtime error if the value is null): >> table["two"]! = 2 # You can also use `or` to provide a fallback value to replace none: >> table["xxx"] or 0 = 0 # Empty tables require specifying the key and value types: empty_table := {:Text=Int} # Tables can be iterated over either by key or key,value: for key in table: pass for key, value in table: pass # Tables also have ".keys" and ".values" fields to explicitly access the # array of keys or values in the table. >> table.keys = ["one", "two"] >> table.values = [1, 2] # Tables can have a fallback table that's used as a fallback when the key # isn't found in the table itself: table2 := {"three"=3; fallback=table} >> table2["two"]! = 2 >> table2["three"]! = 3 # Tables can also be created with comprehension loops: >> {x=10*x for x in 5} = {1=10, 2=20, 3=30, 4=40, 5=50} # If no default is provided and a missing key is looked up, the program # will print an error message and halt. # Any types can be used in tables, for example, a table mapping arrays to # strings: table3 := {[10, 20]="one", [30, 40, 50]="two"} >> table3[[10, 20]]! = "one" # Sets are similar to tables, but they represent an unordered collection of # unique values: set := {10, 20, 30} >> set:has(20) = yes >> set:has(999) = no # You can do some operations on sets: other_set := {30, 40, 50} >> set:with(other_set) = {10, 20, 30, 40, 50} >> set:without(other_set) = {10, 20} >> set:overlap(other_set) = {30} # So far, the datastructures that have been discussed are all *immutable*, # meaning you can't add, remove, or change their contents. If you want to # have mutable data, you need to allocate an area of memory which can hold # different values using the "@" operator (think: "(a)llocate"). my_arr := @[10, 20, 30] my_arr[1] = 999 >> my_arr = @[999, 20, 30] # To call a method, you must use ":" and the name of the method: my_arr:sort() >> my_arr = @[20, 30, 999] # To access the immutable value that resides inside the memory area, you # can use the "[]" operator: >> my_arr[] = [20, 30, 999] # You can think of this like taking a photograph of what's at that memory # location. Later, a new value might end up there, but the photograph will # remain unchanged. snapshot := my_arr[] my_arr:insert(1000) >> my_arr = @[20, 30, 999, 1000] >> snapshot = [20, 30, 999] # Internally, this is implemented using copy-on-write, so it's quite # efficient. # These demos are defined below: demo_keyword_args() demo_structs() demo_enums() demo_lambdas() # Functions must be declared at the top level of a file and must specify the # types of all of their arguments and return value (if any): func add(x:Int, y:Int -> Int): return x + y # Default values for arguments can be provided in place of a type (the type is # inferred from the default value): func show_both(first:Int, second=0 -> Text): return "first=$first second=$second" func demo_keyword_args(): >> show_both(1, 2) = "first=1 second=2" # If unspecified, the default argument is used: >> show_both(1) = "first=1 second=0" # Arguments can be specified by name, in any order: >> show_both(second=20, 10) = "first=10 second=20" # Here are some different type signatures: func takes_many_types( boolean:Bool, integer:Int, floating_point_number:Num, text_aka_string:Text, array_of_ints:[Int], table_of_text_to_bools:{Text=Bool}, pointer_to_mutable_array_of_ints:@[Int], optional_int:Int?, function_from_int_to_text:func(x:Int -> Text), ): pass # Now let's define our own datastructure, a humble struct: struct Person(name:Text, age:Int): # We can define constants here if we want to: max_age := 122 # Methods are defined here as well: func say_age(self:Person): say("My age is $self.age") # If you want to mutate a value, you must have a mutable pointer: func increase_age(self:@Person, amount=1): self.age += amount # Methods don't have to take a Person as their first argument: func get_cool_name(->Text): return "Blade" func demo_structs(): # Creating a struct: alice := Person("Alice", 30) >> alice = Person(name="Alice", age=30) # Accessing fields: >> alice.age = 30 # Calling methods: alice:say_age() # You can call static methods by using the class name and ".": >> Person.get_cool_name() = "Blade" # Comparisons, conversion to text, and hashing are all handled # automatically when you create a struct: bob := Person("Bob", 30) >> alice == bob = no >> "$alice" == 'Person(name="Alice", age=30)' = yes table := {alice="first", bob="second"} >> table[alice]! = "first" # Now let's look at another feature: enums. Tomo enums are tagged unions, also # known as "sum types". You enumerate all the different types of values # something could have, and it's stored internally as a small integer that # indicates which type it is, and any data you want to associate with it. enum Shape( Point, Circle(radius:Num), Rectangle(width:Num, height:Num), ): # Just like with structs, you define methods and constants inside a level # of indentation: func get_area(self:Shape->Num): # In order to work with an enum, it's most often handy to use a 'when' # statement to get the internal values: when self is Point: return 0 is Circle(r): return Num.PI * r^2 is Rectangle(w, h): return w * h # 'when' statements are checked for exhaustiveness, so the compiler # will give an error if you forgot any cases. You can also use 'else:' # if you want a fallback to handle other cases. func demo_enums(): # Enums are constructed like this: my_shape := Shape.Circle(1.0) # If an enum type doesn't have any associated data, it is not invoked as a # function, but is just a static value: other_shape := Shape.Point # Similar to structs, enums automatically define comparisons, conversion # to text, and hashing: >> my_shape == other_shape = no >> "$my_shape" == "Circle(1)" = yes >> {my_shape="nice"} = {Shape.Circle(1)="nice"} func demo_lambdas(): # Lambdas, or anonymous functions, can be used like this: add_one := func(x:Int): x + 1 >> add_one(5) = 6 # Lambdas can capture closure values, but only as a snapshot from when the # lambda was created: n := 10 add_n := func(x:Int): x + n >> add_n(5) = 15 # The lambda's closure won't change when this variable is reassigned: n = -999 >> add_n(5) = 15