description |
---|
This articles lists several existing multi-tenant applications and describes their implementations. |
Multi-tenancy refers to a principle in software architecture where a single instance of the software runs on a server, serving multiple client organizations (referred to as tenants).
Here are some examples of existing multi-tenant applications and descriptions of their implementations.
- Slack implements a login screen that asks for an email.
- The email is then mapped to a Slack account:
https://{account}.slack.com
. - Slack likely maintains a mapping table between email (or email domain) and the Slack subdomain. Notice that in the example, when entering
[email protected]
, the user is redirected tohttps://auth0.slack.com
. - Slack only supports email/password authentication. Asking for an email first is a good approach for an enterprise SSO. Each organization will have a configured identity provider (SAML, ADFS, etc.). Slack will redirect to this provider once the user has entered their email, instead of asking them for credentials.
- With Slack, a user can belong to multiple teams or organizations. Each team can be considered a tenant.
- You can switch from one team to another through an option in Slack.
- Users can have either full or single-channel access within a team.
- A user can have full access on one team but only single-channel access on another.
The User Profile
below captures the intent of the requirements described above:
{
"email": "[email protected]",
"app_metadata": {
"permissions": {
"auth0": {
"role": "admin",
"channels": ["*"]
},
"another-company": {
"role": "single-channel",
"channels": [ "some-channel" ]
}
}
}
}
- Dropbox asks for an email. If there is an organization with a domain that matches the email suffix, Dropbox will hide the password textbox and display a Single Sign On Enabled message on the login screen.
- When the user clicks on Continue, they will be redirected to the configured identity provider.
- If there is no SSO configured for that domain, the user will enter their password.
- Once a user is authenticated, they are granted access to the files they own and those that were shared with them.
- Users are granted access to their personal account and to any organizations they belong to.
{
"email": "[email protected]",
"app_metadata": {
"permissions": {
"personal": {
"role": "full",
},
"company1": {
"role": "user",
},
"company2": {
"role": "admin",
}
}
}
}
Storing all of the folders that the user has access to as part of the user profile would be unwieldy. The user object would get very large if the user has many folders. A more manageable approach is to store information at the company level role. All other ACLs are handled at the application or database level.
- Auth0 has a single Dashboard for all tenants.
- Auth0 supports Google, GitHub, Live and user/password authentication.
- Auth0 also supports Enterprise connections for Platinum level subscriptions. You can request this by opening a support ticket.
- Auth0 uses email domains for home realm discovery (see screen below), making it similar to the Dropbox experience.
- A user can belong to multiple tenants and have different permissions on each (user foo can be an admin on tenant bar and a regular user of tenant xyz).
- This is implemented by assigning a
user_id
property to an account-level entity if the user has access to everything or an app-level entity if the user has only app level permission.
{
"email": "[email protected]",
"app_metadata": {
"permissions": {
"company1": {
"role": "full",
"clients": [ "*" ]
},
"company2": {
"role": "app-owner",
"clients": [ "iaeonoemaoiy2ie029je" ]
}
}
}
}
A typical modern SaaS multi-tenant app has these features:
- Allows users to signup with a custom username/password, especially during trial periods.
- Allows users to login with their existing Google, LinkedIn or Yahoo! credentials.
- Small businesses with employee directories based on Office365 or Google Apps prefer to login using these credentials.
- Larger companies often have identity systems for employees. Many companies rely on Active Directory or LDAP while some have deployed systems for identity federation based on ADFS, PingFederate, Okta, OneLogin, CA Siteminder, or even a custom SAML-P provider.
- The app is often a web app built as a Single Page App (SPA) using AngularJS with a backend API built with nodejs. A mobile app for Android and iOS with a subset of the web app functionality may also be available.
A single database connection is often sufficient. Whether or not a user has access to a certain tenant can be handled with metadata instead of separate database connections.
If you need to isolate a set of users (e.g. staging vs. prod environment), it may make sense to use different database connections. Even then, it would be better to create different accounts in Auth0.
You may require a separate database connection if, for example, tenant-A uses the built-in Auth0 user store but tenant-B has a set of users elsewhere that you want to authenticate. In this case, you could create a custom db connection for tenant-B and reference that association in your application.
One account for all tenants is simpler and allows you to manage them in one place.
Only if you want to share access to the dashboard with tenants would a separate Auth0 account per tenant be required. But to do so would require you to leverage the restricted API to create each new account. However, you can use the regular API to add applications and connections.
NOTE: You can find a sample multi-tenant app where each tenant has its own Auth0 account on Github.
You can use metadata to handle roles per tenant, as mentioned above.
Metadata in Auth0's user profile is a generic way of associating information to any authenticated user. Permissions, Groups, and Roles are all special cases of these attributes.
Here is an example that would model a simplified chat system like Slack:
{
"email": "[email protected]",
"app_metadata": {
"permissions": {
"auth0": {
"role": "admin",
"channels": ["*"]
},
"another-company": {
"role": "single-channel",
"channels": [ "some-channel" ]
}
}
}
}