Added validators for users and recipes (#1052)

* Added validators for users and recipes

provide a simple get api, allowing to test for existence of
- user by username
- recipe by slug
- group by name (not tested yet)

* updated formatting

* Use group_id+slug for recipes, use ValidationRespone

* Fixed Flake8 errors and warnings

* add missing field for TestUser init
This commit is contained in:
Matthias Bilger 2022-03-15 23:28:42 +01:00 committed by GitHub
parent c8c02036a3
commit e109391e9a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 129 additions and 2 deletions

View File

@ -1,3 +1,5 @@
from typing import Union
from sqlalchemy.orm.session import Session
from mealie.db.models.group import Group
@ -22,3 +24,9 @@ class RepositoryGroup(RepositoryGeneric[GroupInDB, Group]):
group: GroupInDB = session.query(self.sql_model).filter_by(**{match_key: match_value}).one_or_none()
return group.mealplans
def get_by_name(self, name: str, limit=1) -> Union[GroupInDB, Group, None]:
dbgroup = self.session.query(self.sql_model).filter_by(**{"name": name}).one_or_none()
if dbgroup is None:
return None
return self.schema.from_orm(dbgroup)

View File

@ -1,5 +1,5 @@
from random import randint
from typing import Any
from typing import Any, Optional
from uuid import UUID
from sqlalchemy import and_, func
@ -143,3 +143,13 @@ class RepositoryRecipes(RepositoryGeneric[Recipe, RecipeModel]):
.order_by(func.random()) # Postgres and SQLite specific
.limit(limit)
]
def get_by_slug(self, group_id: UUID, slug: str, limit=1) -> Optional[Recipe]:
dbrecipe = (
self.session.query(RecipeModel)
.filter(RecipeModel.group_id == group_id, RecipeModel.slug == slug)
.one_or_none()
)
if dbrecipe is None:
return None
return self.schema.from_orm(dbrecipe)

View File

@ -1,5 +1,6 @@
import random
import shutil
from typing import Optional
from mealie.assets import users as users_assets
from mealie.schema.user.user import PrivateUser, User
@ -34,3 +35,9 @@ class RepositoryUsers(RepositoryGeneric[PrivateUser, User]):
# Delete the user's directory
shutil.rmtree(PrivateUser.get_directory(id))
return entry
def get_by_username(self, username: str, limit=1) -> Optional[User]:
dbuser = self.session.query(User).filter(User.username == username).one_or_none()
if dbuser is None:
return None
return self.schema.from_orm(dbuser)

View File

@ -1,6 +1,6 @@
from fastapi import APIRouter
from . import admin, app, auth, comments, groups, organizers, parser, recipe, shared, unit_and_foods, users
from . import admin, app, auth, comments, groups, organizers, parser, recipe, shared, unit_and_foods, users, validators
router = APIRouter(prefix="/api")
@ -15,3 +15,4 @@ router.include_router(comments.router)
router.include_router(parser.router)
router.include_router(unit_and_foods.router)
router.include_router(admin.router)
router.include_router(validators.router)

View File

@ -0,0 +1,9 @@
from fastapi import APIRouter
from . import validators
prefix = "/validators"
router = APIRouter()
router.include_router(validators.router, prefix=prefix, tags=["Validators"])

View File

@ -0,0 +1,34 @@
from uuid import UUID
from fastapi import APIRouter, Depends
from sqlalchemy.orm.session import Session
from mealie.db.db_setup import generate_session
from mealie.repos.all_repositories import get_repositories
from mealie.schema.response import ValidationResponse
router = APIRouter()
@router.get("/user/{name}", response_model=ValidationResponse)
def validate_user(name: str, session: Session = Depends(generate_session)):
"""Checks if a user with the given name exists"""
db = get_repositories(session)
existing_element = db.users.get_by_username(name)
return ValidationResponse(valid=existing_element is None)
@router.get("/group/{name}", response_model=ValidationResponse)
def validate_group(name: str, session: Session = Depends(generate_session)):
"""Checks if a group with the given name exists"""
db = get_repositories(session)
existing_element = db.groups.get_by_name(name)
return ValidationResponse(valid=existing_element is None)
@router.get("/recipe/{group_id}/{slug}", response_model=ValidationResponse)
def validate_recipe(group_id: UUID, slug: str, session: Session = Depends(generate_session)):
"""Checks if a group with the given slug exists"""
db = get_repositories(session)
existing_element = db.recipes.get_by_slug(group_id, slug)
return ValidationResponse(valid=existing_element is None)

View File

@ -1,2 +1,3 @@
# GENERATED CODE - DO NOT MODIFY BY HAND
from .responses import *
from .validation import *

View File

@ -0,0 +1,5 @@
from pydantic import BaseModel
class ValidationResponse(BaseModel):
valid: bool = False

View File

@ -33,6 +33,7 @@ def admin_user(api_client: TestClient, api_routes: utils.AppRoutes):
yield utils.TestUser(
_group_id=user_data.get("groupId"),
user_id=user_data.get("id"),
username=user_data.get("username"),
email=user_data.get("email"),
token=token,
)

View File

@ -28,6 +28,7 @@ def build_unique_user(group: str, api_client: requests) -> utils.TestUser:
_group_id=user_data.get("groupId"),
user_id=user_data.get("id"),
email=user_data.get("email"),
username=user_data.get("username"),
token=token,
)
@ -69,6 +70,7 @@ def g2_user(admin_token, api_client: TestClient, api_routes: utils.AppRoutes):
_group_id=group_id,
token=token,
email=create_data["email"],
username=create_data.get("username"),
)
finally:
# TODO: Delete User after test
@ -93,6 +95,7 @@ def unique_user(api_client: TestClient, api_routes: utils.AppRoutes):
_group_id=user_data.get("groupId"),
user_id=user_data.get("id"),
email=user_data.get("email"),
username=user_data.get("username"),
token=token,
)
finally:
@ -142,6 +145,7 @@ def user_tuple(admin_token, api_client: requests, api_routes: utils.AppRoutes) -
utils.TestUser(
_group_id=user_data.get("groupId"),
user_id=user_data.get("id"),
username=user_data.get("username"),
email=user_data.get("email"),
token=token,
)

View File

@ -0,0 +1,46 @@
from fastapi.testclient import TestClient
from mealie.db.db_setup import create_session
from mealie.schema.recipe.recipe import Recipe
from tests.utils.fixture_schemas import TestUser
class Routes:
user = "/api/validators/user"
recipe = "/api/validators/recipe"
def test_validators_user(api_client: TestClient, unique_user: TestUser):
session = create_session()
# Test existing user
response = api_client.get(Routes.user + f"/{unique_user.username}")
assert response.status_code == 200
response_data = response.json()
assert not response_data["valid"]
# Test non-existing user
response = api_client.get(Routes.user + f"/{unique_user.username}2")
assert response.status_code == 200
response_data = response.json()
assert response_data["valid"]
session.close()
def test_validators_recipe(api_client: TestClient, random_recipe: Recipe):
session = create_session()
# Test existing user
response = api_client.get(Routes.recipe + f"/{random_recipe.group_id}/{random_recipe.slug}")
assert response.status_code == 200
response_data = response.json()
assert not response_data["valid"]
# Test non-existing user
response = api_client.get(Routes.recipe + f"/{random_recipe.group_id}/{random_recipe.slug}-test")
assert response.status_code == 200
response_data = response.json()
assert response_data["valid"]
session.close()

View File

@ -7,6 +7,7 @@ from uuid import UUID
class TestUser:
email: str
user_id: UUID
username: str
_group_id: UUID
token: Any