mirror of
				https://github.com/fastapi-users/fastapi-users.git
				synced 2025-11-04 06:37:51 +08:00 
			
		
		
		
	* Move users router in sub-module * Factorize UserRouter into EventHandlersRouter * Implement OAuth registration/login router * Apply isort/black * Remove temporary pytest marker * Fix httpx-oauth version in lock file * Ensure ON_AFTER_REGISTER event is triggered on OAuth registration * Add API on FastAPIUsers to generate an OAuth router * Improve test coverage of FastAPIUsers * Small fixes * Write the OAuth documentation * Fix SQL unit-tests by avoiding collisions in SQLite db files
		
			
				
	
	
		
			215 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			215 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
import sqlite3
 | 
						|
from typing import AsyncGenerator
 | 
						|
 | 
						|
import pytest
 | 
						|
import sqlalchemy
 | 
						|
from databases import Database
 | 
						|
from sqlalchemy import Column, String
 | 
						|
from sqlalchemy.ext.declarative import DeclarativeMeta, declarative_base
 | 
						|
 | 
						|
from fastapi_users.db.sqlalchemy import (
 | 
						|
    NotSetOAuthAccountTableError,
 | 
						|
    SQLAlchemyBaseOAuthAccountTable,
 | 
						|
    SQLAlchemyBaseUserTable,
 | 
						|
    SQLAlchemyUserDatabase,
 | 
						|
)
 | 
						|
from fastapi_users.password import get_password_hash
 | 
						|
from tests.conftest import UserDB, UserDBOAuth
 | 
						|
 | 
						|
 | 
						|
@pytest.fixture
 | 
						|
async def sqlalchemy_user_db() -> AsyncGenerator[SQLAlchemyUserDatabase, None]:
 | 
						|
    Base: DeclarativeMeta = declarative_base()
 | 
						|
 | 
						|
    class User(SQLAlchemyBaseUserTable, Base):
 | 
						|
        first_name = Column(String, nullable=True)
 | 
						|
 | 
						|
    DATABASE_URL = "sqlite:///./test-sqlalchemy-user.db"
 | 
						|
    database = Database(DATABASE_URL)
 | 
						|
 | 
						|
    engine = sqlalchemy.create_engine(
 | 
						|
        DATABASE_URL, connect_args={"check_same_thread": False}
 | 
						|
    )
 | 
						|
    Base.metadata.create_all(engine)
 | 
						|
 | 
						|
    await database.connect()
 | 
						|
 | 
						|
    yield SQLAlchemyUserDatabase(UserDB, database, User.__table__)
 | 
						|
 | 
						|
    Base.metadata.drop_all(engine)
 | 
						|
 | 
						|
 | 
						|
@pytest.fixture
 | 
						|
async def sqlalchemy_user_db_oauth() -> AsyncGenerator[SQLAlchemyUserDatabase, None]:
 | 
						|
    Base: DeclarativeMeta = declarative_base()
 | 
						|
 | 
						|
    class User(SQLAlchemyBaseUserTable, Base):
 | 
						|
        first_name = Column(String, nullable=True)
 | 
						|
 | 
						|
    class OAuthAccount(SQLAlchemyBaseOAuthAccountTable, Base):
 | 
						|
        pass
 | 
						|
 | 
						|
    DATABASE_URL = "sqlite:///./test-sqlalchemy-user-oauth.db"
 | 
						|
    database = Database(DATABASE_URL)
 | 
						|
 | 
						|
    engine = sqlalchemy.create_engine(
 | 
						|
        DATABASE_URL, connect_args={"check_same_thread": False}
 | 
						|
    )
 | 
						|
    Base.metadata.create_all(engine)
 | 
						|
 | 
						|
    await database.connect()
 | 
						|
 | 
						|
    yield SQLAlchemyUserDatabase(
 | 
						|
        UserDBOAuth, database, User.__table__, OAuthAccount.__table__
 | 
						|
    )
 | 
						|
 | 
						|
    Base.metadata.drop_all(engine)
 | 
						|
 | 
						|
 | 
						|
@pytest.mark.asyncio
 | 
						|
@pytest.mark.db
 | 
						|
async def test_queries(sqlalchemy_user_db: SQLAlchemyUserDatabase[UserDB]):
 | 
						|
    user = UserDB(
 | 
						|
        id="111",
 | 
						|
        email="lancelot@camelot.bt",
 | 
						|
        hashed_password=get_password_hash("guinevere"),
 | 
						|
    )
 | 
						|
 | 
						|
    # Create
 | 
						|
    user_db = await sqlalchemy_user_db.create(user)
 | 
						|
    assert user_db.id is not None
 | 
						|
    assert user_db.is_active is True
 | 
						|
    assert user_db.is_superuser is False
 | 
						|
    assert user_db.email == user.email
 | 
						|
 | 
						|
    # Update
 | 
						|
    user_db.is_superuser = True
 | 
						|
    await sqlalchemy_user_db.update(user_db)
 | 
						|
 | 
						|
    # Get by id
 | 
						|
    id_user = await sqlalchemy_user_db.get(user.id)
 | 
						|
    assert id_user is not None
 | 
						|
    assert id_user.id == user_db.id
 | 
						|
    assert id_user.is_superuser is True
 | 
						|
 | 
						|
    # Get by email
 | 
						|
    email_user = await sqlalchemy_user_db.get_by_email(str(user.email))
 | 
						|
    assert email_user is not None
 | 
						|
    assert email_user.id == user_db.id
 | 
						|
 | 
						|
    # List
 | 
						|
    users = await sqlalchemy_user_db.list()
 | 
						|
    assert len(users) == 1
 | 
						|
    first_user = users[0]
 | 
						|
    assert first_user.id == user_db.id
 | 
						|
 | 
						|
    # Exception when inserting existing email
 | 
						|
    with pytest.raises(sqlite3.IntegrityError):
 | 
						|
        await sqlalchemy_user_db.create(user)
 | 
						|
 | 
						|
    # Exception when inserting non-nullable fields
 | 
						|
    with pytest.raises(sqlite3.IntegrityError):
 | 
						|
        wrong_user = UserDB(id="222", hashed_password="aaa")
 | 
						|
        await sqlalchemy_user_db.create(wrong_user)
 | 
						|
 | 
						|
    # Unknown user
 | 
						|
    unknown_user = await sqlalchemy_user_db.get_by_email("galahad@camelot.bt")
 | 
						|
    assert unknown_user is None
 | 
						|
 | 
						|
    # Delete user
 | 
						|
    await sqlalchemy_user_db.delete(user)
 | 
						|
    deleted_user = await sqlalchemy_user_db.get(user.id)
 | 
						|
    assert deleted_user is None
 | 
						|
 | 
						|
    # Exception when creating/updating a OAuth user
 | 
						|
    user_oauth = UserDBOAuth(
 | 
						|
        id="222",
 | 
						|
        email="lancelot@camelot.bt",
 | 
						|
        hashed_password=get_password_hash("guinevere"),
 | 
						|
    )
 | 
						|
    with pytest.raises(NotSetOAuthAccountTableError):
 | 
						|
        await sqlalchemy_user_db.create(user_oauth)
 | 
						|
    with pytest.raises(NotSetOAuthAccountTableError):
 | 
						|
        await sqlalchemy_user_db.update(user_oauth)
 | 
						|
 | 
						|
    # Exception when trying to get by OAuth account
 | 
						|
    with pytest.raises(NotSetOAuthAccountTableError):
 | 
						|
        await sqlalchemy_user_db.get_by_oauth_account("foo", "bar")
 | 
						|
 | 
						|
 | 
						|
@pytest.mark.asyncio
 | 
						|
@pytest.mark.db
 | 
						|
async def test_queries_custom_fields(
 | 
						|
    sqlalchemy_user_db: SQLAlchemyUserDatabase[UserDB],
 | 
						|
):
 | 
						|
    """It should output custom fields in query result."""
 | 
						|
    user = UserDB(
 | 
						|
        id="111",
 | 
						|
        email="lancelot@camelot.bt",
 | 
						|
        hashed_password=get_password_hash("guinevere"),
 | 
						|
        first_name="Lancelot",
 | 
						|
    )
 | 
						|
    await sqlalchemy_user_db.create(user)
 | 
						|
 | 
						|
    id_user = await sqlalchemy_user_db.get(user.id)
 | 
						|
    assert id_user is not None
 | 
						|
    assert id_user.id == user.id
 | 
						|
    assert id_user.first_name == user.first_name
 | 
						|
 | 
						|
 | 
						|
@pytest.mark.asyncio
 | 
						|
@pytest.mark.db
 | 
						|
async def test_queries_oauth(
 | 
						|
    sqlalchemy_user_db_oauth: SQLAlchemyUserDatabase[UserDBOAuth],
 | 
						|
    oauth_account1,
 | 
						|
    oauth_account2,
 | 
						|
):
 | 
						|
    user = UserDBOAuth(
 | 
						|
        id="111",
 | 
						|
        email="lancelot@camelot.bt",
 | 
						|
        hashed_password=get_password_hash("guinevere"),
 | 
						|
        oauth_accounts=[oauth_account1, oauth_account2],
 | 
						|
    )
 | 
						|
 | 
						|
    # Create
 | 
						|
    user_db = await sqlalchemy_user_db_oauth.create(user)
 | 
						|
    assert user_db.id is not None
 | 
						|
    assert hasattr(user_db, "oauth_accounts")
 | 
						|
    assert len(user_db.oauth_accounts) == 2
 | 
						|
 | 
						|
    # Update
 | 
						|
    user_db.oauth_accounts[0].access_token = "NEW_TOKEN"
 | 
						|
    await sqlalchemy_user_db_oauth.update(user_db)
 | 
						|
 | 
						|
    # Get by id
 | 
						|
    id_user = await sqlalchemy_user_db_oauth.get(user.id)
 | 
						|
    assert id_user is not None
 | 
						|
    assert id_user.id == user_db.id
 | 
						|
    assert id_user.oauth_accounts[0].access_token == "NEW_TOKEN"
 | 
						|
 | 
						|
    # Get by email
 | 
						|
    email_user = await sqlalchemy_user_db_oauth.get_by_email(str(user.email))
 | 
						|
    assert email_user is not None
 | 
						|
    assert email_user.id == user_db.id
 | 
						|
    assert len(email_user.oauth_accounts) == 2
 | 
						|
 | 
						|
    # List
 | 
						|
    users = await sqlalchemy_user_db_oauth.list()
 | 
						|
    assert len(users) == 1
 | 
						|
    first_user = users[0]
 | 
						|
    assert first_user.id == user_db.id
 | 
						|
    assert len(first_user.oauth_accounts) == 2
 | 
						|
 | 
						|
    # Get by OAuth account
 | 
						|
    oauth_user = await sqlalchemy_user_db_oauth.get_by_oauth_account(
 | 
						|
        oauth_account1.oauth_name, oauth_account1.account_id
 | 
						|
    )
 | 
						|
    assert oauth_user is not None
 | 
						|
    assert oauth_user.id == user.id
 | 
						|
 | 
						|
    # Unknown OAuth account
 | 
						|
    unknown_oauth_user = await sqlalchemy_user_db_oauth.get_by_oauth_account(
 | 
						|
        "foo", "bar"
 | 
						|
    )
 | 
						|
    assert unknown_oauth_user is None
 |