Skip to content

Commit f993d7e

Browse files
committed
Fix timezone issue with AccessToken created_at
1 parent 65ca16d commit f993d7e

File tree

6 files changed

+32
-10
lines changed

6 files changed

+32
-10
lines changed

fastapi_users_db_sqlalchemy/__init__.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@
2020
from sqlalchemy.orm import joinedload
2121
from sqlalchemy.sql import Select
2222

23-
from fastapi_users_db_sqlalchemy.guid import GUID
23+
from fastapi_users_db_sqlalchemy.generics import GUID
2424

2525
__version__ = "3.0.1"
2626

fastapi_users_db_sqlalchemy/access_token.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,11 @@
22
from typing import Generic, Optional, Type
33

44
from fastapi_users.authentication.strategy.db import A, AccessTokenDatabase
5-
from sqlalchemy import Column, DateTime, ForeignKey, String, delete, select, update
5+
from sqlalchemy import Column, ForeignKey, String, delete, select, update
66
from sqlalchemy.ext.asyncio import AsyncSession
77
from sqlalchemy.ext.declarative import declared_attr
88

9-
from fastapi_users_db_sqlalchemy.guid import GUID
9+
from fastapi_users_db_sqlalchemy.generics import GUID, TIMESTAMPAware
1010

1111

1212
class SQLAlchemyBaseAccessTokenTable:
@@ -15,7 +15,7 @@ class SQLAlchemyBaseAccessTokenTable:
1515
__tablename__ = "accesstoken"
1616

1717
token = Column(String(length=43), primary_key=True)
18-
created_at = Column(DateTime(timezone=True), index=True, nullable=False)
18+
created_at = Column(TIMESTAMPAware(timezone=True), index=True, nullable=False)
1919

2020
@declared_attr
2121
def user_id(cls):

fastapi_users_db_sqlalchemy/guid.py fastapi_users_db_sqlalchemy/generics.py

+22-3
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,21 @@
11
import uuid
2+
from datetime import datetime, timezone
23

34
from pydantic import UUID4
5+
from sqlalchemy import CHAR, TIMESTAMP, TypeDecorator
46
from sqlalchemy.dialects.postgresql import UUID
5-
from sqlalchemy.types import CHAR, TypeDecorator
67

78

89
class GUID(TypeDecorator): # pragma: no cover
9-
"""Platform-independent GUID type.
10+
"""
11+
Platform-independent GUID type.
1012
1113
Uses PostgreSQL's UUID type, otherwise uses
1214
CHAR(36), storing as regular strings.
1315
"""
1416

1517
class UUIDChar(CHAR):
16-
python_type = UUID4
18+
python_type = UUID4 # type: ignore
1719

1820
impl = UUIDChar
1921
cache_ok = True
@@ -42,3 +44,20 @@ def process_result_value(self, value, dialect):
4244
if not isinstance(value, uuid.UUID):
4345
value = uuid.UUID(value)
4446
return value
47+
48+
49+
class TIMESTAMPAware(TypeDecorator): # pragma: no cover
50+
"""
51+
MySQL and SQLite will always return naive-Python datetimes.
52+
53+
We store everything as UTC, but we want to have
54+
only offset-aware Python datetimes, even with MySQL and SQLite.
55+
"""
56+
57+
impl = TIMESTAMP
58+
cache_ok = True
59+
60+
def process_result_value(self, value: datetime, dialect):
61+
if dialect.name != "postgresql":
62+
return value.replace(tzinfo=timezone.utc)
63+
return value

pyproject.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ classifiers = [
3030
description-file = "README.md"
3131
requires-python = ">=3.7"
3232
requires = [
33-
"fastapi-users >= 9.1.0",
33+
"fastapi-users >= 9.1.0,<10.0.0",
3434
"sqlalchemy[asyncio] >=1.4",
3535
]
3636

requirements.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,2 @@
1-
fastapi-users >= 9.1.0
1+
fastapi-users >= 9.1.0,<10.0.0
22
sqlalchemy[asyncio,mypy] >=1.4

tests/test_access_token.py

+4-1
Original file line numberDiff line numberDiff line change
@@ -78,7 +78,10 @@ async def test_queries(
7878

7979
# Update
8080
access_token_db.created_at = datetime.now(timezone.utc)
81-
await sqlalchemy_access_token_db.update(access_token_db)
81+
updated_access_token = await sqlalchemy_access_token_db.update(access_token_db)
82+
assert updated_access_token.created_at.replace(
83+
microsecond=0
84+
) == access_token_db.created_at.replace(microsecond=0)
8285

8386
# Get by token
8487
access_token_by_token = await sqlalchemy_access_token_db.get_by_token(

0 commit comments

Comments
 (0)