The Bleeding Edge Of A Web Application...
Tweet |
Most web applications have the well-known 3-tiered structure - WebTier > ApplicationTier > DataTier. Both WebTier and ApplicationTier have the web-layer to parse the incoming HTTP requests. Its in the WebTier that one deploy’s load-balancing L4-routers like Apache/Nginx or Netscaler like appliances. HTTP requests are forwarded by the WebTier to the ApplicationTier which is generally served by a much bigger farm of servers. Web-layer in the ApplicationTier is the focus of this blog. Its a challenging area of software development for the following reasons and more -
- Huge volume of requests, with read requests generally surpassing write by an order of magnitude or so
- Change. Website content and web-service APIs both change very often
- Variety of consumers. People read/write to the web. And so do other software applications
Being a Java and JavaScript developer, my interest has been in the emergent software stacks in these two languages. To understand their raison d'être. For that, I start by taking a look at the numbers (HTTP requests) at some of the popular websites. Then move on to some of the core technical problems. And compare some of the competing software stacks.
But before discussing on the web-layer in the ApplicationTier it is instructive to look at the pure WebTier itself. Its instructive to read Netcraft’s September 2013 Web Server Survey. All the top web-servers are C/C++ based. For those unfamiliar with actual web application deployments, these web-servers are not used to host the applications themselves. They serve static pages, act as L4-routers, firewalls and load-balancers. They are placed at the very gate of modern web-shops and all requests go through them. These tasks are well defined, so, it makes sense to develop them in native languages for brute speed.
1. Quantifying the ‘Bleeding Edge’
Here are the numbers from recently published articles on Twitter, WhatsApp and Facebook. There are others who cannot not be far behind like Google, Wikipedia, Amazon, Skype etc.
- Twitter: 300K requests per second (RPS) for reading and 6000 RPS for writing - source1, source2
- WhatsApp: 10 billion requests sent and received in one day - source
- Facebook: 12 million HTTP requests per second - source
(All these articles are quite recent)
2. Why Is It Hard?
Two good resources to start understanding why these scales are hard on software development in ApplicationTier are -
- C10K problem by Kegel and Felix von Leitner
- C10M problem by Robert Graham. And this video by him is very instructive
But let me state the problem(s) simply. The reasons why it is hard to handle HTTP requests are -
- Forking a process: is too expensive a compute operation to perform everytime a request arrives
- Forking a thread: is less expensive on compute. But writing multi-threaded applications for multi-core systems is very tough (and actually forking a new thread is not inexpensive at all)
- Use thread pools: It just shifts the bottleneck. Once you have a thread-pool, each thread has to do a select() or poll() to find the next nonblocking socket ready for IO. But doing a select() or poll() on a huge array of open socket descriptors is extremely inefficient at the kernel level (checkout the deep analysis to C10K problem in the above mentioned links)
- The Event driven model: requires a paradigm shift in thinking and designing applications from bottoms-up. The best way to start grasping the idea is to read The Reactive Manifesto. This model is not very different from the SEDA architecture. Reactive applications is a very fine idea and one of the reasons why I dwelled into this subject in the first place…
3. Software Development Of Web Applications
My current views are that, broadly, there are 3 different language families to develop web applications on server and client sides -
- Server: JVM based, compiled and statically typed; Client: JavaScript
- Server: Ruby/PHP, interpreted and dynamically typed; Client: JavaScript
- Server: NodeJS, interpreted and dynamically typed; Client: JavaScript
Thus, on the server side, the choice is between JVM (polyglot), Ruby/PHP and NodeJS.
4. JVM Based Web Apps
The web-layer in JVM world is filled with 3 types of frameworks -
- Frameworks that support the servlet specification (latest one is 3.0)
- MVC frameworks
- Asynchronous event-driven frameworks based on Netty
(i) Servlet Specification Frameworks
These include Tomcat and Jetty. What is the main motivator for the servlet spec? It is to manage state information that does not exist in the stateless HTTP protocol. HttpServletRequest provides an API getSession() where the HttpSession object is a container to hold attributes for a single transaction spread across multiple HTTP request/responses. Apart from this central feature of sessions, the servlet API also defines the interfaces that the servlet container has to adhere-to to provide concurrent request processing in a sandbox environment. So is there a drawback in this idea? Yes, there is. The very idea of state brings down the performance of these containers. That is the reason why developing highly performant RESTful APIs using servlet containers is a bad idea. In RESTful, the client is expected to maintain the state, if required. Servlet containers can be tuned for statelessness, but then, that goes against one of the fundamental ideas of the spec. And my readings tell me that these frameworks don’t become highly performant on turning off the statefullness.
(ii) MVC Frameworks
These include Spring MVC, Struts, Tapestry, Wicket etc. I have used two of these - Struts2 and Wicket in building applications that have seen deployment. The fundamental motivation for these frameworks is ease-of-development (annotations etc), clean separation of concerns (MVC design pattern), lot of goodies (like templating etc) and integration with other JavaEE stacks (Struts2-Spring integration etc).
(iii) Asynchronous Event-Driven Frameworks
And now I come to the most interesting area of Java web application development. Netty based frameworks like Play! and Vert.x. These frameworks do not comply to the servlet specification. They use Netty underneath for asynchronous event based handling of HTTP requests (I cover what-the-hell-is-asynchronous-event-driven in the NodeJS section below). Netty is stateless making the server side fast and efficient. The frameworks on top are built to match the ease-of-dev and richness offered by the likes of Struts and Tapestry. They also offer APIs for client-side statefulness. So these frameworks are an effort to mix high performance with ease-of-dev. But moving to event-based and asynchronous thinking is not straightforward. It needs a mind shift akin to the transition to Object-oriented-programming. However the promise they hold is to be able to build web applications that defy Amdahl’s law. If you are a new shop with bright Java engineers wanting to build a highly scalable web-application, then, these are the frameworks you should start exploring first…
5. NodeJS - JavaScript on the server side
The list of companies and websites powered by NodeJS is growing long by the day. However NodeJS is still a newbie. Why would somebody want to use NodeJS? NodeJS makes two very interesting promises - * End-to-end JavaScript shop for you web application * High performance through event-based asynchronous model
The first promise is easy to understand. Any good web application requires a team of good designers and client-side programmers. If the programming language on both client-side and server-side are the same, then it reduces the risk of investment in diverse technologies and brings down the barriers between teams and moving people.
The second promise of performance is more interesting. Is NodeJS as fast as the Java based async frameworks? This blog presents an excellent comparison. It goes to show that NodeJS is no match to the JVM based frameworks. Its difficult to beat the JVM!
But moving ahead of comparisons, let me dwell a little more on the aspect of performance promised by the event-driven asynchronous frameworks in general. The hype around such frameworks is increasing day-by-day and is grounded on firm theoretical foundations. So how exactly does async and event-driven help? NodeJS provides a good base to explore since one cannot do anything but asynchronous event-based HTTP processing with NodeJS! Let us study this code fragment for a while - (this comes from this excellent book on NodeJS by O'reilly)
// load http module var http = require('http'); var fs = require('fs'); // create http server http.createServer(function (req, res) { // open and read in a file fs.readFile('textfile.txt', 'utf8', function(err, data) { res.writeHead(200, {'Content-Type': 'text/plain'}); if (err) { res.write('Could not find or open file for reading\n'); } else { // if no error, write file to client res.write(data); } res.end(); }); }).listen(8124, function() { console.log('bound to port 8124'); }); console.log('Server running on 8124/');
Following aspects need to be understood -
- The two instances of asynchronous behaviour - one for http I/O and the other for file I/O
- This program never blocks. NEVER.
- Multiple types of events are emitted - when a request arrives, when file I/O request completes - and these events are consumed in a single giant event loop with the NodeJS framework
- Large (N-squared) like compute algorithms should not be synchronously attempted - they take away all the processing core’s bandwidth bringing the whole system to a halt. So, such event-based asynchronous processing is most suited for applications that can be broken down into multiple stages like a SEDA architecture
- The application itself acts as one giant event-producing and event-consuming engine which should be seen as single-threaded and binding to a single-core. To make use of multiple-cores multiple-instances of NodeJS can be run on the same system
NodeJS has found tremendous traction with developer community. Am heading to JSFoo in Bangalore next week, and one look at the funnel would tell you that every second session has something to do with NodeJS. And NodeJS has a plethora of MVC frameworks which are maturing fast. Sample - Express, Geddy, FlatironJS, EmberJS - these are definitely poised to give MVC frameworks in Ruby and PHP a run for their money in simplicity, performance and features.
6. Ruby and PHP
With JVM based frameworks occupying one end of the spectrum offering high-performance + Maintainability and NodeJS based frameworks at the other end with simplicity + low-cost + ease-of-dev, how much of middle-ground is left for PHP and Ruby? I am not an expert in either of these two, so I will stay away from making predictions. One thing that is in favour of PHP/Ruby is that both are proven in large production applications while reactive Java frameworks and NodeJS are still not. How long will this status last? Will NodeJS and Java reactive frameworks take away a chunk of web applications that would otherwise have been Ruby/PHP’s? Or will the web applications playing field get expanded with the entry of these new players creating room for all?
7. What to use for my project?
I roundoff my blog with a guidance, albeit reluctantly. Apart from the usual suspects of time-to-market, capex-opex investment, engineering-skill and requirements-complexity that make project delivery complex, I propose that few more criterions come into play when it comes to web applications -
- Is the web application part of a packaged-product (bundled in a CDROM) or delivered as part of the general web or Saas?
- Is the web application intra-enterprise or for open-internet usage?
- Is the web application majorly for human consumption or accessed by other software services?
I leave those aspects to the good judgement of the readers. I would definitely like to get feedback on those who disagree from my guidance below. The idea of writing this guidance is to paint broad strokes… Exceptions among project/people always exist!
JVM Based
- Performance: JVM is worlds best VM in performance. Ruby/PHP/NodeJS are interpreted and don't come close in performance (doing anything in JRuby per me is simply a bad idea). Facebook created HipHop for PHP to make it scale - this counts as an exception. Twitter, LinkedIn shifted from Ruby to Scala (which is JVM based) and achieved higher performance numbers. One can find umpteen examples like this…
- Development Time: Java and other JVM languages are slower to develop compared to Ruby/PHP/NodeJS. And thats the reason why frameworks like Play! are trying hard to sell themselves as suited for fast development
- Cost: Java developers are more expensive
- Suited for: Large web applications. Enterprise products. Mission critical applications
Ruby, PHP
- Performance: Definitely not bad
- Development Time: Fast
- Cost: Medium
- Suited for: Medium sized projects
NodeJS
- Performance: The jury is still out. Does the Google V8 engine challenge and beat PHP/Ruby? It will never be able to match the JVM though.
- Development Time: Fast
- Cost: Low, since the whole application is built on a single language stack the server-side developers and client-side developers can co-work
- Suited for: Smaller chatty applications
comments powered by Disqus