Coverage for opt/mealie/lib/python3.12/site-packages/mealie/services/user_services/registration_service.py: 65%
79 statements
« prev ^ index » next coverage.py v7.10.6, created at 2025-12-05 13:45 +0000
« prev ^ index » next coverage.py v7.10.6, created at 2025-12-05 13:45 +0000
1from logging import Logger 1c
2from uuid import uuid4 1c
4from fastapi import HTTPException, status 1c
5from pydantic import UUID4 1c
7from mealie.core.config import get_app_settings 1c
8from mealie.core.security import hash_password 1c
9from mealie.lang.providers import Translator 1c
10from mealie.repos.all_repositories import get_repositories 1c
11from mealie.repos.repository_factory import AllRepositories 1c
12from mealie.schema.group.group_preferences import CreateGroupPreferences 1c
13from mealie.schema.household.household import HouseholdCreate, HouseholdInDB 1c
14from mealie.schema.household.household_preferences import CreateHouseholdPreferences 1c
15from mealie.schema.user.registration import CreateUserRegistration 1c
16from mealie.schema.user.user import GroupBase, GroupInDB, PrivateUser, UserIn 1c
17from mealie.services.group_services.group_service import GroupService 1c
18from mealie.services.household_services.household_service import HouseholdService 1c
19from mealie.services.seeder.seeder_service import SeederService 1c
22class RegistrationService: 1c
23 logger: Logger 1c
24 repos: AllRepositories 1c
26 def __init__(self, logger: Logger, db: AllRepositories, translator: Translator): 1c
27 self.logger = logger 1ba
28 self.repos = db 1ba
29 self.t = translator.t 1ba
31 def _create_new_user(self, group: GroupInDB, household: HouseholdInDB, new_group: bool) -> PrivateUser: 1c
32 new_user = UserIn( 1ba
33 email=self.registration.email,
34 username=self.registration.username,
35 password=hash_password(self.registration.password),
36 full_name=self.registration.full_name,
37 advanced=self.registration.advanced,
38 group=group,
39 household=household,
40 can_invite=new_group,
41 can_manage=new_group,
42 can_manage_household=new_group,
43 can_organize=new_group,
44 )
46 # TODO: problem with repository type, not type here
47 return self.repos.users.create(new_user) # type: ignore 1ba
49 def _register_new_group(self) -> GroupInDB: 1c
50 group_data = GroupBase(name=self.registration.group) 1ba
52 group_preferences = CreateGroupPreferences( 1ba
53 group_id=uuid4(),
54 private_group=self.registration.private,
55 )
57 return GroupService.create_group(self.repos, group_data, group_preferences) 1ba
59 def _fetch_or_register_new_household(self, group_id: UUID4) -> HouseholdInDB: 1c
60 settings = get_app_settings() 1ba
61 new_household_name = self.registration.household or settings.DEFAULT_HOUSEHOLD 1ba
63 group_repos = get_repositories(self.repos.session, group_id=group_id) 1ba
64 household_fetch = group_repos.households.get_by_name(new_household_name) 1ba
65 if household_fetch: 65 ↛ 68line 65 didn't jump to line 68 because the condition on line 65 was always true1ba
66 return household_fetch 1ba
68 household_data = HouseholdCreate(name=new_household_name)
69 household_preferences = CreateHouseholdPreferences(
70 private_household=self.registration.private,
71 first_day_of_week=0,
72 recipe_public=not self.registration.private,
73 recipe_show_nutrition=self.registration.advanced,
74 recipe_show_assets=self.registration.advanced,
75 recipe_landscape_view=False,
76 recipe_disable_comments=self.registration.advanced,
77 )
78 return HouseholdService.create_household(group_repos, household_data, household_preferences)
80 def register_user(self, registration: CreateUserRegistration) -> PrivateUser: 1c
81 self.registration = registration 1ba
83 if self.repos.users.get_by_username(registration.username): 1ba
84 raise HTTPException(status.HTTP_409_CONFLICT, {"message": self.t("exceptions.username-conflict-error")})
85 elif self.repos.users.get_one(registration.email, "email"): 85 ↛ 86line 85 didn't jump to line 86 because the condition on line 85 was never true1ba
86 raise HTTPException(status.HTTP_409_CONFLICT, {"message": self.t("exceptions.email-conflict-error")})
88 token_entry = None 1ba
89 new_group = False 1ba
91 if registration.group_token: 91 ↛ 92line 91 didn't jump to line 92 because the condition on line 91 was never true1ba
92 token_entry = self.repos.group_invite_tokens.get_one(registration.group_token)
93 if not token_entry:
94 raise HTTPException(status.HTTP_400_BAD_REQUEST, {"message": "Invalid group token"})
96 maybe_none_group = self.repos.groups.get_one(token_entry.group_id)
97 if maybe_none_group is None:
98 raise HTTPException(status.HTTP_400_BAD_REQUEST, {"message": "Invalid group token"})
99 group = maybe_none_group
101 maybe_none_household = self.repos.households.get_one(token_entry.household_id)
102 if maybe_none_household is None:
103 raise HTTPException(status.HTTP_400_BAD_REQUEST, {"message": "Invalid group token"})
104 household = maybe_none_household
105 elif registration.group: 105 ↛ 110line 105 didn't jump to line 110 because the condition on line 105 was always true1ba
106 new_group = True 1ba
107 group = self._register_new_group() 1ba
108 household = self._fetch_or_register_new_household(group.id) 1ba
109 else:
110 raise HTTPException(status.HTTP_400_BAD_REQUEST, {"message": "Missing group"})
112 self.logger.info(f"Registering user {registration.username}") 1ba
113 user = self._create_new_user(group, household, new_group) 1ba
115 if new_group and registration.seed_data: 1ba
116 seeder_service = SeederService(self.repos)
117 seeder_service.seed_foods(registration.locale)
118 seeder_service.seed_labels(registration.locale)
119 seeder_service.seed_units(registration.locale)
121 if token_entry and user: 121 ↛ 122line 121 didn't jump to line 122 because the condition on line 121 was never true1b
122 token_entry.uses_left = token_entry.uses_left - 1
124 if token_entry.uses_left == 0:
125 self.repos.group_invite_tokens.delete(token_entry.token)
127 else:
128 self.repos.group_invite_tokens.update(token_entry.token, token_entry)
130 return user 1b