Multitenancy
Refine's architecture allows you to customize your app's data providers, access control and routing to support multi tenant features easily. This guide will provide you with a high level overview of the concepts and how to implement them. To see multi tenant app examples, check out the Examples section.
What is Multitenancy?
Multitenancy, especially in cloud-based systems or software solutions, refers to the ability of a software application or system to serve multiple customers (tenants) simultaneously. While these customers share the same infrastructure and codebase, their data remains separate, and each customer has exclusive access to their own data.
Benefits of Multitenancy:
- Resource Sharing: Efficient use of shared infrastructure reduces costs.
- Cost Savings: Lower maintenance costs passed on to customers.
- Customization: Each tenant can adjust settings to their needs.
- Easy Updates: System-wide updates benefit all tenants at once.
Use Cases:
- Cloud Office Tools: Multiple organizations share document management and collaboration features.
- CRM Systems: Businesses manage customer interactions on a shared platform with secure, custom configurations.
- ERP Systems: Companies use shared ERP solutions with separate data and configurations.
- E-commerce Platforms: Sellers run personalized storefronts on a shared backend.
- LMS Platforms: Schools and organizations deliver courses on a shared learning system.
Implementing Multitenancy in Refine
In the next sections, we'll show you how to set up multitenancy in Refine using a route-based approach. We'll use the multitenancyProvider
from the "@refinedev/enterprise"
package. This Multi-Tenancy Provider is part of the Refine Enterprise Edition. It makes managing multi-tenant applications easier by providing tools like context, hooks, and components that are designed to handle tenants.
1. Setting up the Multitenancy Provider
First, we need to install the @refinedev/enterprise
and @refinedev/multitenancy
packages.
- npm
- pnpm
- yarn
npm i @refinedev/enterprise @refinedev/multitenancy
pnpm add @refinedev/enterprise @refinedev/multitenancy
yarn add @refinedev/enterprise @refinedev/multitenancy
Then we need to change <Refine />
component to <RefineEnterprise />
in your App.tsx
file. You can use same props of <Refine />
component in <RefineEnterprise />
component.
🚨 All the props of the
<Refine />
component are also available in the<RefineEnterprise />
component with additional features.
- import { Refine } from "@refinedev/core";
+ import { RefineEnterprise } from "@refinedev/enterprise";
export const App = () => {
return (
- <Refine>
+ <RefineEnterprise>
{/* Your app code */}
+ </RefineEnterprise>
- </Refine>
);
};
After that, we need to provide the multitenancyProvider
to the <RefineEnterprise />
component. The multitenancyProvider
prop accepts an object with two properties: adapter
and fetchTenants
.
adapter
: The adapter is a function that extracts the tenantId from the current route. You can use the provideduseRouterAdapter
or create your own custom adapter.fetchTenants
: This function is used to fetch the list of tenants. You can fetch the list of tenants from your API and return them in the format{ tenants: Tenant[], defaultTenant: Tenant }
.<WithTenant />
: This component is required to wrap your app code. It fetchestenants
, handling the loading state and error state.fallback
: You can provide a custom fallback component to be displayed while the tenant is not available.loadingComponent
: You can provide a custom loading component to be displayed while the tenant is loading.
When you mount <RefineEnterprise />
and <WithTenant />
components and provide the multitenancyProvider
prop, Refine will automatically extract the tenantId
from the route and pass it to the data provider in the meta
object.
import { RefineEnterprise } from "@refinedev/enterprise";
import { useRouterAdapter, WithTenant } from "@refinedev/multitenancy";
// ... other imports
const App = () => {
return (
<RefineEnterprise
// ... other props
multitenancyProvider={{
adapter: useRouterAdapter(),
fetchTenants: async () => {
const response = await dataProvider(API_URL).getList<ICategory>({
resource: "categories",
pagination: {
mode: "off",
},
});
const tenants = response.data;
const defaultTenant = tenants[0];
return {
tenants,
defaultTenant,
};
},
}}
>
<WithTenant
fallback={<div>Tenant not found</div>}
loadingComponent={<div>Loading...</div>}
>
{/* Your app code */}
</WithTenant>
</RefineEnterprise>
);
};
2. Configuring Multi-tenant Routes
We'll be using routes to determine which tenant is being selected. Once we've setup our routes, useRouterAdapter
will automatically extract the tenantId
from the route.
Note: In the examples below, we are only showing the route definitions. You may need additional code to implement styling and layout depending on your choice of UI library. Regardless of the UI library you choose, the routing implementation will be similar to the examples below.
- React Router Dom
- Next.js
- Remix
React Router Dom
Code Example
// file: App.tsx import { RefineEnterprise } from "@refinedev/enterprise"; import { WithTenant } from "@refinedev/multitenancy"; import dataProvider from "@refinedev/simple-rest"; import routerProvider from "@refinedev/react-router-v6"; import { BrowserRouter, Outlet, Routes, Route } from "react-router"; import { multitenancyProvider } from "./multitenancy"; import { ProductsList, ProductsCreate, ProductsShow, ProductsEdit } from "./products"; export const App: React.FC = () => { return ( <BrowserRouter> <RefineEnterprise multitenancyProvider={multitenancyProvider} dataProvider={dataProvider("<API_URL>")} routerProvider={routerProvider} resources={[ { name: "products", // We're prefixing the routes with `/:tenantId` to make them tenant-aware. list: "/:tenantId/products", show: "/:tenantId/products/:id", edit: "/:tenantId/products/:id/edit", create: "/:tenantId/products/create", }, ]} > <Routes> {/* We're defining the `tenantId` as a route parameter. */} <Route path="/:tenantId" element={ <WithTenant fallback={<div>Tenant not found</div>} loadingComponent={<div>Loading...</div>} > <Outlet /> </WithTenant> }> <Route path="products" element={<ProductsList />} /> <Route path="products/create" element={<ProductsCreate />} /> <Route path="products/:id" element={<ProductsShow />} /> <Route path="products/:id/edit" element={<ProductsEdit />} /> </Route> </Routes> </RefineEnterprise> </BrowserRouter> ); };
// file: multitenancy.ts import type { MultiTenancyProvider } from "@refinedev/core"; import { useRouterAdapter } from "@refinedev/multitenancy"; import dataProvider from "@refinedev/simple-rest"; export type Tenant = { id: string; name: string; }; export const multitenancyProvider: MultiTenancyProvider = { adapter: useRouterAdapter(), fetchTenants: async () => { const response = await dataProvider("<API_URL>").getList<Tenant>({ resource: "categories", pagination: { mode: "off", }, }); const tenants = response.data; const defaultTenant = tenants[0]; return { tenants, defaultTenant, }; }, };
Next.js
Code Example
// file: /pages/_app.tsx import React from "react"; import { RefineEnterprise } from "@refinedev/enterprise"; import { WithTenant } from "@refinedev/multitenancy"; import routerProvider from "@refinedev/nextjs-router/pages"; import dataProvider from "@refinedev/simple-rest"; import type { AppProps } from "next/app"; import { multitenancyProvider } from "./providers/multitenancy"; function App({ Component, pageProps }: AppProps) { return ( <RefineEnterprise multitenancyProvider={multitenancyProvider} dataProvider={dataProvider("<API_URL>")} routerProvider={routerProvider} resources={[ { name: "products", // We're prefixing the routes with `/:tenantId` to make them tenant-aware. list: "/:tenantId/products", show: "/:tenantId/products/:id", edit: "/:tenantId/products/:id/edit", create: "/:tenantId/products/create", }, ]} > <WithTenant fallback={<div>Tenant not found</div>} loadingComponent={<div>Loading...</div>} > <Component {...pageProps} /> </WithTenant> </RefineEnterprise> ); } export default App;
// file: /pages/[tenantId]/products/index.tsx import React from "react"; import { useList } from "@refinedev/core"; export default function ProductsList() { const { data, isLoading } = useList(); if (isLoading) { return <div>Loading...</div>; } return ( <div> <h1>Products</h1> <ul> {data?.data.map((record) => ( <li key={record.id}>{record.name}</li> ))} </ul> </div> ); }
// file: /pages/[tenantId]/products/create.tsx import React from "react"; import { useCreate } from "@refinedev/core"; export default function ProductsCreate() { const { onFinish } = useForm(); return ( <div> <h1>Create Product</h1> <form onSubmit={(event) => { /* ... */ }}> <label htmlFor="name">Name</label> <input id="name" type="text" name="name" /> <button type="submit">Create</button> </form> </div> ); }
// file: /pages/[tenantId]/products/[id]/index.tsx import React from "react"; import { useShow } from "@refinedev/core"; export default function ProductsShow() { const { query: { data, isLoading } } = useShow(); const record = data?.data; if (isLoading) { return <div>Loading...</div>; } return ( <div> <h1>{record?.name}</h1> <p>{record?.description}</p> </div> ); }
// file: /pages/[tenantId]/products/[id]/edit.tsx import React from "react"; import { useForm } from "@refinedev/core"; export default function ProductsEdit() { const { onFinish, query, formLoading } = useForm(); const record = query?.data?.data; if (isLoading) { return <div>Loading...</div>; } return ( <div> <h1>Edit Product</h1> <form onSubmit={(event) => { /* ... */ }}> <label htmlFor="name">Name</label> <input id="name" type="text" name="name" defaultValue={record?.name} /> <button type="submit">Save</button> </form> </div> ); }
// file: /providers/multitenancy.ts import type { MultiTenancyProvider } from "@refinedev/enterprise"; import { useRouterAdapter } from "@refinedev/multitenancy"; import dataProvider from "@refinedev/simple-rest"; export type Tenant = { id: string; name: string; }; export const multitenancyProvider: MultiTenancyProvider = { adapter: useRouterAdapter(), fetchTenants: async () => { const response = await dataProvider("<API_URL>").getList<Tenant>({ resource: "categories", pagination: { mode: "off", }, }); const tenants = response.data; const defaultTenant = tenants[0]; return { tenants, defaultTenant, }; }, };
Remix
Code Example
// file: /app/root.tsx import React from "react"; import { Links, LiveReload, Meta, Outlet, Scripts, ScrollRestoration, } from "@remix-run/react"; import { RefineEnterprise } from "@refinedev/enterprise"; import { WithTenant } from "@refinedev/multitenancy"; import routerProvider from "@refinedev/remix-router"; import dataProvider from "@refinedev/simple-rest"; import { multitenancyProvider } from "./providers/multitenancy"; export default function App() { return ( <html lang="en"> <head> <Meta /> <Links /> </head> <body> <RefineEnterprise multitenancyProvider={multitenancyProvider} dataProvider={dataProvider("<API_URL>")} routerProvider={routerProvider} resources={[ { name: "products", // We're prefixing the routes with `/:tenantId` to make them tenant-aware. list: "/:tenantId/products", show: "/:tenantId/products/:id", edit: "/:tenantId/products/:id/edit", create: "/:tenantId/products/create", }, ]} > <WithTenant fallback={<div>Tenant not found</div>} loadingComponent={<div>Loading...</div>} > <Outlet /> </WithTenant> </RefineEnterprise> <ScrollRestoration /> <Scripts /> <LiveReload /> </body> </html> ); }
// file: /app/routes/$tenantId.products._index.tsx import React from "react"; import { useList } from "@refinedev/core"; export default function ProductsList() { const { data, isLoading } = useList(); if (isLoading) { return <div>Loading...</div>; } return ( <div> <h1>Products</h1> <ul> {data?.data.map((record) => ( <li key={record.id}>{record.name}</li> ))} </ul> </div> ); }
// file: /app/routes/$tenantId.products.create.tsx import React from "react"; import { useCreate } from "@refinedev/core"; export default function ProductsCreate() { const { onFinish } = useForm(); return ( <div> <h1>Create Product</h1> <form onSubmit={(event) => { /* ... */ }}> <label htmlFor="name">Name</label> <input id="name" type="text" name="name" /> <button type="submit">Create</button> </form> </div> ); }
// file: /app/routes/$tenantId.products.$id._index.tsx import React from "react"; import { useShow } from "@refinedev/core"; export default function ProductsShow() { const { query: { data, isLoading }, } = useShow(); const record = data?.data; if (isLoading) { return <div>Loading...</div>; } return ( <div> <h1>{record?.name}</h1> <p>{record?.description}</p> </div> ); }
// file: /app/routes/$tenantId.products.$id.edit.tsx import React from "react"; import { useForm } from "@refinedev/core"; export default function ProductsEdit() { const { onFinish, query, formLoading } = useForm(); const record = query?.data?.data; if (isLoading) { return <div>Loading...</div>; } return ( <div> <h1>Edit Product</h1> <form onSubmit={(event) => { /* ... */ }}> <label htmlFor="name">Name</label> <input id="name" type="text" name="name" defaultValue={record?.name} /> <button type="submit">Save</button> </form> </div> ); }
// file: /app/providers/multitenancy.ts import type { MultiTenancyProvider } from "@refinedev/enterprise"; import { useRouterAdapter } from "@refinedev/multitenancy"; import dataProvider from "@refinedev/simple-rest"; export type Tenant = { id: string; name: string; }; export const multiTenancyProvider: MultiTenancyProvider = { adapter: useRouterAdapter(), fetchTenants: async () => { const response = await dataProvider("<API_URL>").getList<Tenant>({ resource: "categories", pagination: { mode: "off", }, }); const tenants = response.data; const defaultTenant = tenants[0]; return { tenants, defaultTenant, }; }, };
Handling Multi-tenant Requests in Data Providers
We'll be using the tenantId
from the route to determine which tenant is being accessed. Refine will infer the tenantId
from the current route and pass it to the data provider in meta
. You can access the tenantId
from the meta
object in your data provider and use it in your API calls.
To customize the data providers, you can override each method in the data provider instance or use the swizzle
command to be fully able to customize the data provider for your needs.
An example implementation of a custom getList
method is shown below.
import dataProvider from "@refinedev/simple-rest";
const API_URL = "<API_URL>";
const baseDataProvider = dataProvider(API_URL);
const customDataProvider = {
...baseDataProvider,
getList: async ({ resource, filters = [], meta, ...props }) => {
const { tenantId } = meta;
// We're adding the tenantId to the filters
// Your API may have a different way of handling this
if (meta?.tenantId) {
filters.push({
field: "organization",
operator: "eq",
value: meta.tenantId,
});
}
// Call the base data provider's getList method with the updated filters
return baseDataProvider.getList({
resource,
filters,
meta,
...props,
});
},
};
Adding a Tenant Selector to the UI
Now we've defined our routes and data providers to use tenantId
to determine which tenant is being accessed. We'll need to add a tenant selector to the UI to allow users to switch between tenants.
You can use the Tenant selector components from the @refinedev/multitenancy
package to easily add a tenant selector to your app.
- Ant Design
- Material UI
Ant Design
import { TenantSelect } from "@refinedev/multitenancy/antd";
<TenantSelect />;
Material UI
import { TenantSelect } from "@refinedev/multitenancy/mui";
<TenantSelect />;
Examples
Here are some examples of multi-tenant apps built with Refine Enterprise Edition: