Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
11 changes: 5 additions & 6 deletions sqlmodel/main.py
Original file line number Diff line number Diff line change
Expand Up @@ -40,7 +40,6 @@
from sqlalchemy.orm import (
Mapped,
RelationshipProperty,
declared_attr,
registry,
relationship,
)
Expand Down Expand Up @@ -519,6 +518,7 @@ def Relationship(

@__dataclass_transform__(kw_only_default=True, field_descriptors=(Field, FieldInfo))
class SQLModelMetaclass(ModelMetaclass, DeclarativeMeta):
__tablename__: str
__sqlmodel_relationships__: dict[str, RelationshipInfo]
model_config: SQLModelConfig
model_fields: ClassVar[dict[str, FieldInfo]]
Expand Down Expand Up @@ -565,6 +565,10 @@ def __new__(
"__sqlmodel_relationships__": relationships,
"__annotations__": pydantic_annotations,
}
# Set default __tablename__ before class creation so it's part of the
# class dict, unless the user supplied one.
if "__tablename__" not in class_dict:
dict_used["__tablename__"] = name.lower()
# Duplicate logic from Pydantic to filter config kwargs because if they are
# passed directly including the registry Pydantic will pass them over to the
# superclass causing an error
Expand Down Expand Up @@ -804,7 +808,6 @@ def get_column_from_field(field: Any) -> Column:
class SQLModel(BaseModel, metaclass=SQLModelMetaclass, registry=default_registry):
# SQLAlchemy needs to set weakref(s), Pydantic will set the other slots values
__slots__ = ("__weakref__",)
__tablename__: ClassVar[str | Callable[..., str]]
__sqlmodel_relationships__: ClassVar[builtins.dict[str, RelationshipInfo]]
__name__: ClassVar[str]
metadata: ClassVar[MetaData]
Expand Down Expand Up @@ -864,10 +867,6 @@ def __repr_args__(self) -> Sequence[tuple[str | None, Any]]:
if not (isinstance(k, str) and k.startswith("_sa_"))
]

@declared_attr # type: ignore
def __tablename__(cls) -> str:
return cls.__name__.lower()

@classmethod
def model_validate( # ty: ignore[invalid-method-override]
cls: type[_TSQLModel],
Expand Down
106 changes: 106 additions & 0 deletions tests/test_tablename.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,106 @@
from pydantic.alias_generators import to_snake
from sqlalchemy import inspect
from sqlalchemy.orm import declared_attr
from sqlmodel import Field, Session, SQLModel, create_engine, select
from sqlmodel.pool import StaticPool


def _engine():
return create_engine(
"sqlite://", connect_args={"check_same_thread": False}, poolclass=StaticPool
)


def test_default_tablename() -> None:
"""table=True models get __tablename__ = classname.lower() by default."""

class Gadget(SQLModel, table=True):
id: int | None = Field(default=None, primary_key=True)

assert Gadget.__tablename__ == "gadget"

engine = _engine()
SQLModel.metadata.create_all(engine)
assert inspect(engine).has_table("gadget")


def test_explicit_tablename() -> None:
"""An explicit __tablename__ overrides the default."""

class Widget(SQLModel, table=True):
__tablename__ = "custom_widgets"
id: int | None = Field(default=None, primary_key=True)
name: str

assert Widget.__tablename__ == "custom_widgets"

engine = _engine()
SQLModel.metadata.create_all(engine)
assert inspect(engine).has_table("custom_widgets")
assert not inspect(engine).has_table("widget")

with Session(engine) as session:
session.add(Widget(name="sprocket"))
session.commit()

with Session(engine) as session:
row = session.exec(select(Widget)).first()
assert row is not None
assert row.name == "sprocket"


def test_tablename_inheritance_default() -> None:
"""A subclass that is also a table gets its own default __tablename__."""

class BaseThing(SQLModel, table=True):
id: int | None = Field(default=None, primary_key=True)
kind: str = "base"

class SubThing(BaseThing, table=True):
extra: str | None = None

assert BaseThing.__tablename__ == "basething"
assert SubThing.__tablename__ == "subthing"


def test_tablename_inheritance_explicit_child() -> None:
"""A subclass can set its own __tablename__, visible on the class."""

class Vehicle(SQLModel, table=True):
id: int | None = Field(default=None, primary_key=True)
kind: str = ""

class Truck(Vehicle, table=True):
__tablename__ = "trucks"
payload: int | None = None

assert Vehicle.__tablename__ == "vehicle"
assert Truck.__tablename__ == "trucks"


def test_tablename_default_on_plain_model() -> None:
"""Non-table models also get a default __tablename__."""

class Schema(SQLModel):
name: str

assert Schema.__tablename__ == "schema"


def test_tablename_declared_attr_directive() -> None:
"""A dynamically computed __tablename__ via @declared_attr.directive works."""

class FirstWidget(SQLModel, table=True):
id: int | None = Field(default=None, primary_key=True)
name: str

@declared_attr.directive
@classmethod
def __tablename__(cls) -> str:
return to_snake(cls.__name__)

assert FirstWidget.__tablename__ == "first_widget"

engine = _engine()
SQLModel.metadata.create_all(engine)
assert inspect(engine).has_table("first_widget")
Loading