Fastadmin
FastAdmin is an easy-to-use Admin Dashboard App for FastAPI/Flask/Django inspired by Django Admin.
FastAdmin is an easy-to-use admin dashboard for FastAPI, Django, and Flask, inspired by Django Admin. The project is written primarily in Python, distributed under the MIT License license, first published in 2023. Key topics include: admin, dashboard, django, fastadmin, fastapi.
Admin Dashboard for FastAPI / Flask / Django
Demo

Introduction
<a href='https://github.com/vsdudakov/fastadmin' target='_blank'>FastAdmin</a> is an easy-to-use admin dashboard for FastAPI, Django, and Flask, inspired by Django Admin.
FastAdmin is built with relationships in mind and admiration for Django Admin. Its design focuses on making it as easy as possible to configure your admin dashboard for FastAPI, Django, or Flask.
FastAdmin aims to be minimal, functional, and familiar.
Getting Started
If you have questions beyond this documentation, feel free to <a href='mailto:vsdudakov@gmail.com' target='_blank'>email us</a>.
Installation
Follow the steps below to set up FastAdmin:
Install the package with pip:
On zsh and macOS, use quotes: <code>pip install 'fastadmin[fastapi,django]'</code>
bashpip install fastadmin[fastapi,django] # FastAPI with Django ORM pip install fastadmin[fastapi,tortoise-orm] # FastAPI with Tortoise ORM pip install fastadmin[fastapi,pony] # FastAPI with Pony ORM pip install fastadmin[fastapi,sqlalchemy] # FastAPI with SQLAlchemy (includes greenlet) pip install fastadmin[django] # Django with Django ORM pip install fastadmin[django,pony] # Django with Pony ORM pip install fastadmin[flask,sqlalchemy] # Flask with SQLAlchemy (includes greenlet)
Or install with Poetry:
bashpoetry add 'fastadmin[fastapi,django]' poetry add 'fastadmin[fastapi,tortoise-orm]' poetry add 'fastadmin[fastapi,pony]' poetry add 'fastadmin[fastapi,sqlalchemy]' poetry add 'fastadmin[django]' poetry add 'fastadmin[django,pony]' poetry add 'fastadmin[flask,sqlalchemy]'
When using SQLAlchemy, the <code>greenlet</code> package is required (included in the <code>fastadmin[sqlalchemy]</code> extra).
Configure the required settings with environment variables:
You can add these variables to a <code>.env</code> file and load them with python-dotenv. See <a href='https://vsdudakov.github.io/fastadmin#settings'>all settings</a> in the full documentation.
bashexport ADMIN_USER_MODEL=User export ADMIN_USER_MODEL_USERNAME_FIELD=username export ADMIN_SECRET_KEY=secret_key
Quick Examples
ORM setup (User, UserAttachment, actions, widgets)
Tortoise ORM
pythonfrom tortoise import fields from tortoise.models import Model class User(Model): username = fields.CharField(max_length=255, unique=True) hash_password = fields.CharField(max_length=255) is_superuser = fields.BooleanField(default=False) is_active = fields.BooleanField(default=True) avatar_url = fields.TextField(null=True) class UserAttachment(Model): user = fields.ForeignKeyField("models.User", related_name="attachments") attachment_url = fields.TextField()
pythonfrom fastadmin import ( TortoiseInlineModelAdmin, TortoiseModelAdmin, WidgetType, action, register, widget_action, ) from fastadmin.models.schemas import ( WidgetActionChartProps, WidgetActionInputSchema, WidgetActionResponseSchema, WidgetActionType, ) from .models import User, UserAttachment class UserAttachmentInline(TortoiseInlineModelAdmin): model = UserAttachment formfield_overrides = { "attachment_url": (WidgetType.UploadFile, {"required": True}), } async def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # save file to media directory or to s3/filestorage here return f"/media/{file_name}" @register(User) class UserAdmin(TortoiseModelAdmin): list_display = ("id", "username", "is_superuser", "is_active") inlines = (UserAttachmentInline,) formfield_overrides = { "avatar_url": (WidgetType.UploadImage, {"required": False}), } actions = ("activate", "deactivate") widget_actions = ("users_chart", "users_list") @action(description="Activate selected users") async def activate(self, ids: list[int]) -> None: await self.model_cls.filter(id__in=ids).update(is_active=True) @action(description="Deactivate selected users") async def deactivate(self, ids: list[int]) -> None: await self.model_cls.filter(id__in=ids).update(is_active=False) async def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # handle avatar_url uploads for User (and other file fields if needed) return f"/media/{file_name}" @widget_action( widget_action_type=WidgetActionType.ChartLine, widget_action_props=WidgetActionChartProps(x_field="x", y_field="y", series_field="series"), tab="Analytics", title="Users over time", ) async def users_chart(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: return WidgetActionResponseSchema( data=[ {"x": "2026-01-01", "y": 10, "series": "Active"}, {"x": "2026-01-02", "y": 15, "series": "Active"}, {"x": "2026-01-01", "y": 3, "series": "Inactive"}, {"x": "2026-01-02", "y": 5, "series": "Inactive"}, ] ) @widget_action( widget_action_type=WidgetActionType.Action, tab="Data", title="Users list", description="Simple action widget that returns a table of users.", ) async def users_list(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: return WidgetActionResponseSchema( data=[ {"id": 1, "username": "alice"}, {"id": 2, "username": "bob"}, ] )
Django ORM
pythonfrom django.db import models class User(models.Model): username = models.CharField(max_length=255, unique=True) password = models.CharField(max_length=255) is_superuser = models.BooleanField(default=False) is_active = models.BooleanField(default=True) avatar_url = models.ImageField(null=True) class UserAttachment(models.Model): user = models.ForeignKey(User, related_name="attachments", on_delete=models.CASCADE) attachment_url = models.FileField()
pythonfrom fastadmin import ( DjangoInlineModelAdmin, DjangoModelAdmin, WidgetType, action, register, widget_action, ) from fastadmin.models.schemas import ( WidgetActionArgumentProps, WidgetActionInputSchema, WidgetActionProps, WidgetActionResponseSchema, WidgetActionType, ) from .models import User, UserAttachment class UserAttachmentInline(DjangoInlineModelAdmin): model = UserAttachment formfield_overrides = { "attachment_url": (WidgetType.UploadFile, {"required": True}), } def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # save file to media directory or to s3/filestorage here return f"/media/{file_name}" @register(User) class UserAdmin(DjangoModelAdmin): list_display = ("id", "username", "is_superuser", "is_active") inlines = (UserAttachmentInline,) formfield_overrides = { "avatar_url": (WidgetType.UploadImage, {"required": False}), } actions = ("activate", "deactivate") widget_actions = ("users_summary", "users_chart") @action(description="Activate selected users") def activate(self, ids): self.model_cls.objects.filter(id__in=ids).update(is_active=True) @action(description="Deactivate selected users") def deactivate(self, ids): self.model_cls.objects.filter(id__in=ids).update(is_active=False) def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # handle avatar_url uploads for User (and other file fields if needed) return f"/media/{file_name}" @widget_action( widget_action_type=WidgetActionType.Action, widget_action_props=WidgetActionProps( arguments=[ WidgetActionArgumentProps( name="only_active", widget_type=WidgetType.Switch, widget_props={"required": False}, ) ] ), tab="Data", title="Users summary", ) def users_summary(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: qs = self.model_cls.objects.filter(is_active=True) if payload.arguments.get("only_active") else self.model_cls.objects.all() return WidgetActionResponseSchema( data=[{"id": u.id, "username": u.username} for u in qs[:5]] ) @widget_action( widget_action_type=WidgetActionType.ChartLine, widget_action_props=WidgetActionChartProps(x_field="label", y_field="value", series_field="series"), tab="Analytics", title="Active vs inactive users", ) def users_chart(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: active = self.model_cls.objects.filter(is_active=True).count() inactive = self.model_cls.objects.filter(is_active=False).count() return WidgetActionResponseSchema( data=[ {"label": "users", "value": active, "series": "active"}, {"label": "users", "value": inactive, "series": "inactive"}, ] )
SQLAlchemy
pythonfrom sqlalchemy import Boolean, ForeignKey, Integer, String, Text from sqlalchemy.ext.asyncio import async_sessionmaker, create_async_engine from sqlalchemy.orm import DeclarativeBase, Mapped, mapped_column, relationship engine = create_async_engine("sqlite+aiosqlite:///:memory:") sessionmaker = async_sessionmaker(engine, expire_on_commit=False) class Base(DeclarativeBase): pass class User(Base): __tablename__ = "user" id: Mapped[int] = mapped_column(Integer, primary_key=True) username: Mapped[str] = mapped_column(String(255), unique=True) password: Mapped[str] = mapped_column(String(255)) is_superuser: Mapped[bool] = mapped_column(Boolean, default=False) is_active: Mapped[bool] = mapped_column(Boolean, default=True) avatar_url: Mapped[str | None] = mapped_column(Text, nullable=True) attachments: Mapped[list["UserAttachment"]] = relationship(back_populates="user") class UserAttachment(Base): __tablename__ = "user_attachment" id: Mapped[int] = mapped_column(Integer, primary_key=True) user_id: Mapped[int] = mapped_column(ForeignKey("user.id")) attachment_url: Mapped[str] = mapped_column(Text) user: Mapped[User] = relationship(back_populates="attachments")
pythonfrom sqlalchemy import update from fastadmin import ( SqlAlchemyInlineModelAdmin, SqlAlchemyModelAdmin, WidgetType, action, register, widget_action, ) from fastadmin.models.schemas import ( WidgetActionChartProps, WidgetActionInputSchema, WidgetActionResponseSchema, WidgetActionType, ) from .models import User, UserAttachment, sessionmaker class UserAttachmentInline(SqlAlchemyInlineModelAdmin): model = UserAttachment formfield_overrides = { "attachment_url": (WidgetType.UploadFile, {"required": True}), } async def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # save file to media directory or to s3/filestorage here return f"/media/{file_name}" @register(User, sqlalchemy_sessionmaker=sessionmaker) class UserAdmin(SqlAlchemyModelAdmin): list_display = ("id", "username", "is_superuser", "is_active") inlines = (UserAttachmentInline,) formfield_overrides = { "avatar_url": (WidgetType.UploadImage, {"required": False}), } actions = ("activate", "deactivate") widget_actions = ("users_chart", "users_list") @action(description="Activate selected users") async def activate(self, ids): sm = self.get_sessionmaker() async with sm() as s: await s.execute(update(User).where(User.id.in_(ids)).values(is_active=True)) await s.commit() @action(description="Deactivate selected users") async def deactivate(self, ids): sm = self.get_sessionmaker() async with sm() as s: await s.execute(update(User).where(User.id.in_(ids)).values(is_active=False)) await s.commit() async def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # handle avatar_url uploads for User (and other file fields if needed) return f"/media/{file_name}" @widget_action( widget_action_type=WidgetActionType.ChartBar, widget_action_props=WidgetActionChartProps(x_field="label", y_field="value", series_field="series"), tab="Analytics", title="Users count", ) async def users_chart(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: return WidgetActionResponseSchema( data=[ {"label": "users", "value": 42, "series": "all"}, ] ) @widget_action( widget_action_type=WidgetActionType.Action, tab="Data", title="Users list", ) async def users_list(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: # In a real app, fetch from the DB; here it's just a static example return WidgetActionResponseSchema( data=[ {"id": 1, "username": "alice"}, {"id": 2, "username": "bob"}, ] )
Pony ORM
pythonfrom pony.orm import Database, LongStr, PrimaryKey, Required, Set db = Database() class User(db.Entity): # type: ignore[misc] _table_ = "user" id = PrimaryKey(int, auto=True) username = Required(str) password = Required(str) is_superuser = Required(bool, default=False) is_active = Required(bool, default=True) avatar_url = Required(LongStr, nullable=True) attachments = Set("UserAttachment") class UserAttachment(db.Entity): # type: ignore[misc] _table_ = "user_attachment" id = PrimaryKey(int, auto=True) user = Required(User) attachment_url = Required(LongStr)
pythonfrom pony.orm import commit, db_session from fastadmin import ( PonyORMInlineModelAdmin, PonyORMModelAdmin, WidgetType, action, register, widget_action, ) from fastadmin.models.schemas import ( WidgetActionInputSchema, WidgetActionResponseSchema, WidgetActionType, ) from .models import User, UserAttachment class UserAttachmentInline(PonyORMInlineModelAdmin): model = UserAttachment formfield_overrides = { "attachment_url": (WidgetType.UploadFile, {"required": True}), } def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # save file to media directory or to s3/filestorage here return f"/media/{file_name}" @register(User) class UserAdmin(PonyORMModelAdmin): list_display = ("id", "username", "is_superuser", "is_active") inlines = (UserAttachmentInline,) formfield_overrides = { "avatar_url": (WidgetType.UploadImage, {"required": False}), } actions = ("activate", "deactivate") widget_actions = ("users_list", "users_chart") @action(description="Activate selected users") @db_session def activate(self, ids): for u in User.select(lambda o: o.id in ids): u.is_active = True commit() @action(description="Deactivate selected users") @db_session def deactivate(self, ids): for u in User.select(lambda o: o.id in ids): u.is_active = False commit() def upload_file(self, field_name: str, file_name: str, file_content: bytes) -> str: # handle avatar_url uploads for User (and other file fields if needed) return f"/media/{file_name}" @widget_action(widget_action_type=WidgetActionType.Action, tab="Data", title="Users list") @db_session def users_list(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: return WidgetActionResponseSchema( data=[{"id": u.id, "username": u.username} for u in User.select()[:5]] ) @widget_action(widget_action_type=WidgetActionType.ChartPie, tab="Analytics", title="Users by activity") @db_session def users_chart(self, payload: WidgetActionInputSchema) -> WidgetActionResponseSchema: active = User.select(lambda u: u.is_active).count() inactive = User.select(lambda u: not u.is_active).count() return WidgetActionResponseSchema( data=[ {"type": "active", "value": active}, {"type": "inactive", "value": inactive}, ] )
Request and user context in admin methods
You can access the current request and authenticated user in your admin methods via self.request and self.user. This works the same way for both ModelAdmin and InlineModelAdmin.
pythonfrom fastadmin import TortoiseModelAdmin, register from .models import Event @register(Event) class EventAdmin(TortoiseModelAdmin): async def has_change_permission(self, user_id: int | None = None) -> bool: # you can either use user_id to load the user from the DB, # or rely on self.user – the current authenticated admin user if self.user and self.user.get("is_superuser"): return True return False async def save_model(self, id: int | None, payload: dict) -> dict: # self.request is the current HTTP request if self.request and getattr(self.request, "client", None): payload["changed_from_ip"] = getattr( self.request.client, "host", None, ) return await super().save_model(id, payload)
Inline admins get the same properties (self.user, self.request), so you can reuse this pattern in inline-specific hooks like save_model or custom action / widget_action methods.
Framework integration (register User admin)
FastAPI
pythonfrom fastapi import FastAPI from fastadmin import fastapi_app as admin_app import myapp.admin # import to register User admin app = FastAPI() app.mount("/admin", admin_app)
Django
pythonfrom django.urls import path from fastadmin import get_django_admin_urls as get_admin_urls from fastadmin.settings import settings import myapp.admin # imports @register(User) urlpatterns = [ path(f"{settings.ADMIN_PREFIX}/", get_admin_urls()), ]
Flask
pythonfrom flask import Flask from fastadmin import flask_app as admin_app from fastadmin.settings import settings import myapp.admin # imports @register(User) app = Flask(__name__) app.register_blueprint(admin_app, url_prefix=f"/{settings.ADMIN_PREFIX}")
Documentation
Full documentation is available at vsdudakov.github.io/fastadmin.
License
This project is licensed under the MIT License — see the LICENSE file for details.
Contributors
Showing top 7 contributors by commit count.
