majuna/app/portal/onion.py

91 lines
3.7 KiB
Python
Raw Normal View History

2022-05-04 15:36:36 +01:00
from datetime import datetime
2022-05-16 11:44:03 +01:00
from typing import Optional
2022-05-04 15:36:36 +01:00
from flask import flash, redirect, render_template, Response, Blueprint
2022-05-16 11:44:03 +01:00
from flask.typing import ResponseReturnValue
2022-05-04 15:36:36 +01:00
from flask_wtf import FlaskForm
from sqlalchemy import exc
from wtforms import StringField, SelectField, SubmitField
from flask_wtf.file import FileField
2022-11-09 13:36:12 +00:00
from wtforms.validators import DataRequired
2022-05-04 15:36:36 +01:00
from app.extensions import db
from app.models.base import Group
from app.models.onions import Onion
from app.portal.util import response_404, view_lifecycle
bp = Blueprint("onion", __name__)
2022-05-16 11:44:03 +01:00
class EditOnionForm(FlaskForm): # type: ignore
2022-12-06 19:41:11 +00:00
domain_name = StringField('Domain Name', validators=[DataRequired()])
2022-12-07 19:25:51 +00:00
description = StringField('Description', validators=[DataRequired()])
2022-11-09 13:36:12 +00:00
tls_private_key = FileField('TLS Private Key (PEM format)',
description="If no file is submitted, the TLS key will remain unchanged.")
tls_public_key = FileField('TLS Certificate (PEM format)',
description="If no file is submitted, the TLS certificate will remain unchanged.")
2022-12-07 19:25:51 +00:00
group = SelectField('Group', validators=[DataRequired()])
2022-05-04 15:36:36 +01:00
submit = SubmitField('Save Changes')
@bp.route("/new", methods=['GET', 'POST'])
@bp.route("/new/<group_id>", methods=['GET', 'POST'])
2022-05-16 11:44:03 +01:00
def onion_new(group_id: Optional[int] = None) -> ResponseReturnValue:
return redirect("/ui/web/onions/new")
2022-05-04 15:36:36 +01:00
@bp.route('/edit/<onion_id>', methods=['GET', 'POST'])
2022-05-16 11:44:03 +01:00
def onion_edit(onion_id: int) -> ResponseReturnValue:
onion: Optional[Onion] = Onion.query.filter(Onion.id == onion_id).first()
2022-05-04 15:36:36 +01:00
if onion is None:
return Response(render_template("error.html.j2",
section="onion",
header="404 Onion Not Found",
message="The requested onion service could not be found."),
status=404)
form = EditOnionForm(group=onion.group_id,
2022-12-07 19:25:51 +00:00
domain_name=onion.domain_name,
2022-05-04 15:36:36 +01:00
description=onion.description)
form.group.choices = [(x.id, x.group_name) for x in Group.query.all()]
if form.validate_on_submit():
onion.group_id = form.group.data
onion.description = form.description.data
2022-12-06 19:41:11 +00:00
onion.domain_name = form.domain_name.data
2022-11-09 13:36:12 +00:00
for at in [
"tls_private_key",
"tls_public_key"
]:
2023-01-21 14:10:40 +00:00
if getattr(form, at).data is not None:
# Don't clear the key if no key is uploaded
setattr(onion, at, getattr(form, at).data.read())
2022-05-04 15:36:36 +01:00
onion.updated = datetime.utcnow()
try:
db.session.commit()
flash("Saved changes to group.", "success")
except exc.SQLAlchemyError:
flash("An error occurred saving the changes to the group.", "danger")
return render_template("onion.html.j2",
section="onion",
onion=onion, form=form)
@bp.route("/list")
2022-05-16 11:44:03 +01:00
def onion_list() -> ResponseReturnValue:
return redirect("/ui/web/onions")
2022-05-04 15:36:36 +01:00
@bp.route("/destroy/<onion_id>", methods=['GET', 'POST'])
def onion_destroy(onion_id: str) -> ResponseReturnValue:
onion = Onion.query.filter(Onion.id == int(onion_id), Onion.destroyed.is_(None)).first()
2022-05-04 15:36:36 +01:00
if onion is None:
return response_404("The requested onion service could not be found.")
return view_lifecycle(
header=f"Destroy onion service {onion.onion_name}",
message=onion.description,
success_message="Successfully removed onion service.",
2022-05-04 15:36:36 +01:00
success_view="portal.onion.onion_list",
section="onion",
resource=onion,
action="destroy"
)