% getting started % - % - # getting started with alive `alive` is a language for creating and changing realtime programs while they are running continuously. `alive` can be used to create music, visuals or installations, but by itself creates neither sound nor video. Rather, `alive` is used together with other tools and synthesizers (for example [SuperCollider][supercollider] or [Pilot][pilot]). In such an ensemble of tools, `alive` takes the role of a 'conductor', telling the other tools what to play when by sending commands to them using a variety of protocols, such as OSC and MIDI. Before we get to making sound though, we should learn a bit about the `alive` programming language, and how to install and use it. ## installation `alive` is written in the Lua programming language, and is compatible with both Lua 5.3 and luajit. `alive` requires a number of other Lua libraries to run. To manage these dependencies `luarocks` is used, follow [this link][luarocks] for instructions on setting it up. Once you have luarocks, you can install the dependencies for `alive`: $ luarocks install moonscript $ luarocks install luasystem $ luarocks install osc $ luarocks install luasocket $ luarocks install https://raw.githubusercontent.com/s-ol/lua-rtmidi/master/lua-rtmidi-dev-1.rockspec While `moonscript` and `luasystem` are required by the core of `alive`, the other packages (`osc`, `luasocket` and `lua-rtmidi`) are specific to some modules of the alive language, and as long as you don't need to use these modules installation is optional. In a later part of this guide, we will be using modules that require `osc` and `luasocket`, so it is recommended to install at least these two. However it is possible to follow a large portion of the guide without any of the three. There will be a note marking the parts of the guide where specific dependencies are required. After installing the dependencies, you can clone the [`alivecoding` repository][git] using git: $ git clone https://github.com/s-ol/alivecoding.git You should now be able to run `alive` from within the repository, but first, you will need to create a file for it to run. Create an empty file in the text editor you want to use, and save it as `hello.alv` in the `alivecoding` repository. Then launch the `alive` copilot like so: $ moon init.moon hello.alv hello.alv changed at 1583171231 You should see a similar output message, indicating that `alive` processed the file. From now on, whenever you change `hello.alv` and save the file, `alive` will reload it and execute your new code. When you are done, you can stop the copilot at any time using `^C` (control-C). ## evaluating code `alive`'s syntax is very similar to Lisp. Expressions take the form of parenthesized lists like `(head a b c...)`, where the first element of the list (`head`) is the name of an operator or function, which defines what the expression as a whole will do, while the other elements are parameters whose meaning depends on the `head`. Let's start with a simple operator, [trace][]: [trace][] is used to inspect values by printing them to the copilot. Enter the following in your file and save it: (trace "hello world") As soon as you save the file, you should notice two things happening: 1. The copilot should print two new lines to the terminal: hello.alv changed at 1583424169 trace "hello world": In the first line, it notifies us that the file has changed. In the second line, you can see the output from [trace][]: it lets you know that `"hello world"` evaluated to a `Value` with the type `str` (string) and contents `hello world`. 2. The copilot will make a small modification to your file. Depending on the editor you are using, this may either result in you seeing the modification immediately, or a notice appearing that offers the option to reload the file. If it is the latter, confirm the notification to accept the changes. If there is an option to do so, you may want to configure your editor to always reload the file automatically. The code should now look like this: ([1]trace "hello world") The `[1]` that the copilot added to our expression is that expression's `tag`. In `alive`, every expression has a tag that helps the copilot to identify the individual expressions as you make changes to your code. The copilot will make sure that all expressions are tagged by adding missing tags when you save the file, but you have to watch out not to duplicate a tag when copying and pasting code. When you duplicate code that already has tags, you can either manually change the tags to be unique, or simply delete the whole tag (including the square brackets) and let the copilot generate a new one for you the next time you save the file. ### basic types Strings can be written in two ways: using double quotes (`"`), as we did above, or using single quotes (`'`). In both types of strings, you can escape a quote that otherwise would signify the end of the string by adding a single backslash before it. Consequently, backslashes also have to be escaped in the same way. The following are all valid strings: "hello world" 'hello world' "it's a beautiful day" 'it\'s a beautiful day' "this is a backslash: \\" "this is a double quote: \"" "" '' Aside from strings, there are two more types of values that you can use when writing alive programs: numbers and booleans. Numbers use the digits 0-9 and can be integers, contain a decimal point, or start or end with a decimal point. Numbers can start with a negetive sign. The following are all valid numbers: 0 12 -7 0.1 10. .1 123. There are only two boolean values, `true` and `false`: true false You can try using [trace][] with all of these values to get used to how they are printed. ### comments To annotate your code, you can use comments. In `alive`, comments begin with `#(` and end on a matching `)`. This way you can comment out a complete expression simply by adding a `#` character in front. #(this is a comment) #(this is a long, multi-line comment, (and it also has nested parentheses). It ends after this sentence.) You can put comments anywhere in your program where whitespace is allowed and it will simply be ignored by `alive`. ### importing modules Apart from [trace][], there are only very little builtin operators in `alive` - you can see all of them in the *builtins* section of the [reference][:/:]. All of the 'real' functionality of `alive` is grouped into *modules*, that have to be loaded individually. *Modules* help organize all of the operators so that it is less overwhelming to look for a concrete feature. It is also possible to create your own plugins as new modules, which will be covered in another guide soon. Let's try using the [`+` operator][:math/+:] from the [math/][] module. To use operators from a module, we need to tell `alive` to load it first: We can load *all* the operators from the [math/][] module into the current scope using the [import*][] builtin: (import* math) (trace (+ 1 2)) prints trace (+ 1 2): Because it can get a bit confusing when all imported operators are mixed in the global scope, it is also possible to load the module into its own scope and use it with a prefix. This is what the [import][] builtin is for: (import math) (trace (math/+ 1 2)) ### defining symbols Both [import][] and [import*][] are actually shorthands for other builtins: [def][] and [use][]. [def][] is used to *define a symbol* in the current scope. You can use it to associate a *symbol* (a name, like `hello`, `trace`, or `+`) with a value. After a symbol is defined, the name becomes an alias that behaves like the value itself. For example, we can use [def][] to give the result of our calculation a name, and then refer to it by that symbol in the [trace][] operator: (import* math) (def result (+ 1 2)) (trace result) Symbols need to start with a letter or one of the characters `-_+*/.!?=%`. After the first character, numbers are also allowed. There are two types of symbols that are treated specially: symbols containing a slash (`math/+`), and symbols starting and ending with asterisks (`*clock*`): - Symbols containing slashes (except at beginning and end of the symbol) are split into multiple symbols, and looked up recursively in the scope. For example, `math/+` is found by first looking for a value for the symbol `math`, and then looking for the symbol `+` in that value. If the value for the symbol `math` is not a scope, an error is thrown. - Symbols starting and ending with asterisks are called `dynamic symbols` and are looked up in a different way inside user-defined functions. This will be covered in detail later. - The two special formats can be mixed: when evaluating `*hello*/world`, `alive` will look for the symbol `world` within the scope found by dynamically resolving `*hello*`. The [import][] builtin is actually a shorthand for the following expression: (def math (require "math")) (trace (math/+ 1 2)) [require][] returns a *scope*, which is defined as the symbol `math`. Then `math/+` is resolved by looking for `+` in this nested scope. Note that the symbol that the scope is defined as and the name of the module that is loaded do not have to be the same, you could call the alias whatever you want: (def fancy-math (require "math")) (trace (fancy-math/+ 1 2)) In practice, this is rarely useful, which is why the [import][] shortcut exists. The full version of [import*][], on the other hand, defines every symbol from the imported module individually. The expanded version is the following: (use (require "math")) (trace (+ 1 2)) [use][] copies all symbol definitions from the scope it is passed to the current scope. Note that [import][], [import*][], [def][], and [use][] all can take multiple arguments: (import* math logic) (import midi osc) is the same as (use (require "math") (require "logic")) (def midi (require "midi") osc (require "osc")) It is common to have an [import][] and [import*][] expression at the top of an `alive` program to load all of the modules that will be used later, but the modules don't necessarily have to be loaded at the very beginning, as long as all symbols are defined before they are being used. ### nested scopes Once a symbol is defined, it cannot be changed or removed: (def a 3) (def a 4) #(error!) It is, however, possible to 'shadow' a symbol by re-defining it in a nested scope: So far, all symbols we have defined - using `def`, [import][] and [import*][] - have been defined in the *global scope*, the scope that is active in the whole `alive` program. The [do][] builtin can be used to create a new scope and evaluate some expressions in it: (import string) (def a 1 b 2) (trace (.. "first: " a " " b)) (do (def a 3) (trace (.. "second: " a " " b)) (trace (.. "third: " a " " b)) This example prints the following: trace (.. "first: " a " " b): trace (.. "second: " a " " b): trace (.. "third: " a " " b): As you can see, within a nested scope it is possible to overwrite a definition from the parent scope. Symbols that are not explicitly redefined in a nested scope keep their values, and changes in the nested scope do not impact the parent scope. ### defining functions Another builtin that creates a nested scope is [fn][], which is used to create a *user-defined function*, which can be used to simplify repetitive code, amongst other things: (import* math) (def add-and-trace (fn (a b) (trace (+ a b)))) (add-and-trace 1 2) (add-and-trace 3 4) Here a *function* `add-and-trace` is defined. When defining a function, first the names of the parameters have to be given. The function defined here takes two parameters, `a` and `b`. The last part of the function definition is called the *function body*. A function created using [fn][] can be called just like an operator. When a function is called, the parameters to the function are defined with the names given in the definition, and then the function body is executed. The previous example is equivalent to the following: (import* math) (def add-and-trace (fn (a b) (trace (+ a b))) (do (let a 1 b 2) (trace (+ a b))) (do (let a 3 b 4) (trace (+ a b))) and the output of both is: trace (+ a b): trace (+ a b): In `alive`, functions are first-class values and can be passed around just like numbers, strings, etc. However it is very common to define a function with a name, so there is the `defn` shorthand, which combines the `def` and `fn` builtins into a single expression. Compare this equivalent definition of the `add-and-trace` function: (defn add-and-trace (a b) (trace (+ a b))) ### evaltime and runtime So far, `alive` may seem a lot like any other programming language - you write some code, save the file, and it runs, printing some output. "What about the 'continuously running' aspect from the introduction?", you may ask yourself. So far, we have only seen *evaltime* execution in alive - but there is also *runtime* behavior. At *evaltime*, that is whenever there is change to the source code, `alive` behaves similar to a Lisp. This is the part we have seen so far. But once one such *eval cycle* has executed, *runtime* starts, and `alive` behaves like a dataflow system like [PureData][pd], [Max/MSP][max] or [vvvv][vvvv]. What looked so far like static constants are actually *streams* of values. Whenever an input to an operator changes, the operator (may) update and respond with a change to its output as well. To see this in action, we need to start with a changing value. Number literals like `1` and `2`, which we used so far, are *evaltime constant*, which means simply that they will never update. Since all inputs to our [math/+][] operator are *evaltime constant*, the result is constant as well. To get some *runtime* activity, we have to introduce a side-effect input from somewhere outside the system. The [time/][] module contains a number of operators whose outputs update over time. Lets take a look at [time/tick][]: (import* time) (trace (tick 1)) This will print a series of numbers, incrementing by 1 every second. The parameter to [time/tick][] controls how quickly it counts - try changing it to `0.5` or `2`. As you can see, we can change [time/tick][] *while it is running*, but it doesn't lose track of where it was! All of the other things we learned above apply to streams of values as well - we can use [def][] to store them in the scope, transform them using the ops from the [math/][] module and so on: (import* time math) (def tik (tick 0.25)) (trace (/ tik 4)) Note that if you leave the [time/tick][]'s *tag* in place when you move it into the [def][] expression, it will keep on running steadily even then. ## making sound As mentioned earlier, `alive` doesn't produce sound by itself. Instead, it is paired with other tools, and takes the role of a 'Conductor', sending commands and sequencing other tools. For the sake of this guide, we will be controlling [Pilot][pilot], a simple UDP-controlled synthesizer. You can go ahead and download and open it now. You should see a small window with a bunch of cryptic symbols and a little command line at the bottom. To verify that everything is working so far, try typing in `84c` and hitting enter. This should play a short sound (the note 4C, played by the 8th default synthesizer voice in Pilot). To talk to Pilot from `alive`, we will use the [pilot/][] module. Note that for this module to work, you have to have the `osc` and `luasocket` dependencies installed. To play the same sound we played by entering `84c` above every 0.5 seconds, we can use [time/every][] to send a `bang` to [pilot/play][]: (import* time) (import pilot) (pilot/play (every 0.5) 8 4 'c') You can play with the voice, octave and note values a bit. To add a simple melody, we can use [util/switch][], which will cycle through a list of parameters when used together with [time/tick][]: (import* time util) (import pilot) (pilot/play (every 0.5) 8 4 (switch (tick 0.5) 'c' 'd' 'a' 'f')) Now we can have the voice change every other loop as well: (import* time util) (import pilot) (pilot/play (every 0.5) (switch (tick 4) 8 9) 4 (switch (tick 0.5) 'c' 'd' 'a' 'f')) To round off the sound a bit, we can turn on Pilot's reverb using [pilot/effect][]. Add the following somewhere in your file: (pilot/effect "REV" 2 8) Now it's time to add some rhythm. The kick drum is voice 12 by default, and we can also add something like a snare on channel 3: (pilot/play (every 0.75) 12 2 'd' 3) (pilot/play (every 2) 13 4 'a' 4) Note that since we are using multiple individual [time/every][] instances, the timing of our voices relative to each other is not aligned - each voice started playing when the file was first saved with it added, and kept the rhythmn since. By deleting all their tags and re-saving the file, we can force `alive` to re-instantiate them all at the same time, thereby synchronising them. [supercollider]: https://supercollider.github.io/ [pilot]: https://github.com/hundredrabbits/Pilot [pd]: http://puredata.info/ [max]: https://cycling74.com/products/max [vvvv]: https://vvvv.org/ [luarocks]: https://github.com/luarocks/luarocks/#installing [git]: https://github.com/s-ol/alivecoding [reference]: reference/