python sqlalchemy rls integration package for psql
Project description
rls
Adds PostgreSQL row-level security (RLS) support to your Python application by extending sqlalchemy and alembic.
Installation
pip install rls
Usage
Defining Policies
Attach __rls_policies__ to any SQLAlchemy model to declare which RLS policies should apply to it. For a full working example see test/models.py.
class User(Base):
__tablename__ = "users"
id = Column(Integer, primary_key=True, index=True)
username = Column(String, unique=True, index=True)
__rls_policies__ = [
Permissive(
condition_args=[
ConditionArg(comparator_name="account_id", type=Integer),
],
cmd=[Command.select, Command.update],
custom_expr=lambda x: column("id") == x,
custom_policy_name="equal_to_accountId_policy",
),
]
class Item(Base):
__tablename__ = "items"
id = Column(Integer, primary_key=True, index=True)
title = Column(String, index=True)
description = Column(String)
owner_id = Column(Integer, ForeignKey("users.id", ondelete="CASCADE"))
owner = relationship("User")
__rls_policies__ = [
Permissive(
condition_args=[
ConditionArg(comparator_name="account_id", type=Integer),
],
cmd=[Command.select, Command.update],
custom_expr=lambda x: column("owner_id") == x,
custom_policy_name="equal_to_accountId_policy",
),
Permissive(
condition_args=[
ConditionArg(comparator_name="account_id", type=Integer),
],
cmd=[Command.select],
custom_expr=lambda x: column("owner_id") > x,
custom_policy_name="greater_than_accountId_policy",
),
Permissive(
condition_args=[
ConditionArg(comparator_name="account_id", type=Integer),
],
cmd=[Command.all],
custom_expr=lambda x: column("owner_id") <= x,
custom_policy_name="smaller_than_or_equal_accountId_policy",
),
]
ConditionArg
ConditionArg describes a variable that will be set on the PostgreSQL session before a query runs, allowing the policy expression to reference it.
comparator_name: the PostgreSQL session variable nametype: the SQLAlchemy type of the variable
from sqlalchemy import Integer
ConditionArg(comparator_name="account_id", type=Integer)
Commands
Command is an enum of SQL operations a policy can target:
| Value | Applies to |
|---|---|
select |
SELECT |
insert |
INSERT |
update |
UPDATE |
delete |
DELETE |
all |
all of the above |
Expressions
Policy expressions are lambdas that receive the ConditionArg value(s) and return a SQLAlchemy boolean expression. For example:
from sqlalchemy import column
lambda x: column("owner_id") == x
This restricts rows to those whose owner_id matches the value of account_id supplied in the session context.
Alembic
RLS policies are stored as SQLAlchemy metadata and managed through Alembic migrations. In your env.py, replace the standard metadata assignment:
target_metadata = Base.metadata
with the RLS wrapper:
from rls.alembic_rls import set_metadata_info
target_metadata = set_metadata_info(Base).metadata
Then generate a revision and run alembic upgrade head as normal — the policies will be created or dropped automatically.
For details on the custom Alembic operations used internally, see the alembic docs.
Using Policies at Runtime
Policies are enforced through RlsSession, a drop-in replacement for SQLAlchemy's Session. You supply a Pydantic context object whose fields match the comparator_name values in your policies, plus a bound engine:
class MyContext(BaseModel):
account_id: int
provider_id: int
context = MyContext(account_id=1, provider_id=2)
session = RlsSession(context=context, bind=engine)
res = session.execute(text("SELECT * FROM users")).fetchall()
# Temporarily bypass RLS with a context manager
with session.bypass_rls() as session:
res2 = session.execute(text("SELECT * FROM items")).fetchall()
RlsSessioner
For applications that build a session per request or operation, RlsSessioner wraps a sessionmaker and a ContextGetter to produce ready-to-use RlsSession instances.
sessionmaker: a SQLAlchemysessionmakerconfigured withclass_=RlsSessioncontext_getter: a subclass ofContextGetterthat constructs the context object from arbitraryargs/kwargs
from sqlalchemy.orm import sessionmaker
from rls.rls_session import RlsSession
from rls.rls_sessioner import RlsSessioner, ContextGetter
from pydantic import BaseModel
from test.engines import sync_engine as engine
from sqlalchemy import text
class ExampleContext(BaseModel):
account_id: int
provider_id: int
class ExampleContextGetter(ContextGetter):
def get_context(self, *args, **kwargs) -> ExampleContext:
return ExampleContext(
account_id=kwargs.get("account_id", 1),
provider_id=kwargs.get("provider_id", 2),
)
session_maker = sessionmaker(
class_=RlsSession, autoflush=False, autocommit=False, bind=engine
)
my_sessioner = RlsSessioner(sessionmaker=session_maker, context_getter=ExampleContextGetter())
with my_sessioner(account_id=22, provider_id=99) as session:
res = session.execute(text("SELECT * FROM users")).fetchall()
print(res) # users scoped to account_id=22, provider_id=99
with my_sessioner(account_id=11, provider_id=44) as session:
res = session.execute(text("SELECT * FROM users")).fetchall()
print(res) # users scoped to account_id=11, provider_id=44
Frameworks
FastAPI
In a FastAPI application every endpoint that touches the database needs the correct RLS context derived from the incoming request (e.g. the authenticated user's account_id). Without a structured approach it is easy for individual routes to set the context inconsistently, or to forget to set it at all.
fastapi_dependency_function wraps an RlsSessioner as a FastAPI dependency so that the RLS context is populated from the request automatically and uniformly for every endpoint that declares it. The session injected into the handler already has the correct PostgreSQL session variables set, ensuring all queries are transparently scoped to the caller's data without any per-endpoint boilerplate.
For a complete runnable example see test/fastapi_sample.py.
LICENSE
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distributions
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file rls-0.3.2-py3-none-any.whl.
File metadata
- Download URL: rls-0.3.2-py3-none-any.whl
- Upload date:
- Size: 15.2 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
f4c3cf00cde43ec7659afefb7a18cda826f61eac82315f7c437c4d1a75647172
|
|
| MD5 |
fcca5f0195a83b3dd567ecf620023102
|
|
| BLAKE2b-256 |
0cdfd026900993e890069bc38691c579e4428dff5d9cc0fb97dfcb87a9a088bc
|
Provenance
The following attestation bundles were made for rls-0.3.2-py3-none-any.whl:
Publisher:
release.yml on DelfinaCare/rls
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
rls-0.3.2-py3-none-any.whl -
Subject digest:
f4c3cf00cde43ec7659afefb7a18cda826f61eac82315f7c437c4d1a75647172 - Sigstore transparency entry: 1373239411
- Sigstore integration time:
-
Permalink:
DelfinaCare/rls@2739b07103dbe004e3dea4045c11ddbaae734462 -
Branch / Tag:
refs/tags/v0.3.2 - Owner: https://github.com/DelfinaCare
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@2739b07103dbe004e3dea4045c11ddbaae734462 -
Trigger Event:
release
-
Statement type: