Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Split up pycroft.lib.user into separate packages #746

Merged
merged 34 commits into from
Sep 28, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
8b9c9d7
qa: extract `user_from_pre_member`
lukasjuhrich Sep 4, 2024
b6d9590
Turn pycroft.lib.user into package
lukasjuhrich Sep 4, 2024
21eb6d9
move `pycroft.lib.user` contents into `_old` module
lukasjuhrich Sep 4, 2024
ab30dbf
Extract `lib.user.member_request`, `.exc`, and dependees
lukasjuhrich Sep 4, 2024
a0b172b
Extract `pycroft.lib.user_id`
lukasjuhrich Sep 4, 2024
b735b84
extract `lib.user.edit`
lukasjuhrich Sep 4, 2024
7a7315a
extract `lib.user.passwords`
lukasjuhrich Sep 4, 2024
d765be1
extract `lib.user.mail`
lukasjuhrich Sep 4, 2024
8eb0331
extract `can_target` to `.permissions` module
lukasjuhrich Sep 4, 2024
3b8fda3
Extract `lib.user.user_sheet`
lukasjuhrich Sep 17, 2024
4dec3db
Move user identification heuristics to `lib.user.member_request`
lukasjuhrich Sep 17, 2024
d61e72c
Extract `lib.user.info`
lukasjuhrich Sep 17, 2024
d28c991
Migrate `lib.user.migrate_user_host` to `lib.host.migrate_host`
lukasjuhrich Sep 20, 2024
5e4c796
Explicitly pass `session.session` in `migrate_host`
lukasjuhrich Sep 20, 2024
e7a84c3
Migrate `lib.user.setup_ipv4_networking` to `lib.host`
lukasjuhrich Sep 20, 2024
96a7d4c
Move `membership_{begin,end}_date` to `lib.user.info`
lukasjuhrich Sep 20, 2024
6cac442
Improve naming of `membership_*_date` functions
lukasjuhrich Sep 20, 2024
104219c
Make `scheduled_membership_*` implementations more concise
lukasjuhrich Sep 20, 2024
1977309
extract `lib.user.lifecycle`
lukasjuhrich Sep 20, 2024
a4a9060
Stricter typing for `lib.user`
lukasjuhrich Sep 20, 2024
bbd4ae0
typing: Strict nullability for `lib.user.mail_confirmation`
lukasjuhrich Sep 20, 2024
40137d4
Remove superfluous args from `check_new_user_data` and fix call
lukasjuhrich Sep 20, 2024
dbdd8f9
Strict nullability for `lib.user.member_request`
lukasjuhrich Sep 20, 2024
8c72119
Strict nullability for `lib.user.lifecycle`
lukasjuhrich Sep 20, 2024
5e6504f
extract `lib.user.blocking`
lukasjuhrich Sep 20, 2024
cd4b437
Strict nullability for rest of `lib.user`
lukasjuhrich Sep 20, 2024
eaec37c
migrate `send_password_reset_mail` to `.mail` and delete `_old`
lukasjuhrich Sep 20, 2024
634b88e
Merge branch 'lib_user_nullability' into lib_user_explode
lukasjuhrich Sep 20, 2024
f826097
Move up `mail_from` config dependency
lukasjuhrich Sep 24, 2024
f1d1002
Group mail config context in `MailConfig`
lukasjuhrich Sep 24, 2024
11e1176
lib: use configvar for mail config
lukasjuhrich Sep 28, 2024
fab32b3
Fix nullability requirement of `reply_to`
lukasjuhrich Sep 28, 2024
e1a0e13
lib.mail: Move MIMEText transformation to `Mail` class
lukasjuhrich Sep 28, 2024
12d11e5
Merge branch 'mail_config_extract' into lib_user_explode
lukasjuhrich Sep 28, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion pycroft/helpers/printing/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -100,7 +100,7 @@ def generate_user_sheet(
new_user: bool,
wifi: bool,
bank_account: BankAccount,
user: User | None = None,
user: User = None,
user_id: str | None = None,
plain_user_password: str | None = None,
generation_purpose: str = "",
Expand Down
52 changes: 50 additions & 2 deletions pycroft/lib/host.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,6 @@
from pycroft.helpers.net import port_name_sort_key
from pycroft.lib.logging import log_user_event
from pycroft.lib.net import get_subnets_for_room, get_free_ip, delete_ip
from pycroft.lib.user import migrate_user_host
from pycroft.model.facilities import Room
from pycroft.model.host import Interface, IP, Host, SwitchPort
from pycroft.model.port import PatchPort
Expand Down Expand Up @@ -95,7 +94,46 @@ def host_edit(host: Host, owner: User, room: Room, name: str, processor: User) -
host.owner = owner

if host.room != room:
migrate_user_host(host, room, processor)
migrate_host(session, host, room, processor)


def migrate_host(session: Session, host: Host, new_room: Room, processor: User) -> None:
"""
Migrate a Host to a new room and if necessary to a new subnet.
If the host changes subnet, it will get a new IP address.

:param host: Host to be migrated
:param new_room: new room of the host
:param processor: User processing the migration
:return:
"""
old_room = host.room
host.room = new_room

subnets_old = get_subnets_for_room(old_room)
subnets = get_subnets_for_room(new_room)

if subnets_old != subnets:
for interface in host.interfaces:
old_ips = tuple(ip for ip in interface.ips)
for old_ip in old_ips:
ip_address, subnet = get_free_ip(subnets)
new_ip = IP(interface=interface, address=ip_address, subnet=subnet)
session.add(new_ip)

old_address = old_ip.address
session.delete(old_ip)

message = deferred_gettext("Changed IP of {mac} from {old_ip} to {new_ip}.").format(
old_ip=str(old_address), new_ip=str(new_ip.address), mac=interface.mac
)
log_user_event(author=processor, user=host.owner, message=message.to_json())

message = deferred_gettext("Moved host '{name}' from {room_old} to {room_new}.").format(
name=host.name, room_old=old_room.short_name, room_new=new_room.short_name
)

log_user_event(author=processor, user=host.owner, message=message.to_json())


@with_transaction
Expand Down Expand Up @@ -234,3 +272,13 @@ def get_conflicting_interface(
if new_mac == current_mac:
return None
return session.scalar(select(Interface).filter_by(mac=new_mac))


def setup_ipv4_networking(session: Session, host: Host) -> None:
"""Add suitable ips for every interface of a host"""
subnets = get_subnets_for_room(host.room)

for interface in host.interfaces:
ip_address, subnet = get_free_ip(subnets)
new_ip = IP(interface=interface, address=ip_address, subnet=subnet)
session.add(new_ip)
147 changes: 107 additions & 40 deletions pycroft/lib/mail.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,45 +2,35 @@
pycroft.lib.mail
~~~~~~~~~~~~~~~~
"""

from __future__ import annotations
import logging
import os
import smtplib
import ssl
import traceback
import typing as t
from dataclasses import dataclass
from contextvars import ContextVar
from dataclasses import dataclass, field, InitVar
from email.header import Header
from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText
from email.utils import make_msgid, formatdate
from functools import lru_cache

import jinja2
from werkzeug.local import LocalProxy

from pycroft.lib.exc import PycroftLibException

mail_envelope_from = os.environ.get('PYCROFT_MAIL_ENVELOPE_FROM')
mail_from = os.environ.get('PYCROFT_MAIL_FROM')
mail_reply_to = os.environ.get('PYCROFT_MAIL_REPLY_TO')
smtp_host = os.environ.get('PYCROFT_SMTP_HOST')
smtp_port = int(os.environ.get('PYCROFT_SMTP_PORT', 465))
smtp_user = os.environ.get('PYCROFT_SMTP_USER')
smtp_password = os.environ.get('PYCROFT_SMTP_PASSWORD')
smtp_ssl = os.environ.get('PYCROFT_SMTP_SSL', 'ssl')
template_path_type = os.environ.get('PYCROFT_TEMPLATE_PATH_TYPE', 'filesystem')
template_path = os.environ.get('PYCROFT_TEMPLATE_PATH', 'pycroft/templates')

# TODO proxy and DI; set at app init
_config_var: ContextVar[MailConfig] = ContextVar("config")
config: MailConfig = LocalProxy(_config_var) # type: ignore[assignment]

logger = logging.getLogger('mail')
logger.setLevel(logging.INFO)

template_loader: jinja2.BaseLoader
if template_path_type == 'filesystem':
template_loader = jinja2.FileSystemLoader(searchpath=f'{template_path}/mail')
else:
template_loader = jinja2.PackageLoader(package_name='pycroft',
package_path=f'{template_path}/mail')

template_env = jinja2.Environment(loader=template_loader)


@dataclass
class Mail:
Expand All @@ -51,6 +41,16 @@ class Mail:
body_html: str | None = None
reply_to: str | None = None

@property
def body_plain_mime(self) -> MIMEText:
return MIMEText(self.body_plain, "plain", _charset="utf-8")

@property
def body_html_mime(self) -> MIMEText | None:
if not self.body_html:
return None
return MIMEText(self.body_html, "html", _charset="utf-8")


class MailTemplate:
template: str
Expand All @@ -61,27 +61,35 @@ def __init__(self, **kwargs: t.Any) -> None:
self.args = kwargs

def render(self, **kwargs: t.Any) -> tuple[str, str]:
plain = template_env.get_template(self.template).render(mode='plain', **self.args, **kwargs)
html = template_env.get_template(self.template).render(mode='html', **self.args, **kwargs)
plain = self.jinja_template.render(mode="plain", **self.args, **kwargs)
html = self.jinja_template.render(mode="html", **self.args, **kwargs)

return plain, html

@property
def jinja_template(self) -> jinja2.Template:
return _get_template(self.template)

def compose_mail(mail: Mail) -> MIMEMultipart:
msg = MIMEMultipart('alternative', _charset='utf-8')
msg['Message-Id'] = make_msgid()
msg['From'] = mail_from
msg['To'] = Header(mail.to_address)
msg['Subject'] = mail.subject
msg['Date'] = formatdate(localtime=True)

msg.attach(MIMEText(mail.body_plain, 'plain', _charset='utf-8'))
@lru_cache(maxsize=None)
def _get_template(template_location: str) -> jinja2.Template:
if config is None:
raise RuntimeError("`mail.config` not set up!")
return config.template_env.get_template(template_location)

if mail.body_html is not None:
msg.attach(MIMEText(mail.body_html, 'html', _charset='utf-8'))

if mail.reply_to is not None or mail.reply_to is not None:
msg['Reply-To'] = mail_reply_to if mail.reply_to is None else mail.reply_to
def compose_mail(mail: Mail, from_: str, default_reply_to: str | None) -> MIMEMultipart:
msg = MIMEMultipart("alternative", _charset="utf-8")
msg["Message-Id"] = make_msgid()
msg["From"] = from_
msg["To"] = str(Header(mail.to_address))
msg["Subject"] = mail.subject
msg["Date"] = formatdate(localtime=True)
msg.attach(mail.body_plain_mime)
if (html := mail.body_html_mime) is not None:
msg.attach(html)
if reply_to := mail.reply_to or default_reply_to:
msg["Reply-To"] = reply_to

print(msg)

Expand All @@ -100,12 +108,19 @@ def send_mails(mails: list[Mail]) -> tuple[bool, int]:
:param mails: A list of mails

:returns: Whether the transmission succeeded
:context: config
"""

if not smtp_host:
logger.critical("No mailserver config available")

raise RuntimeError
if config is None:
raise RuntimeError("`mail.config` not set up!")

mail_envelope_from = config.mail_envelope_from
mail_from = config.mail_from
mail_reply_to = config.mail_reply_to
smtp_host = config.smtp_host
smtp_port = config.smtp_port
smtp_user = config.smtp_user
smtp_password = config.smtp_password
smtp_ssl = config.smtp_ssl

use_ssl = smtp_ssl == 'ssl'
use_starttls = smtp_ssl == 'starttls'
Expand Down Expand Up @@ -159,7 +174,7 @@ def send_mails(mails: list[Mail]) -> tuple[bool, int]:

for mail in mails:
try:
mime_mail = compose_mail(mail)
mime_mail = compose_mail(mail, from_=mail_from, default_reply_to=mail_reply_to)
assert mail_envelope_from is not None
smtp.sendmail(from_addr=mail_envelope_from, to_addrs=mail.to_address,
msg=mime_mail.as_string())
Expand Down Expand Up @@ -248,3 +263,55 @@ def send_template_mails(
from pycroft.task import send_mails_async

send_mails_async.delay(mails)


@dataclass
class MailConfig:
mail_envelope_from: str
mail_from: str
mail_reply_to: str | None
smtp_host: str
smtp_user: str
smtp_password: str
smtp_port: int = field(default=465)
smtp_ssl: str = field(default="ssl")

template_path_type: InitVar[str | None] = None
template_path: InitVar[str | None] = None
template_env: jinja2.Environment = field(init=False)

@classmethod
def from_env(cls) -> t.Self:
env = os.environ
config = cls(
mail_envelope_from=env["PYCROFT_MAIL_ENVELOPE_FROM"],
mail_from=env["PYCROFT_MAIL_FROM"],
mail_reply_to=env.get("PYCROFT_MAIL_REPLY_TO"),
smtp_host=env["PYCROFT_SMTP_HOST"],
smtp_user=env["PYCROFT_SMTP_USER"],
smtp_password=env["PYCROFT_SMTP_PASSWORD"],
template_path_type=env.get("PYCROFT_TEMPLATE_PATH_TYPE"),
template_path=env.get("PYCROFT_TEMPLATE_PATH"),
)
if (smtp_port := env.get("PYCROFT_SMTP_PORT")) is not None:
config.smtp_port = int(smtp_port)
if (smtp_ssl := env.get("PYCROFT_SMTP_SSL")) is not None:
config.smtp_ssl = smtp_ssl

return config

def __post_init__(self, template_path_type: str | None, template_path: str | None) -> None:
template_loader: jinja2.BaseLoader
if template_path_type is None:
template_path_type = "filesystem"
if template_path is None:
template_path = "pycroft/templates"

if template_path_type == "filesystem":
template_loader = jinja2.FileSystemLoader(searchpath=f"{template_path}/mail")
else:
template_loader = jinja2.PackageLoader(
package_name="pycroft", package_path=f"{template_path}/mail"
)

self.template_env = jinja2.Environment(loader=template_loader)
8 changes: 5 additions & 3 deletions pycroft/lib/task.py
Original file line number Diff line number Diff line change
Expand Up @@ -114,12 +114,13 @@ class UserMoveTaskImpl(UserTaskImpl[UserMoveParams]):

def _execute(self, task: UserTask, parameters: UserMoveParams) -> None:
from pycroft.lib import user as lib_user
from pycroft.lib.facilities import get_room
if task.user.room is None:
self.errors.append("Tried to move in user, "
"but user was already living in a dormitory.")
return

room = lib_user.get_room(
room = get_room(
room_number=parameters.room_number,
level=parameters.level,
building_id=parameters.building_id,
Expand Down Expand Up @@ -147,13 +148,14 @@ class UserMoveInTaskImpl(UserTaskImpl):

def _execute(self, task: UserTask, parameters: UserMoveInParams) -> None:
from pycroft.lib import user as lib_user
from pycroft.lib.facilities import get_room

if task.user.room is not None:
self.errors.append("Tried to move in user, "
"but user was already living in a dormitory.")
return

room = lib_user.get_room(
room = get_room(
room_number=parameters.room_number,
level=parameters.level,
building_id=parameters.building_id,
Expand Down Expand Up @@ -195,7 +197,7 @@ def schedule_user_task(
due: DateTimeTz,
user: User,
parameters: TaskParams,
processor: User,
processor: User | None,
) -> UserTask:
if due < session.utcnow():
raise ValueError("the due date must be in the future")
Expand Down
Loading
Loading