Mono to micro-services: Split the fat application
@xav-b|October 16, 2015 (9y ago)5 views
As articles state everywhere, we're living in a fast pace digital age. Project complexity, or business growth, challenges existing development patterns. That's why many developers are evolving from the monolithic application toward micro-services. Facebook is moving away from its big blue app. Soundcloud is embracing microservices.
Yet this can be a daunting process, so what for ?
- Scale. Better plugging new components than digging into an ocean of code.
- Split a complex problem into smaller ones, easier to solve and maintain.
- Distribute work through independent teams.
- Open technologies friendliness. Isolating a service into a container makes it straightforward to distribute and use.
- It also allows different, loosely coupled stacks to communicate.
Once upon a time, there were a fat code block called Intuition, my algorithmic trading platform. In this article, we will engineer a simplified version divided into well defined components.
Code Components
First, we're going to write the business logic, following the single responsability principle and one of my favorite code mentra :
Prefer composition over inheritance
The point is to identify key components of the problem, and code a specific solution for each of them. It will articulate our application around the collaboration of clear abstractions.
As an illustration, we start with the RandomAlgo
class. Nowadays Python tends
to be the go-to language for data analysis and rapid prototyping. A
great fit for our purpose.
This implementation focuses on a single thing : detecting buy signals. But once you get such signal, how do you invest your portfolio ? This is the responsibility of a new component.
Then we can improve our previous algorithm's event
method, taking advantage of
composition.
Here are two simple components that produce readable and efficient code. Now we can develop more sophisticated portfolio optimizations without touching the algorithm internals. This is also a huge gain early in a project when we're not sure how things will evolve.
Developers/quants should only focus on this core logic. In the next section, we're going to unfold a separate part of the system. The communication layer will solve one question: how do we produce and consume events ?
Inter-components messaging
Let's state the problem. We want each algorithm to receive interesting events and publish its own data. The kind of challenge Internet of Things (IoT) is tackling. We will find empirically that our modular approach allows us to pick the right tool, even within a-priori unrelated fields.
The code below leverages MQTT to bring M2M messaging to the application. Notice we're diversifying our stack with node.js. Indeed it's one of the most convenient language to deal with event-oriented systems (Javascript, in general, is gaining some traction in the IoT space).
That's neat! But we still need to connect this messaging layer with the actual python algorithm. RPC (Remote Procedure Call) protocol comes handy for the task, especially with zerorpc. Here is the full implementation with more explanations.
The code above calls our algorithm's methods. Here is how to expose them over RPC.
At this point we are ready to run the app. Let's fire up 3 terminals, install requirements, and make the machines to trade.
In this state, our implementation is over-engineered. But we designed a sustainable architecture to wire up small components. And from here we can extend the system.
- One can focus on algorithms without worrying about events plumbing.
- The corollary: switching to a new messaging technology won't affect the way we develop algorithms.
- We can even swipe algorithms by changing the rpc address. A service discovery component could expose which backends are available and how to reach them.
- A project like octoblu adds devices authentification, data sharing, ...
- We could implement data sources that connect to live market or databases, compute indicators like moving averages and publish them to algorithms.
Conclusion
Given our API definition, a contributor can hack on any component without breaking the project as a whole. In fast pace environment, with constant iterations, this architecture can make or break products.
This is especially true in the raising container world. Assuming we package each component into specialized containers, we smooth the way to a scalable infrastructure we can test, distribute, deploy and grow.