The Alda client and server are packaged together in the same uberjar. You can build the project by running a single command (requires Boot), boot build
, while in the root directory of this repo. Note that this command requires an -o/--output-dir
argument, which is the directory where the executable files alda
and alda.exe
will be written. I like to use /tmp
(boot build -o /tmp
). I can then try out changes by running /tmp/alda <cli args here>
.
The
build
task requires Launch4j in order to build the Windows executable.(If you're on a Mac with Homebrew installed, you can install Launch4j by running
brew install launch4j
.)
Note that the client forks server processes into the background, which continue to run even after the client has exited. If you're testing out changes you've made to the server code, you will need to restart the server by running alda restart
, which will stop the server and start a new process using your new build.
You should run boot test
prior to submitting any Pull Request involving changes to the server. This will run automated tests that live in the server/test
directory.
It is generally good to add to the existing tests wherever it makes sense, i.e. whenever there is a new test case that Alda needs to consider. Test-driven development is a good idea.
If you find yourself adding a new file to the tests, be sure to add its namespace to the test
task option in build.boot
so that it will be included when you run the tests via boot test
.
The Alda client is a fairly straightforward Java CLI app that uses JCommander to parse command-line arguments.
Interaction with servers is done via simple HTTP requests. Unless specified via the -H/--host
option, the Alda client assumes the servers are run locally and sends requests to localhost. The default port is 27713.
Running alda start
forks a new Alda process in the background, passing it the (hidden) server
command to start the server. Server output is hidden from the user (though the client will report errors). To see server output for development purposes, you can start a server in the foreground by running alda server
. You may specify a port via the -p/--port
option (e.g. alda -p 2000 server
) -- just make sure you're sending requests from the client to the right port (e.g. alda -p 2000 play -f my-score.alda
).
To stop a server, run alda stop
. To restart a server (e.g. to try out changes to the server code), run alda restart
after re-building the project.
The Alda server is written in Clojure. It handles a variety of things, including parsing Alda code into executable Clojure code, executing the code to create or modify a score, modeling a musical score as a Clojure map of data, and using the map of data as input to interpret the score (generating sound).
Parsing begins with the parse-input
function in the alda.parser
namespace. This function uses a series of parsers built using Instaparse, an excellent parser-generator library for Clojure.
The grammars for each step of the parsing process are composed from small files written in BNF (with some Instaparse-specific sugar); if you find yourself editing any of these files, it may be helpful to read up on Instaparse. The tutorial in the Instaparse README is comprehensive and excellent.
The parsers transform Alda code into an intermediate AST, which looks something like this:
[:score
[:part
[:calls [:name "piano"]]
[:note
[:pitch "c"]
[:duration
[:note-length [:positive-number "8"]]]]
[:note
[:pitch "e"]]
[:note
[:pitch "g"]]
[:chord
[:note
[:pitch "c"]
[:duration [:note-length [:positive-number "1"]]]]
[:note
[:pitch "f"]]
[:note
[:pitch "a"]]]]]
These parse trees are then transformed into Clojure code which, when run, will produce a data representation of a musical score.
Clojure is a Lisp; in Lisp, code is data and data is code. This powerful concept allows us to represent a morsel of code as a list of elements. The first element in the list is a function, and every subsequent element is an argument to that function. These code morsels can even be nested, just like our parse tree. Alda's parser's transformation phase translates each type of node in the parse tree into a Clojure expression that can be evaluated with the help of the alda.lisp
namespace.
alda.parser=> (parse-input "piano: c8 e g c1/f/a")
(alda.lisp/score
(alda.lisp/part {:names ["piano"]}
(alda.lisp/note
(alda.lisp/pitch :c)
(alda.lisp/duration (alda.lisp/note-length 8)))
(alda.lisp/note
(alda.lisp/pitch :e))
(alda.lisp/note
(alda.lisp/pitch :g))
(alda.lisp/chord
(alda.lisp/note
(alda.lisp/pitch :c)
(alda.lisp/duration (alda.lisp/note-length 1)))
(alda.lisp/note
(alda.lisp/pitch :f))
(alda.lisp/note
(alda.lisp/pitch :a)))))
When you evaluate a score S-expression like the one above, the result is a map of score information, which provides all of the data that Alda's audio component needs to make an audible version of your score.
{:events
#{{:offset 750.0,
:instrument "piano-foyYJ",
:volume 1.0,
:track-volume 0.7874015748031497,
:midi-note 69,
:pitch 440.0,
:duration 1800.0}
{:offset 500.0,
:instrument "piano-foyYJ",
:volume 1.0,
:track-volume 0.7874015748031497,
:midi-note 67,
:pitch 391.99543598174927,
:duration 225.0}
{:offset 750.0,
:instrument "piano-foyYJ",
:volume 1.0,
:track-volume 0.7874015748031497,
:midi-note 65,
:pitch 349.2282314330039,
:duration 1800.0}
{:offset 0,
:instrument "piano-foyYJ",
:volume 1.0,
:track-volume 0.7874015748031497,
:midi-note 60,
:pitch 261.6255653005986,
:duration 225.0}
{:offset 750.0,
:instrument "piano-foyYJ",
:volume 1.0,
:track-volume 0.7874015748031497,
:midi-note 60,
:pitch 261.6255653005986,
:duration 1800.0}
{:offset 250.0,
:instrument "piano-foyYJ",
:volume 1.0,
:track-volume 0.7874015748031497,
:midi-note 64,
:pitch 329.6275569128699,
:duration 225.0}},
:markers {:start 0},
:instruments
{"piano-foyYJ"
{:octave 4,
:current-offset {:offset 2750.0},
:config {:type :midi, :patch 1},
:duration 4,
:volume 1.0,
:last-offset {:offset 750.0},
:id "piano-foyYJ",
:quantization 0.9,
:tempo 120,
:panning 0.5,
:current-marker :start,
:stock "midi-acoustic-grand-piano",
:track-volume 0.7874015748031497}}}
There are 3 keys in this map:
- :events -- a set of note events
- :markers -- a map of marker names to offsets, expressed as milliseconds from the beginning of the score (
:start
is a special marker that is always placed at offset 0) - :instruments -- a map of randomly-generated ids to all of the information that Alda has about an instrument, at the point where the score ends.
A note event contains information such as the pitch, MIDI note and duration of a note, which instrument instance is playing the note, and what its offset is relative to the beginning of the score (i.e., where the note is in the score)
Because alda.lisp
is a Clojure DSL, it's possible to use it to build scores within a Clojure program, as an alternative to using Alda syntax:
(ns my-clj-project.core
(:require [alda.lisp :refer :all]))
(score
(part "piano"
(note (pitch :c) (duration (note-length 8)))
(note (pitch :d))
(note (pitch :e))
(note (pitch :f))
(note (pitch :g))
(note (pitch :a))
(note (pitch :b))
(octave :up)
(note (pitch :c))))
The alda.sound
namespace handles the implementation details of playing the score.
There is an "audio type" abstraction which refers to different ways to generate audio, e.g. MIDI, waveform synthesis, samples, etc. Adding a new audio type is as simple as providing an implementation for each of the multimethods in this namespace, i.e. set-up-audio-type!
, refresh-audio-type!
, tear-down-audio-type!
, start-event!
and stop-event!
.
The play!
function handles playing an entire Alda score. It does this by using a JSyn SynthesisEngine to schedule all of the note events to be played in realtime. The time that each event starts and stops is determined by its :offset
and :duration
.
What happens, exactly, at the beginning and end of an event, is determined by the start-event!
/stop-event!
implementations for each instrument type. For example, for MIDI instruments, start-event!
sets parameters such as volume and panning and sends a MIDI note-on message at the beginning of the score plus :offset
milliseconds, and stop-event!
sends a note-off message :duration
milliseconds later.
Although technically a part of alda.lisp
, stock instrument configurations are defined here, which are included in an Alda score map, and then used by alda.sound
to provide details about how to play an instrument's note events. Each instrument's :config
field is available to the alda.sound/play-event!
function via the :instrument
field of the event.
alda.repl
is the codebase for Alda's Read-Eval-Play Loop, which lets you build a score interactively by entering Alda code one line at a time.
There are built-in commands defined in alda.repl.commands
that are defined using the defcommand
macro. Defining a command here makes it available from the Alda REPL prompt by typing a colon before the name of the command, i.e. :score
.
The core logic for what goes on behind the curtain when you use the REPL lives in alda.repl.core
. A good practice for implementing a REPL command in alda.repl.commands
is to move implementation details into alda.repl.core
(or perhaps into a new sub-namespace of alda.repl
, if appropriate) if the body of the command definition starts to get too long.
alda.server/start-server!
is the entrypoint to the Alda server. It starts a Clojure web app using Ring and Compojure, and serves it via Jetty.
Requests can be made to the server via any HTTP client, e.g. curl, or via the Alda client, which communicates with the server by making HTTP requests.