Deltacloud API using modular Sinatra way

Posted on | 1075 words | ~6mins

When we started the Deltacloud API project three years ago, we thought the best way how to do it would be to use the Sinatra framework. This Ruby framework provides a simple DSL for writing small-size web applications. And since Deltacloud API does not use any database or complicated messaging system it is perfect use-case for Sinatra application.

However, after a while, we realized that using just plain Sinatra routes is not perfectly DRY, since we repeated too much code and actions. So we developed the Rabbit. This small DSL allows us to build a robust REST API application with many collections and operations with just little effort.

After a while, the Sinatra guys came with a new approach how to write applications. Instead of using the application way, where you type routes directly to the Ruby file, you can create classes, inherit from Sinatra::Base class and then run these classes as Rack containers. This approach has many benefits. The biggest one is that every Sinatra class could be mounted as a Rack container to other Rack compatible framework (like Rails). In simple words, you can connect pieces of your web application like puzzle. The second benefit is that your application doesn’t run immediately after it’s launched by the Ruby interpreter. Instead you need to use some ‘rack’ deployer, like rackup or thin to spawn the whole application.

We realized that using the ‘old’ way to write Sinatra apps could be somehow limiting us in future and it could disallow us to use power of the Rack containers. Also, we keep listening to community and community is demanding a Ruby library, that will use our ‘drivers’ with a rock solid API, just like fog is doing.

So I spent several hours thinking about how to simply rewrite Deltacloud API in the Sinatra modular manner. It looked to be an impossible task in begging but I accepted that challenge.

The first puzzle which was needed was the Rabbit DSL. The web part of Deltacloud API is written entirely in this framework, so changing it back to plain routes would cause Rabbit to loose all the powerful features it has.

So I extracted Rabbit out from Deltacloud API and published it on Github. I rewrote it from scratch, but I tried very hard to preserve all the features and syntax we are used to work with. I think I was pretty successful in this and Rabbit is now feature complete with ~90% of code coverage.

The next hard step was to deal with different features, we have in Deltacloud API that are very tide to our drivers, like ‘features’, ‘dynamic driver switching’ or ‘capabilities’. Those bits were very important to us, so changing them or removing them would break our promise of backward compatibility.

So currently, I have almost all collections ported to modular Deltacloud API, and almost all drivers work as well. Those drivers which do not work now , require just small tweaks to start working properly. You can see the progress in another Github repository.

Of course, I have made big changes to our internal code structure. First, all collections are implemented as independent Sinatra::Base classes and isolated in application as ‘modules’. The simple collection looks like this:

As you can see, the syntax is the same as in internal Deltacloud API Rabbit, however there are some tweaks I made to make it more effective. The first tweak you can see is that for the :show operation, there is no param :id defined. I realized that some particular REST operations we have in Rabbit, always set this parameter, so now Rabbit will do it automatically. The next thing is that I removed description (well is still there). The truth is that we did not used it too often and the description of collections and operations was just repeating all the time. Now Rabbit is generating this automatically as well.

There are more tweaks I want to demonstrate on the other collection example:

The first things you can see are the check_capability and the check_features methods in the very beginning of the class. These ’triggers’ will set a lambda (Ruby stored procedure) and will call this lambda when processing the HTTP request.

The capability check will assure that method required for executing operation is available in the driver. So in other words, when the driver does not have create_instance method defined, the :create operation will return HTTP status 417 (Precondition failed) to the client. The reason I used lambda function is that the driver variable can change ‘per-request’ as the client switch the Deltacloud API driver. So the evaluation whether the operation should be executed or not is done for every request. The :with_capability option will tell Rabbit what method should be checked on driver.

The next important things are features. In Deltacloud API we support many cloud providers and for some of them you can use additional parameters in the HTTP request to set or change features, that are available only for a particular driver.

And since the Rabbit and Deltacloud API are now completely separated, there is no longer connection from Rabbit to the drivers code. In fact, Rabbit can be used now for any other Sinatra-based project. So I added another lambda, to check if the driver that is currently set for the current HTTP request has defined the given feature.

If it has the feature defined, then we add a new parameter to the particular operation. In simple words, if the EC2 driver has user_data feature enabled in the driver, then additional :user_data parameter will be added to the :create operation.

The last, but important change is that I renamed those operation which always acts as ‘actions’ to action. This operation will automatically get the :id parameter and is defined as HTTP POST (this can be changed using :http_method).

I think the biggest pain is now resolved and there is now a way to go to implement modular Deltacloud API. There is still a ton of work to be done, like importing all the unit tests and cucumber tests we have in Deltacloud currently, or the automatic documentation system.

The project could be now deployed or mounted as regular Rack container. I used the config.ru file, which define how it should be started:

The same approach can be used in Rails or Padrino framework, so in theory, you can access Deltacloud API without your application, without running it as separate daemon.