2021-02-20 17:58:50 +00:00
|
|
|
"""Challenge helpers"""
|
2021-02-17 22:52:49 +00:00
|
|
|
from enum import Enum
|
2021-02-20 19:16:20 +00:00
|
|
|
from typing import TYPE_CHECKING, Optional
|
2021-02-17 22:52:49 +00:00
|
|
|
|
2021-02-20 18:41:32 +00:00
|
|
|
from django.db.models.base import Model
|
2021-02-17 22:52:49 +00:00
|
|
|
from django.http import JsonResponse
|
2021-02-20 22:19:27 +00:00
|
|
|
from rest_framework.fields import ChoiceField, DictField
|
2021-02-17 22:52:49 +00:00
|
|
|
from rest_framework.serializers import CharField, Serializer
|
|
|
|
|
2021-02-20 17:58:50 +00:00
|
|
|
from authentik.flows.transfer.common import DataclassEncoder
|
|
|
|
|
2021-02-20 19:16:20 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from authentik.flows.stage import StageView
|
|
|
|
|
2021-02-17 22:52:49 +00:00
|
|
|
|
|
|
|
class ChallengeTypes(Enum):
|
2021-02-20 17:58:50 +00:00
|
|
|
"""Currently defined challenge types"""
|
2021-02-17 22:52:49 +00:00
|
|
|
|
|
|
|
native = "native"
|
|
|
|
shell = "shell"
|
|
|
|
redirect = "redirect"
|
2021-02-20 22:19:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ErrorDetailSerializer(Serializer):
|
|
|
|
"""Serializer for rest_framework's error messages"""
|
|
|
|
|
|
|
|
string = CharField()
|
|
|
|
code = CharField()
|
|
|
|
|
|
|
|
def create(self, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
|
|
|
|
|
|
|
def update(self, instance: Model, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
2021-02-17 22:52:49 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Challenge(Serializer):
|
2021-02-20 17:58:50 +00:00
|
|
|
"""Challenge that gets sent to the client based on which stage
|
|
|
|
is currently active"""
|
2021-02-17 22:52:49 +00:00
|
|
|
|
2021-03-08 10:14:00 +00:00
|
|
|
type = ChoiceField(
|
|
|
|
choices=[(x.name, x.name) for x in ChallengeTypes],
|
|
|
|
)
|
2021-02-17 22:52:49 +00:00
|
|
|
component = CharField(required=False)
|
2021-02-20 18:41:32 +00:00
|
|
|
title = CharField(required=False)
|
2021-03-22 12:44:17 +00:00
|
|
|
background = CharField(required=False)
|
2021-02-20 18:41:32 +00:00
|
|
|
|
2021-02-20 22:19:27 +00:00
|
|
|
response_errors = DictField(
|
|
|
|
child=ErrorDetailSerializer(many=True), allow_empty=False, required=False
|
|
|
|
)
|
|
|
|
|
2021-02-20 18:41:32 +00:00
|
|
|
def create(self, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
|
|
|
|
|
|
|
def update(self, instance: Model, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
2021-02-17 22:52:49 +00:00
|
|
|
|
|
|
|
|
2021-02-20 22:19:27 +00:00
|
|
|
class RedirectChallenge(Challenge):
|
|
|
|
"""Challenge type to redirect the client"""
|
|
|
|
|
|
|
|
to = CharField()
|
|
|
|
|
|
|
|
|
|
|
|
class ShellChallenge(Challenge):
|
|
|
|
"""Legacy challenge type to render HTML as-is"""
|
|
|
|
|
|
|
|
body = CharField()
|
|
|
|
|
|
|
|
|
2021-02-21 12:15:45 +00:00
|
|
|
class WithUserInfoChallenge(Challenge):
|
|
|
|
"""Challenge base which shows some user info"""
|
|
|
|
|
|
|
|
pending_user = CharField()
|
|
|
|
pending_user_avatar = CharField()
|
|
|
|
|
|
|
|
|
2021-03-23 16:23:44 +00:00
|
|
|
class AccessDeniedChallenge(Challenge):
|
|
|
|
"""Challenge when a flow's active stage calls `stage_invalid()`."""
|
|
|
|
|
|
|
|
error_message = CharField(required=False)
|
|
|
|
|
|
|
|
|
2021-02-21 12:15:45 +00:00
|
|
|
class PermissionSerializer(Serializer):
|
|
|
|
"""Permission used for consent"""
|
|
|
|
|
|
|
|
name = CharField()
|
|
|
|
id = CharField()
|
|
|
|
|
|
|
|
def create(self, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
|
|
|
|
|
|
|
def update(self, instance: Model, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
|
|
|
|
|
|
|
|
2021-02-17 22:52:49 +00:00
|
|
|
class ChallengeResponse(Serializer):
|
2021-02-20 17:58:50 +00:00
|
|
|
"""Base class for all challenge responses"""
|
2021-02-17 22:52:49 +00:00
|
|
|
|
2021-02-20 19:16:20 +00:00
|
|
|
stage: Optional["StageView"]
|
|
|
|
|
2021-03-08 10:14:00 +00:00
|
|
|
def __init__(self, instance=None, data=None, **kwargs):
|
2021-02-20 19:16:20 +00:00
|
|
|
self.stage = kwargs.pop("stage", None)
|
|
|
|
super().__init__(instance=instance, data=data, **kwargs)
|
|
|
|
|
2021-02-20 18:41:32 +00:00
|
|
|
def create(self, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
|
|
|
|
|
|
|
def update(self, instance: Model, validated_data: dict) -> Model:
|
|
|
|
return Model()
|
|
|
|
|
2021-02-17 22:52:49 +00:00
|
|
|
|
|
|
|
class HttpChallengeResponse(JsonResponse):
|
2021-02-20 17:58:50 +00:00
|
|
|
"""Subclass of JsonResponse that uses the `DataclassEncoder`"""
|
|
|
|
|
2021-02-20 22:19:27 +00:00
|
|
|
def __init__(self, challenge, **kwargs) -> None:
|
2021-02-20 17:58:50 +00:00
|
|
|
# pyright: reportGeneralTypeIssues=false
|
2021-02-20 17:28:11 +00:00
|
|
|
super().__init__(challenge.data, encoder=DataclassEncoder, **kwargs)
|