Alki (AL-kai) is a Dependency Injection framework for Ruby. Alki is designed to help organize and scale your project, so you can focus on the important stuff. It can be used alongside frameworks such as Ruby on Rails.
Some high level features:
-
Easily manage objects and dependencies
-
Enables writing reusable and testable code
-
Requires no annotations, mixins, or other changes to your code.
-
Developer console (built on pry)
-
Automatic code reloading
-
Powerful DSL toolkit
-
Extensible
For questions or discussion about Alki, head over to the Google Group.
Add this line to your application’s Gemfile:
gem 'alki'
And then execute:
$ bundle
Or install it yourself as:
$ gem install alki
Alki simplifies project organization by pulling out all of the "connective tissue" that connects your classes and modules together. Alki provides a central place to build your application objects, where they can easily refer to other objects, and access configuration settings.
For example, if your project has an http client
module MyProject
class HttpClient
...
end
end
you might normally use it by either creating a new instance of it when needed, or directly calling methods on the class.
client = MyProject::HttpClient.new
client.get '/page1.html'
# -or-
MyProject::HttpClient.get '/page1.html'
With Alki, you would instead define instructions for how to build the http client.
Alki do
service :http_client do
require 'my_project/http_client'
MyProject::HttpClient.new
end
end
The 'service' method defines a new object with the name 'http_client', and the provided block contains the code needed to build it.
To use the http client, an "Assembly" for the project has to be created. Assemblies are container objects that hold your application objects once they’re built.
Typically you can just add the following code to your project’s main library file. More information can be found about this method here.
require 'alki'
Alki.project_assembly!
This will add a 'new' method to the the MyProject module, which will return a new Assembly instance. The http client can be accessed by calling the method that matches the name we chose for it when it was defined, 'http_client'.
require 'my_project'
app = MyProject.new
app.http_client.get '/page1.html'
Most projects, at some point, have to deal with configuration settings.
In Ruby the typical pattern is to reference Modules and Classes directly, either by constructing them on demand when needed, or calling class methods on them.
Often this leaves Modules and Classes to essentially have to configure themselves. Sometimes this is by pulling configuration from the environment or a configration file, sometimes it’s by requiring a global variable to be set before the class is used.
This means that classes are often cluttered with configuration code. Classes from different libraries might have different, inconsistent ways of being configured, and it can be difficult if not impossible to change how they’re configured.
In Alki, configuration becomes straightforward. Classes can just take configuration settings as initializer parameters, without worrying about where they comes from.
module MyProject
class HttpClient
def initializer(base_uri)
@base_uri = base_uri
end
...
end
end
Then the Assembly can be updated to pass in a configuration parameter. We can also define a new element for our base_url setting.
Alki do
set :http_base_url do
ENV.fetch('HTTP_BASE_URL')
end
service :http_client do
require 'my_project/http_client'
MyProject::HttpClient.new http_base_url
end
end
The 'set' method is used to define simple values in the Assembly, and the http client can refer to it by name.
Nothing changes with how we use the http client, but it’s now pulling it’s configuration from the environment.
Changing this to have a default value, pull from a different source, or have multiple clients with different configuration parameters can be done easily without changing the class code.
For example, to change the settings to come from a YAML file we can define a new element to parse the YAML file, and then redifine our setting to pull it’s value from the parsed options. Note that the order doesn’t matter. Even if a dependency is defined after an element that uses it, the objects will still be built in the correct order.
Alki do
set :yaml_settings do
require 'yaml'
YAML.load_file File.join(config_dir,'settings.yml')
end
set :http_base_url do
yaml_settings['http_base_ur']
end
service :http_client do
require 'my_project/http_client'
MyProject::HttpClient.new http_base_url
end
end
No changes to the class, or in this case even the build code for the http client, are required.
Configuration settings can be grouped together in the assembly, to namespace them. This group can be called anything, but we’ll call it 'settings'.
Alki do
group :settings do
set :http_base_url do
yaml_settings['http_base_ur']
end
set :yaml_settings do
require 'yaml'
YAML.parse(File.read(File.join(config_dir,'settings.yml')))
end
end
service :http_client do
require 'my_project/http_client'
MyProject::HttpClient.new settings.http_base_url
end
end
The can also be moved to an entirely different file to keep them seperate from your application objects.
Alki do
set :http_base_url do
yaml_settings['http_base_ur']
end
set :yaml_settings do
require 'yaml'
YAML.parse(File.read(File.join(config_dir,'settings.yml')))
end
end
Alki do
load :settings
service :http_client do
require 'my_project/http_client'
MyProject::HttpClient.new settings.http_base_url
end
end
Dependency Injection is a way of designing your classes to maximise their modularity and reusability.
Much like with configuration settings, instead of directly referencing a classes dependencies a depenency injected class takes it’s dependencies as arguments.
The advantage of this is that classes can be developed and tested in isolation, can have dependencies easily changed by other code (like users of a library), or have multiple instances with different dependencies.
Like configuration settings, a lot of Ruby projects end up needing some amount of dependency injection, but it’s typically implemented in an ad-hoc way with class variables or other global variables.
With Alki, dependency injection is the norm and very straightforward.
Perhaps we have client for API that works over HTTP. We can write our API client so that it takes an HTTP client as an arugment and just calls methods on it. It doesn’t need to pick a particular client library or decide what configuration settings it should have.
module MyProject
class ApiClient
def initialize(http_client)
@http_client = http_client
end
def all
parse_response @http_client.get('/all.json')
end
...
end
end
With this design it’s easy to change out the http client, or provide a stubbed or mocked one for testing. It maximizes the reusability of our API client because it only handles one thing (Single Responsibility Principle).
To add it to the previous assembly definition.
load :settings
service :api_client do
require 'my_project/api_client'
MyProject::ApiClient.new http_client
end
service :http_client do
require 'my_project/http_client'
MyProject::HttpClient.new settings.http_base_url
end
Note
|
Full "todo" source code can be found here |
To demonstrate how a full Alki project can be put together, we’ll go through an example application. This application is a little todo list command line utility that stores the todo list in a json file.
require 'alki'
Alki.project_assembly!
This will create a module called Todo
that is an empty assembly:
$ bundle exec irb -Ilib
2.4.0 :001 > require 'todo'
=> true
2.4.0 :002 > todo = Todo.new
=> #<Todo:21964520>
Adding things to the assembly requires an Assembly definition file.
By convention this is named config/assembly.rb
and is built using a DSL.
There are a handful of different element types in Assemblies.
Elements can refer to other elements, and can be defined in any order.
Full documentation of the DSL can be found here.
Below is an example Assembly definition for the todo application which uses a few of the core element types.
Alki do
load :settings
service :interface do
require 'todo/readline_interface'
Todo::ReadlineInterface.new settings.prompt, handler
end
service :handler do
require 'todo/command_handler'
Todo::CommandHandler.new db
end
service :db do
require 'todo/store_db'
Todo::StoreDb.new file_store
end
service :file_store do
require 'todo/json_file_store'
Todo::JsonFileStore.new settings.db_path
end
end
Along with a settings file to define our application settings.
Alki do
set(:home) { ENV['HOME'] }
set(:db_path) { ENV['TODO_DB_PATH'] || File.join(home,'.todo_db') }
set :prompt, 'todo> '
end
Settings and services can be accessed by requiring the main "lib/todo.rb" file, creating a new assembly instance, and then calling methods to refer to the different elements.
$ bundle exec irb -Ilib
2.4.0 :001 > require 'todo'
=> true
2.4.0 :002 > todo = Todo.new
=> #<Todo:21964520>
2.4.0 :003 > todo.settings.prompt
=> "> "
2.4.0 :004 > todo.interface.run
> ?
All commands can be shortened to their first letters
print
add <description>
edit <id> <description>
complete <id>
uncomplete <id>
remove <id>
move <from> <to>
quit
The alki-console developer console gem can also be used to quickly work with assemblies.
It can be used by adding gem 'alki-console'
to the Gemfile and running bundle --binstubs
to add the console script to the 'bin' directory.
Elements can be accessed directly without having to manually create an assembly instance.
$ bin/alki-console
todo> settings.prompt
=> 'todo> '
Read more about creating executables with Alki, including how to create executables to distribute in a gem, here
In the todo example, it’s a CLI utility so it requires an executable. The executable just needs to require the main project file, create a new instance of the assembly, and call a method on a service.
#!/usr/bin/env ruby
# Setup
require 'bundler/setup'
require 'alki/bin'
require 'todo'
Todo.new.interface.run
$ bin/todo
> print
1. take out the trash
> quit
$
More documentation can be found at alki.io, with in-depth explanations of Assemblies, the Assembly DSL, and other projects that are part of Alki.