The Feathers Philosophy
We know! You're probably screaming "Not another JavaScript framework!". We've also become frustrated with all the Rails clones and MVC frameworks that don't do anything different. Instead, a few years ago we started to explore a different approach to building web applications using services and cross cutting concerns while also being careful not to reinvent the wheel.
With this experimentation Feathers has grown into what it is today. Our core philosophy that guides Feathers is still the same as it was years ago:
"Monolithic apps tend to fall apart at scale, either because of performance or because there are too many people in the code. What if we could make it easy to build applications that can naturally become service oriented from day one, rather than having to start with a large application and painfully tease it apart?
What if we could make a framework that grows with you and your business and makes it easy for you to transition to a series of microservices, or easily change databases without ripping our code apart?
What if we could make real-time less intimidating rather than a hacky, complex after thought? What if we could remove the boilerplate needed for building REST APIs? Could we build a framework that provides enough structure to get going easily and add all the common pieces that modern apps need, but still keep everything flexible and optional?
A framework itself should not be opinionated. It should be made up of small, reusable, optional components that do one thing well but are combined in an opinionated way. By keeping the components of your application small, flexible and optional you eliminate much of the engineering obstacles that prevent moving fast and scaling."
We strongly believe that your UI, data and business logic are the core of any web or mobile application and your framework should take care of the rest so you can focus on the things that matter.
Services
Many web frameworks focus on things like rendering views, defining routes and handling HTTP requests and responses without providing a structure for implementing application logic separate from those secondary concerns. The result - even when using the MVC pattern - are monolithic applications with messy controllers or fat models. Your actual application logic and how your data is accessed are all mixed up together.
Feathers brings 3 important concepts together that help to separate those concerns from how your application works and give you incredible flexibility while still keeping things DRY.
A service layer which helps to decouple your application logic from how it is being accessed and represented. Besides also making things a lot easier to test - you just call your service methods instead of having to make fake HTTP requests - this is what allows Feathers to provide the same API through both HTTP REST and websockets. It can even be extended to use any other RPC protocol and you won't have to change any of your services.
Uniform Interfaces
Every Feathers service exposes a uniform interface modeled after REST. Where, just like one of the key constraints of REST, your action context is immediately apparent due to the naming convention. With REST you have the HTTP verbs (GET, POST, PUT, PATCH and DELETE). This translates naturally to a Feathers service object interface:
const myService = {
// GET /path
find(params, callback) {},
// GET /path/<id>
get(id, params, callback) {},
// POST /path
create(data, params, callback) {},
// PUT /path/<id>
update(id, data, params, callback) {},
// PATCH /path/<id>
patch(id, data, params, callback) {},
// DELETE /path/<id>
remove(id, params, callback) {}
}
This interface also makes it easier to "hook" into the execution of those methods and emit events when they return which can naturally be used to provide real-time functionality.
Hooks
Cross cutting concerns are an extremely powerful part of aspect oriented programming. They are a very good fit for web and mobile applications since the majority are primarily CRUD applications with lots of shared functionality. Keeping with the Unix philosophy we believe that small modules that do one thing are better than large complex ones. That's why you can create before
and after
hooks and chain them together to create very complex processes while still maintaining modularity and flexibility.
Built on the Shoulders of Giants
Because we utilize some already proven modules, we spend less time re-inventing the wheel, are able to move incredibly fast, and have small well-tested, stable modules.
Here's how we use some of the tech under the hood:
- Feathers extends Express 4, the most popular web framework for NodeJS.
- Our CLI tool uses commander and its generators are built with Yeoman.
- We wrap Socket.io or Primus as your websocket transport.
- Our service adapters typically wrap mature ORMs like mongoose, sequelize or knex.
- npm for package management.
- passport for much of the feathers-authentication work.