-
Notifications
You must be signed in to change notification settings - Fork 549
Puma vs Phusion Passenger
Puma is an open source multi-threaded application server for Ruby, written by Evan Phoenix and based on Mongrel. We often get questions about how Puma differs from Phusion Passenger. Here are some differences:
Puma is completely open source and free. Phusion Passenger is also open source and free, but there is a commercial (paid) version - Phusion Passenger Enterprise - which provides more features as well as commercial support. Selling Phusion Passenger Enterprise is our means to sponsor the continued development of both the open source and the Enterprise version.
Puma is multithreaded-only. The open source variant of Phusion Passenger is multi-process single-threaded. The Enterprise variant can be configured to be either single-threaded or multithreaded.
Multithreading allows less memory usage and provides higher concurrency than multi-process single-threading. Multithreading is especially suitable for applications that require high I/O concurrency, e.g. applications that perform a lot of HTTP API calls or otherwise block on I/O, or applications which serve WebSockets.
Both Puma and Phusion Passenger Enterprise can be hybrid multi-process multi-threaded. That is, running multiple multithreaded processes. Hybrid mode allows Ruby and Python, which despite having a Global Interpreter Lock, to fully utilize all CPU cores. In Puma, the hybrid mode is called "clustered".
Where Puma and Phusion Passenger Enterprise differ are the threading model. Puma has a dynamic thread pool. Phusion Passenger Enterprise's thread pool is optimized for performance, and is therefore static. It was decided early in the implementation process that managing threads dynamically, especially in Ruby, involves too much overhead and makes the code more complicated. However in some situations you may find a dynamic thread pool more suitable.
On the other hand, Phusion Passenger Enterprise has a dynamic process pool that can be changed on-the-fly, while Puma in clustering ode has a static process pool.
Puma provides a control server which allows you to stop or restart Puma and to query its status. However, it appears to be rather minimalistic (it only displays the size of the backlog and the number of running requests), and doesn't appear to be a first-class citizen. For example, the control server does not work in clustered mode.
Phusion Passenger, both the open source and Enterprise variants, have management tools that provide much more insight. Phusion Passenger allows you to stop, restart and to query its status through command line tools like passenger-status
, passenger-config
, passenger-memory-stats
. These tools are regular command line tools, and their access can be controlled through sudo
, which is a very Unix way of doing things. These tools display everything Puma's status server displays, plus the exact requests that are currently running, how long they've been running, the application's CPU and memory usage, etc.
Phusion Passenger supports two deployment models. The most common model is where it integrates directly into the web server. Apps are started together with the web server and stopped together with the web server, and configuration is done through the web server configuration file. Phusion Passenger infers most information from the web server configuration file, to keep configuration to a minimum amount. Normally you only have to set a virtual host, set the document root, set passenger_enabled on
, and you're done.
Puma supports the reverse proxy model. Puma is its own application and listens on a TCP or Unix domain socket for HTTP requests. The administrator is then supposed to connect the front end web server to Puma through a reverse proxy setup. You have to do this for every Puma app.
Puma's deployment model also requires additional configuration for serving static assets through the web server, instead of through Puma. Phusion Passenger handles this sort of stuff automatically.
Puma's reverse proxy model is also supported by Phusion Passenger, in the form of Phusion Passenger Standalone. Phusion Passenger Standalone behaves just like Puma, in that it is its own application, listens on a TCP or Unix domain socket. So for those who prefer the reverse proxy model for architectural reasons, Phusion Passenger can accomodate them too.
Phusion Passenger is designed for multi-app deployment by default. There is no clustered mode to turn on, it just works. This shows in both usage and the management tools. With a single Phusion Passenger install, you can easily deploy multiple apps. With a single set of management tools, you can manage all your apps.
With Puma, you have to manage each app individually, with a different control server per app. Puma does have a tool called Jungle which allows you to manage multiple apps through SysV init scripts or Upstart.
- Jungle can add new apps or remove existing apps on the fly. The open source version of Phusion Passenger does not have such a feature. Phusion Passenger Enterprise offers a similar feature in the form of Flying Passenger.
- Jungle is capable of starting each app as a different user, through the use of the SysV init script functions. This does not set environment variables from bashrc, so if the user expected that to work then it may cause some confusion. Phusion Passenger loads apps through bash by default, and preserves environment variables in bashrc.
- With Phusion Passenger, you only have to configure per-app stuff in the web server config file. No per-app reverse proxy settings. With Jungle, you still need per-app reverse proxy settings, and you need to run a Jungle command in addition to modifying the web server config file.
- Jungle's app removal is currently easier to use than Flying Passenger's (Phusion Passenger 4.0.5). With Flying Passenger you currently have to go through some hoops to remove an app. We intend to address this in a future release.
- Jungle does not appear to be able to modify the number of processes for an app on-the-fly. Phusion Passenger Enterprise with Flying Passenger can do that.
It should be noted that the SysV init script version of Jungle manages Puma, but does not supervise/monitor Puma, as in watching whether the process has quit, watching memory usage, watching CPU usage, etc. SysV init scripts don't monitor anything and are merely a wrapper interface around running certain commands, so Jungle is a tool for managing the startup and stopping of multiple Puma apps. As an example, use the SysV init version of Jungle to start a Puma, and then kill the master process. Notice that the OS does not restart Puma.
Phusion Passenger restarts all crashed processes. Phusion Passenger even restarts itself if it crashes, thanks to its watchdog architecture.
Unlike SysV init, Upstart is a monitoring/supervision system that can restart a process if it crashes. The Upstart version of Jungle restarts Puma if it crashes. There are still differences between Jungle-Upstart and Phusion Passenger though:
- Jungle-Upstart starts all apps as the same user. There does not appear to be a way to configure this on a per-app basis. Phusion Passenger can handle different-user-per-app just fine. We actually recommend different-user-per-app for security reasons, as we've described in our PivotalLabs talk Securing Ruby apps at the OS level.
- Like Jungle-SysVinit, it does not set environment variables from bashrc.
- Jungle-Upstart is not capable of on-the-fly adding new Puma apps, or removing existing Puma apps. The list of apps is static, and they must all be started and stopped at the same time.
Performance characteristics depends on the workload, so this should be explained in two parts.
For CPU-bound, fast requests that don't involve blocking I/O, Puma and Phusion Passenger (both the open source and Enterprise variant) perform similarly in production, but differently in microbenchmarks. In microbenchmarks Puma is faster because in Phusion Passenger, all data goes through an additional process, the PassengerHelperAgent, which sanitizes request headers, coordinates process spawning, collects statistics, etc. The overhead is not big, approximately a little more than an extra read()
/write()
call to the kernel. But in microbenchmarks where you are benchmarking how quickly the app can do nothing, Phusion Passenger will appear to be twice as slow because of the extra proxy layer. On the other hand, that extra proxy layer is what allows us to provide accurate statistics and to implement robust process coordination, so it's not there for nothing. But we have some ideas on how to address even this in the future.
-
Against Phusion Passenger open source
For slow requests that are bound by blocking I/O, Puma achieves higher concurrency than the open source version of Phusion Passenger. Note that concurrency is not the same as performance. Because of the limited amount of concurrency provided by the multi-process model, Phusion Passenger cannot achieve a high throughput for these kinds of workloads because it will spend a lot of time waiting for the kernel to give it more data. While waiting, it does not use CPU, so your system is being underutilized. You can offset this by spawning more processes, but that requires quite some memory. -
Against Phusion Passenger Enterprise
Puma and Phusion Passenger Enterprise can achieve the same concurrency and the same performance in production. Performance in microbenchmarks is still different, as explained above.
The open source version provides tools for debugging stuck applications by displaying all threads' backtraces, while Puma does not appear to have such functionality. Phusion Passenger Enterprise provides a live IRB console that you can attach to any live, running process for inspection. It also provides ruby-debug integration that you can use even in multi-process mode. In Puma, the debugger is only available in non-clustered mode.
Phusion Passenger, both the open source and Enterprise variant, imposes a time limit on the starting and stopping of the application. Puma does not impose any time limit. If your application is stuck during startup or shutdown (database problem, filesystem problem, network problem, or just a bug) then you will have to find that out manually by seeing that you get Bad Gateway on your requests. Phusion Passenger logs the problem.
If your application throws an error during startup, Puma just keeps trying infinitely, using 100% CPU, while Phusion Passenger only tries once per request. This can be problematic on a redeploy: if you have a startup bug that you did not catch during development or staging, then on a redeploy your site is down (and uses 100% CPU) until you fix it. On Phusion Passenger Enterprise, you can configure it not to retry until the next deploy. It can even hold on to all of its former application processes, to avoid visitors from noticing a problem (the Deployment Error Resistance feature).
Phusion Passenger Enterprise provides features for limiting the request time and memory usage of applications. This is useful if the application - or one of the components it interacts with - has bugs which can cause it to become stuck or use a lot of memory. Unlike many memory management tools, Phusion Passenger Enterprise's memory limiting feature is graceful, so the application only shuts down after it has finished its request, preventing clients from noticing a problem.
Puma has no such features.
Phusion Passenger, both the open source and Enterprise variant, provides out-of-band garbage collection, even in multithreaded mode. Although the GC problem is reduced when using Rubinius or JRuby, most people are still on MRI, and out-of-band garbage collection is an excellent way to reduce latency caused by the GC.
Puma has no support for out-of-band garbage collection.
Phusion Passenger is a polyglot, multi-application server. It supports Python and Node.js. Puma is Ruby-only.
Puma is interesting technology. Compared to the open source version of Phusion Passenger, there are both advantages and disadvantages. Compared to the Enterprise version of Phusion Passenger, we believe Phusion Passenger Enterprise has the upper hand, with the exception of the dynamic thread pool.
If having a price tag is not a problem, then Phusion Passenger Enterprise is an excellent choice. It provides a ton of stability, robustness, ease of use and insight into a single package, that under the hood still follows the Unix bunch-of-simple-components-working-together philosophy.
Is being free is your most important concern, then depending on your requirements, Puma or Phusion Passenger open source may be better for you.