mongoid-multitenancy adds the ability to scope Mongoid models to a tenant in a shared database strategy. Tenants are represented by a tenant model, such as Client
. mongoid-multitenancy will help you set the current tenant on each request and ensures all 'tenant models' are always properly scoped to the current tenant: when viewing, searching and creating.
It is directly inspired by the acts_as_tenant gem for Active Record.
In addition, mongoid-multitenancy:
- allows you to set the current tenant
- redefines some mongoid functions like
index
,validates_with
anddelete_all
to take in account the multitenancy - allows shared items between the tenants
- allows you to define an immutable tenant field once it is persisted
- is thread safe.
Add this line to your application's Gemfile:
gem 'mongoid-multitenancy'
And then execute:
$ bundle
Or install it yourself as:
$ gem install mongoid-multitenancy
There are two steps in adding multi-tenancy to your app with acts_as_tenant:
- setting the current tenant and
- scoping your models.
There are two ways to set the current tenant: (1) by setting the current tenant manually, or (2) by setting the current tenant for a block.
Setting the current tenant in a controller, manually
Mongoid::Multitenancy.current_tenant = client_instance
Setting the current_tenant yourself requires you to use a before_filter to set the Mongoid::Multitenancy.current_tenant variable.
Setting the current tenant for a block
Mongoid::Multitenancy.with_tenant(client_instance) do
# Current tenant is set for all code in this block
end
This approach is useful when running background processes for a specified tenant. For example, by putting this in your worker's run method, any code in this block will be scoped to the current tenant. All methods that set the current tenant are thread safe.
Note: If the current tenant is not set by one of these methods, mongoid-multitenancy will apply a global scope to your models, not related to any tenant. So make sure you use one of the two methods to tell mongoid-multitenancy about the current tenant.
class Client
include Mongoid::Document
field :name, :type => String
validates_uniqueness_of :name
end
class Article
include Mongoid::Document
include Mongoid::Multitenancy::Document
tenant(:client)
field :title, :type => String
end
Adding tenant
to your model declaration will scope that model to the current tenant BUT ONLY if a current tenant has been set.
The association passed to the tenant
function must be valid.
tenant
accepts several options:
- :optional : set to true when the tenant is optional (default value is
false
) - :immutable : set to true when the tenant field is immutable (default value is
true
) - :class_name, etc. : all the other options will be passed to the mongoid relation
Some examples to illustrate this behavior:
# This manually sets the current tenant for testing purposes. In your app this is handled by the gem.
Mongoid::Multitenancy.current_tenant = Client.find_by(:name => 'Perfect Memory') # => <#Client _id:50ca04b86c82bfc125000025, :name: "Perfect Memory">
# All searches are scoped by the tenant, the following searches will only return objects belonging to the current client.
Article.all # => all articles where client_id => 50ca04b86c82bfc125000025
# New objects are scoped to the current tenant
Article.new(:title => 'New blog') # => <#Article _id: nil, title: 'New blog', :client_id: 50ca04b86c82bfc125000025>
# It can make the tenant field immutable once it is persisted to avoid inconsistency
article.persisted? # => true
article.client = another_client
article.valid? # => false
Optional tenant
When setting an optional tenant, for example to allow shared instances between all the tenants, the default scope will return both the tenant and the free-tenant items. That means that using Article.delete_all
or Article.destroy_all
will remove the shared items too.
Article.all # => all articles where client_id.in [50ca04b86c82bfc125000025, nil]
If you are using Rails, you may want to set the current tenant at each request.
Manually set the current tenant in ApplicationController using the host request
class ApplicationController < ActionController::Base
before_filter :set_current_client
def set_current_client
current_client = Client.find_by_host(request.host)
Mongoid::Multitenancy.current_tenant = current_client
end
end
Setting the current_tenant yourself requires you to use a before_filter to set the Mongoid::Multitenancy.current_tenant variable.
mongoid-multitenancy will automatically add the tenant foreign key in the scope list for each of uniqueness validators in order to avoid to redefine all your validators.
class Article
include Mongoid::Document
include Mongoid::Multitenancy::Document
tenant(:client)
field :slug
validates_uniqueness_of :slug # => :scope => client_id is added automatically
end
mongoid-multitenancy automatically adds the tenant foreign key in all your mongoid indexes to avoid to redefine all your validators. If you prefer to define manually the indexes, you can use the option full_indexes: false
.
To create a single index on the tenant field, you can use the option index: true
like any belongs_to
declaration (false by default)
On the example below, only one indexe will be created:
- { 'title_id' => 1, 'client_id' => 1 }
class Article
include Mongoid::Document
include Mongoid::Multitenancy::Document
tenant :client, full_indexes: true
field :title
index({ :title => 1 })
end
On the example below, 2 indexes will be created:
- { 'client_id' => 1 }
- { 'title_id' => 1 }
class Article
include Mongoid::Document
include Mongoid::Multitenancy::Document
tenant :client, index: true
field :title
index({ :title => 1 })
end
mongoid-multitenancy is written by Aymeric Brisse, from Perfect Memory.
- Fork it
- Create your feature branch (
git checkout -b my-new-feature
) - Commit your changes (
git commit -am 'Added some feature'
) - Push to the branch (
git push origin my-new-feature
) - Create new Pull Request