Compare commits
	
		
			38 Commits
		
	
	
		
			enterprise
			...
			version/20
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 31fe0e5923 | |||
| 8b619635ea | |||
| 1f1db523c0 | |||
| bbc23e1d77 | |||
| c30b7ee3e9 | |||
| 2ba79627bc | |||
| 198cbe1d9d | |||
| db6da159d5 | |||
| 9862e32078 | |||
| a7714e2892 | |||
| 073e1d241b | |||
| 5c5cc1c7da | |||
| 3dccce1095 | |||
| 78f997fbee | |||
| ed83c2b0b1 | |||
| af780deb27 | |||
| a4be38567f | |||
| 39aafbb34a | |||
| 07eb5fe533 | |||
| 301a89dd92 | |||
| cd6d0a47f3 | |||
| 8a23eaef1e | |||
| 8f285fbcc5 | |||
| 5d391424f7 | |||
| 2de11f8a69 | |||
| b2dcf94aba | |||
| adb532fc5d | |||
| 5d3b35d1ba | |||
| 433a94d9ee | |||
| f28d622d10 | |||
| 50a68c22c5 | |||
| 13c99c8546 | |||
| 7243add30f | |||
| 6611a64a62 | |||
| 5262f61483 | |||
| 9dcbb4af9e | |||
| 0665bfac58 | |||
| 790e0c4d80 | 
| @ -1,5 +1,5 @@ | ||||
| [bumpversion] | ||||
| current_version = 2024.12.3 | ||||
| current_version = 2025.2.1 | ||||
| tag = True | ||||
| commit = True | ||||
| parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)(?:-(?P<rc_t>[a-zA-Z-]+)(?P<rc_n>[1-9]\\d*))? | ||||
|  | ||||
| @ -2,7 +2,7 @@ | ||||
|  | ||||
| from os import environ | ||||
|  | ||||
| __version__ = "2024.12.3" | ||||
| __version__ = "2025.2.1" | ||||
| ENV_GIT_HASH_KEY = "GIT_BUILD_HASH" | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -50,7 +50,6 @@ from authentik.enterprise.providers.microsoft_entra.models import ( | ||||
|     MicrosoftEntraProviderGroup, | ||||
|     MicrosoftEntraProviderUser, | ||||
| ) | ||||
| from authentik.enterprise.providers.rac.models import ConnectionToken | ||||
| from authentik.enterprise.providers.ssf.models import StreamEvent | ||||
| from authentik.enterprise.stages.authenticator_endpoint_gdtc.models import ( | ||||
|     EndpointDevice, | ||||
| @ -72,6 +71,7 @@ from authentik.providers.oauth2.models import ( | ||||
|     DeviceToken, | ||||
|     RefreshToken, | ||||
| ) | ||||
| from authentik.providers.rac.models import ConnectionToken | ||||
| from authentik.providers.scim.models import SCIMProviderGroup, SCIMProviderUser | ||||
| from authentik.rbac.models import Role | ||||
| from authentik.sources.scim.models import SCIMSourceGroup, SCIMSourceUser | ||||
|  | ||||
| @ -35,8 +35,7 @@ from authentik.flows.planner import ( | ||||
|     FlowPlanner, | ||||
| ) | ||||
| from authentik.flows.stage import StageView | ||||
| from authentik.flows.views.executor import NEXT_ARG_NAME, SESSION_KEY_GET, SESSION_KEY_PLAN | ||||
| from authentik.lib.utils.urls import redirect_with_qs | ||||
| from authentik.flows.views.executor import NEXT_ARG_NAME, SESSION_KEY_GET | ||||
| from authentik.lib.views import bad_request_message | ||||
| from authentik.policies.denied import AccessDeniedResponse | ||||
| from authentik.policies.utils import delete_none_values | ||||
| @ -47,8 +46,9 @@ from authentik.stages.user_write.stage import PLAN_CONTEXT_USER_PATH | ||||
|  | ||||
| LOGGER = get_logger() | ||||
|  | ||||
| SESSION_KEY_OVERRIDE_FLOW_TOKEN = "authentik/flows/source_override_flow_token"  # nosec | ||||
| PLAN_CONTEXT_SOURCE_GROUPS = "source_groups" | ||||
| SESSION_KEY_SOURCE_FLOW_STAGES = "authentik/flows/source_flow_stages" | ||||
| SESSION_KEY_OVERRIDE_FLOW_TOKEN = "authentik/flows/source_override_flow_token"  # nosec | ||||
|  | ||||
|  | ||||
| class MessageStage(StageView): | ||||
| @ -219,28 +219,28 @@ class SourceFlowManager: | ||||
|             } | ||||
|         ) | ||||
|         flow_context.update(self.policy_context) | ||||
|         if SESSION_KEY_OVERRIDE_FLOW_TOKEN in self.request.session: | ||||
|             token: FlowToken = self.request.session.get(SESSION_KEY_OVERRIDE_FLOW_TOKEN) | ||||
|             self._logger.info("Replacing source flow with overridden flow", flow=token.flow.slug) | ||||
|             plan = token.plan | ||||
|             plan.context[PLAN_CONTEXT_IS_RESTORED] = token | ||||
|             plan.context.update(flow_context) | ||||
|             for stage in self.get_stages_to_append(flow): | ||||
|                 plan.append_stage(stage) | ||||
|             if stages: | ||||
|                 for stage in stages: | ||||
|                     plan.append_stage(stage) | ||||
|             self.request.session[SESSION_KEY_PLAN] = plan | ||||
|             flow_slug = token.flow.slug | ||||
|             token.delete() | ||||
|             return redirect_with_qs( | ||||
|                 "authentik_core:if-flow", | ||||
|                 self.request.GET, | ||||
|                 flow_slug=flow_slug, | ||||
|             ) | ||||
|         flow_context.setdefault(PLAN_CONTEXT_REDIRECT, final_redirect) | ||||
|  | ||||
|         if not flow: | ||||
|             # We only check for the flow token here if we don't have a flow, otherwise we rely on | ||||
|             # SESSION_KEY_SOURCE_FLOW_STAGES to delegate the usage of this token and dynamically add | ||||
|             # stages that deal with this token to return to another flow | ||||
|             if SESSION_KEY_OVERRIDE_FLOW_TOKEN in self.request.session: | ||||
|                 token: FlowToken = self.request.session.get(SESSION_KEY_OVERRIDE_FLOW_TOKEN) | ||||
|                 self._logger.info( | ||||
|                     "Replacing source flow with overridden flow", flow=token.flow.slug | ||||
|                 ) | ||||
|                 plan = token.plan | ||||
|                 plan.context[PLAN_CONTEXT_IS_RESTORED] = token | ||||
|                 plan.context.update(flow_context) | ||||
|                 for stage in self.get_stages_to_append(flow): | ||||
|                     plan.append_stage(stage) | ||||
|                 if stages: | ||||
|                     for stage in stages: | ||||
|                         plan.append_stage(stage) | ||||
|                 redirect = plan.to_redirect(self.request, token.flow) | ||||
|                 token.delete() | ||||
|                 return redirect | ||||
|             return bad_request_message( | ||||
|                 self.request, | ||||
|                 _("Configured flow does not exist."), | ||||
| @ -259,6 +259,8 @@ class SourceFlowManager: | ||||
|         if stages: | ||||
|             for stage in stages: | ||||
|                 plan.append_stage(stage) | ||||
|         for stage in self.request.session.get(SESSION_KEY_SOURCE_FLOW_STAGES, []): | ||||
|             plan.append_stage(stage) | ||||
|         return plan.to_redirect(self.request, flow) | ||||
|  | ||||
|     def handle_auth( | ||||
| @ -295,6 +297,8 @@ class SourceFlowManager: | ||||
|         # When request isn't authenticated we jump straight to auth | ||||
|         if not self.request.user.is_authenticated: | ||||
|             return self.handle_auth(connection) | ||||
|         # When an override flow token exists we actually still use a flow for link | ||||
|         # to continue the existing flow we came from | ||||
|         if SESSION_KEY_OVERRIDE_FLOW_TOKEN in self.request.session: | ||||
|             return self._prepare_flow(None, connection) | ||||
|         connection.save() | ||||
|  | ||||
| @ -67,6 +67,8 @@ def clean_expired_models(self: SystemTask): | ||||
|                 raise ImproperlyConfigured( | ||||
|                     "Invalid session_storage setting, allowed values are db and cache" | ||||
|                 ) | ||||
|     if CONFIG.get("session_storage", "cache") == "db": | ||||
|         DBSessionStore.clear_expired() | ||||
|     LOGGER.debug("Expired sessions", model=AuthenticatedSession, amount=amount) | ||||
|  | ||||
|     messages.append(f"Expired {amount} {AuthenticatedSession._meta.verbose_name_plural}") | ||||
|  | ||||
| @ -11,6 +11,7 @@ | ||||
|         build: "{{ build }}", | ||||
|         api: { | ||||
|             base: "{{ base_url }}", | ||||
|             relBase: "{{ base_url_rel }}", | ||||
|         }, | ||||
|     }; | ||||
|     window.addEventListener("DOMContentLoaded", function () { | ||||
|  | ||||
| @ -8,6 +8,8 @@ | ||||
|     <head> | ||||
|         <meta charset="UTF-8"> | ||||
|         <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"> | ||||
|         {# Darkreader breaks the site regardless of theme as its not compatible with webcomponents, and we default to a dark theme based on preferred colour-scheme #} | ||||
|         <meta name="darkreader-lock"> | ||||
|         <title>{% block title %}{% trans title|default:brand.branding_title %}{% endblock %}</title> | ||||
|         <link rel="icon" href="{{ brand.branding_favicon_url }}"> | ||||
|         <link rel="shortcut icon" href="{{ brand.branding_favicon_url }}"> | ||||
|  | ||||
| @ -53,6 +53,7 @@ class InterfaceView(TemplateView): | ||||
|         kwargs["build"] = get_build_hash() | ||||
|         kwargs["url_kwargs"] = self.kwargs | ||||
|         kwargs["base_url"] = self.request.build_absolute_uri(CONFIG.get("web.path", "/")) | ||||
|         kwargs["base_url_rel"] = CONFIG.get("web.path", "/") | ||||
|         return super().get_context_data(**kwargs) | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -1,14 +0,0 @@ | ||||
| """RAC app config""" | ||||
|  | ||||
| from authentik.enterprise.apps import EnterpriseConfig | ||||
|  | ||||
|  | ||||
| class AuthentikEnterpriseProviderRAC(EnterpriseConfig): | ||||
|     """authentik enterprise rac app config""" | ||||
|  | ||||
|     name = "authentik.enterprise.providers.rac" | ||||
|     label = "authentik_providers_rac" | ||||
|     verbose_name = "authentik Enterprise.Providers.RAC" | ||||
|     default = True | ||||
|     mountpoint = "" | ||||
|     ws_mountpoint = "authentik.enterprise.providers.rac.urls" | ||||
| @ -16,7 +16,6 @@ TENANT_APPS = [ | ||||
|     "authentik.enterprise.audit", | ||||
|     "authentik.enterprise.providers.google_workspace", | ||||
|     "authentik.enterprise.providers.microsoft_entra", | ||||
|     "authentik.enterprise.providers.rac", | ||||
|     "authentik.enterprise.providers.ssf", | ||||
|     "authentik.enterprise.stages.authenticator_endpoint_gdtc", | ||||
|     "authentik.enterprise.stages.source", | ||||
|  | ||||
| @ -9,13 +9,16 @@ from django.utils.timezone import now | ||||
| from guardian.shortcuts import get_anonymous_user | ||||
|  | ||||
| from authentik.core.models import Source, User | ||||
| from authentik.core.sources.flow_manager import SESSION_KEY_OVERRIDE_FLOW_TOKEN | ||||
| from authentik.core.sources.flow_manager import ( | ||||
|     SESSION_KEY_OVERRIDE_FLOW_TOKEN, | ||||
|     SESSION_KEY_SOURCE_FLOW_STAGES, | ||||
| ) | ||||
| from authentik.core.types import UILoginButton | ||||
| from authentik.enterprise.stages.source.models import SourceStage | ||||
| from authentik.flows.challenge import Challenge, ChallengeResponse | ||||
| from authentik.flows.models import FlowToken | ||||
| from authentik.flows.models import FlowToken, in_memory_stage | ||||
| from authentik.flows.planner import PLAN_CONTEXT_IS_RESTORED | ||||
| from authentik.flows.stage import ChallengeStageView | ||||
| from authentik.flows.stage import ChallengeStageView, StageView | ||||
| from authentik.lib.utils.time import timedelta_from_string | ||||
|  | ||||
| PLAN_CONTEXT_RESUME_TOKEN = "resume_token"  # nosec | ||||
| @ -49,6 +52,7 @@ class SourceStageView(ChallengeStageView): | ||||
|     def get_challenge(self, *args, **kwargs) -> Challenge: | ||||
|         resume_token = self.create_flow_token() | ||||
|         self.request.session[SESSION_KEY_OVERRIDE_FLOW_TOKEN] = resume_token | ||||
|         self.request.session[SESSION_KEY_SOURCE_FLOW_STAGES] = [in_memory_stage(SourceStageFinal)] | ||||
|         return self.login_button.challenge | ||||
|  | ||||
|     def create_flow_token(self) -> FlowToken: | ||||
| @ -77,3 +81,19 @@ class SourceStageView(ChallengeStageView): | ||||
|  | ||||
|     def challenge_valid(self, response: ChallengeResponse) -> HttpResponse: | ||||
|         return self.executor.stage_ok() | ||||
|  | ||||
|  | ||||
| class SourceStageFinal(StageView): | ||||
|     """Dynamic stage injected in the source flow manager. This is injected in the | ||||
|     flow the source flow manager picks (authentication or enrollment), and will run at the end. | ||||
|     This stage uses the override flow token to resume execution of the initial flow the | ||||
|     source stage is bound to.""" | ||||
|  | ||||
|     def dispatch(self): | ||||
|         token: FlowToken = self.request.session.get(SESSION_KEY_OVERRIDE_FLOW_TOKEN) | ||||
|         self._logger.info("Replacing source flow with overridden flow", flow=token.flow.slug) | ||||
|         plan = token.plan | ||||
|         plan.context[PLAN_CONTEXT_IS_RESTORED] = token | ||||
|         response = plan.to_redirect(self.request, token.flow) | ||||
|         token.delete() | ||||
|         return response | ||||
|  | ||||
| @ -19,7 +19,6 @@ from authentik.core.api.used_by import UsedByMixin | ||||
| from authentik.core.api.utils import JSONDictField, ModelSerializer, PassiveSerializer | ||||
| from authentik.core.models import Provider | ||||
| from authentik.enterprise.license import LicenseKey | ||||
| from authentik.enterprise.providers.rac.models import RACProvider | ||||
| from authentik.lib.utils.time import timedelta_from_string, timedelta_string_validator | ||||
| from authentik.outposts.api.service_connections import ServiceConnectionSerializer | ||||
| from authentik.outposts.apps import MANAGED_OUTPOST, MANAGED_OUTPOST_NAME | ||||
| @ -31,6 +30,7 @@ from authentik.outposts.models import ( | ||||
| ) | ||||
| from authentik.providers.ldap.models import LDAPProvider | ||||
| from authentik.providers.proxy.models import ProxyProvider | ||||
| from authentik.providers.rac.models import RACProvider | ||||
| from authentik.providers.radius.models import RadiusProvider | ||||
|  | ||||
|  | ||||
|  | ||||
| @ -18,8 +18,6 @@ from kubernetes.config.kube_config import KUBE_CONFIG_DEFAULT_LOCATION | ||||
| from structlog.stdlib import get_logger | ||||
| from yaml import safe_load | ||||
|  | ||||
| from authentik.enterprise.providers.rac.controllers.docker import RACDockerController | ||||
| from authentik.enterprise.providers.rac.controllers.kubernetes import RACKubernetesController | ||||
| from authentik.events.models import TaskStatus | ||||
| from authentik.events.system_tasks import SystemTask, prefill_task | ||||
| from authentik.lib.config import CONFIG | ||||
| @ -41,6 +39,8 @@ from authentik.providers.ldap.controllers.docker import LDAPDockerController | ||||
| from authentik.providers.ldap.controllers.kubernetes import LDAPKubernetesController | ||||
| from authentik.providers.proxy.controllers.docker import ProxyDockerController | ||||
| from authentik.providers.proxy.controllers.kubernetes import ProxyKubernetesController | ||||
| from authentik.providers.rac.controllers.docker import RACDockerController | ||||
| from authentik.providers.rac.controllers.kubernetes import RACKubernetesController | ||||
| from authentik.providers.radius.controllers.docker import RadiusDockerController | ||||
| from authentik.providers.radius.controllers.kubernetes import RadiusKubernetesController | ||||
| from authentik.root.celery import CELERY_APP | ||||
|  | ||||
| @ -128,7 +128,7 @@ class GeoIPPolicy(Policy): | ||||
|                 (geoip_data["lat"], geoip_data["long"]), | ||||
|             ) | ||||
|             if self.check_history_distance and dist.km >= ( | ||||
|                 self.history_max_distance_km - self.distance_tolerance_km | ||||
|                 self.history_max_distance_km + self.distance_tolerance_km | ||||
|             ): | ||||
|                 return PolicyResult( | ||||
|                     False, _("Distance from previous authentication is larger than threshold.") | ||||
| @ -139,7 +139,7 @@ class GeoIPPolicy(Policy): | ||||
|             # clamped to be at least 1 hour | ||||
|             rel_time_hours = max(int((_now - previous_login.created).total_seconds() / 3600), 1) | ||||
|             if self.check_impossible_travel and dist.km >= ( | ||||
|                 (MAX_DISTANCE_HOUR_KM * rel_time_hours) - self.distance_tolerance_km | ||||
|                 (MAX_DISTANCE_HOUR_KM * rel_time_hours) + self.distance_tolerance_km | ||||
|             ): | ||||
|                 return PolicyResult(False, _("Distance is further than possible.")) | ||||
|         return PolicyResult(True) | ||||
|  | ||||
| @ -148,10 +148,10 @@ class PasswordPolicy(Policy): | ||||
|             user_inputs.append(request.user.email) | ||||
|         if request.http_request: | ||||
|             user_inputs.append(request.http_request.brand.branding_title) | ||||
|         # Only calculate result for the first 100 characters, as with over 100 char | ||||
|         # Only calculate result for the first 72 characters, as with over 100 char | ||||
|         # long passwords we can be reasonably sure that they'll surpass the score anyways | ||||
|         # See https://github.com/dropbox/zxcvbn#runtime-latency | ||||
|         results = zxcvbn(password[:100], user_inputs) | ||||
|         results = zxcvbn(password[:72], user_inputs) | ||||
|         LOGGER.debug("password failed", check="zxcvbn", score=results["score"]) | ||||
|         result = PolicyResult(results["score"] > self.zxcvbn_score_threshold) | ||||
|         if not result.passing: | ||||
|  | ||||
| @ -6,13 +6,12 @@ from rest_framework.viewsets import GenericViewSet | ||||
| from authentik.core.api.groups import GroupMemberSerializer | ||||
| from authentik.core.api.used_by import UsedByMixin | ||||
| from authentik.core.api.utils import ModelSerializer | ||||
| from authentik.enterprise.api import EnterpriseRequiredMixin | ||||
| from authentik.enterprise.providers.rac.api.endpoints import EndpointSerializer | ||||
| from authentik.enterprise.providers.rac.api.providers import RACProviderSerializer | ||||
| from authentik.enterprise.providers.rac.models import ConnectionToken | ||||
| from authentik.providers.rac.api.endpoints import EndpointSerializer | ||||
| from authentik.providers.rac.api.providers import RACProviderSerializer | ||||
| from authentik.providers.rac.models import ConnectionToken | ||||
| 
 | ||||
| 
 | ||||
| class ConnectionTokenSerializer(EnterpriseRequiredMixin, ModelSerializer): | ||||
| class ConnectionTokenSerializer(ModelSerializer): | ||||
|     """ConnectionToken Serializer""" | ||||
| 
 | ||||
|     provider_obj = RACProviderSerializer(source="provider", read_only=True) | ||||
| @ -14,10 +14,9 @@ from structlog.stdlib import get_logger | ||||
| from authentik.core.api.used_by import UsedByMixin | ||||
| from authentik.core.api.utils import ModelSerializer | ||||
| from authentik.core.models import Provider | ||||
| from authentik.enterprise.api import EnterpriseRequiredMixin | ||||
| from authentik.enterprise.providers.rac.api.providers import RACProviderSerializer | ||||
| from authentik.enterprise.providers.rac.models import Endpoint | ||||
| from authentik.policies.engine import PolicyEngine | ||||
| from authentik.providers.rac.api.providers import RACProviderSerializer | ||||
| from authentik.providers.rac.models import Endpoint | ||||
| from authentik.rbac.filters import ObjectFilter | ||||
| 
 | ||||
| LOGGER = get_logger() | ||||
| @ -28,7 +27,7 @@ def user_endpoint_cache_key(user_pk: str) -> str: | ||||
|     return f"goauthentik.io/providers/rac/endpoint_access/{user_pk}" | ||||
| 
 | ||||
| 
 | ||||
| class EndpointSerializer(EnterpriseRequiredMixin, ModelSerializer): | ||||
| class EndpointSerializer(ModelSerializer): | ||||
|     """Endpoint Serializer""" | ||||
| 
 | ||||
|     provider_obj = RACProviderSerializer(source="provider", read_only=True) | ||||
| @ -10,7 +10,7 @@ from rest_framework.viewsets import ModelViewSet | ||||
| from authentik.core.api.property_mappings import PropertyMappingSerializer | ||||
| from authentik.core.api.used_by import UsedByMixin | ||||
| from authentik.core.api.utils import JSONDictField | ||||
| from authentik.enterprise.providers.rac.models import RACPropertyMapping | ||||
| from authentik.providers.rac.models import RACPropertyMapping | ||||
| 
 | ||||
| 
 | ||||
| class RACPropertyMappingSerializer(PropertyMappingSerializer): | ||||
| @ -5,11 +5,10 @@ from rest_framework.viewsets import ModelViewSet | ||||
| 
 | ||||
| from authentik.core.api.providers import ProviderSerializer | ||||
| from authentik.core.api.used_by import UsedByMixin | ||||
| from authentik.enterprise.api import EnterpriseRequiredMixin | ||||
| from authentik.enterprise.providers.rac.models import RACProvider | ||||
| from authentik.providers.rac.models import RACProvider | ||||
| 
 | ||||
| 
 | ||||
| class RACProviderSerializer(EnterpriseRequiredMixin, ProviderSerializer): | ||||
| class RACProviderSerializer(ProviderSerializer): | ||||
|     """RACProvider Serializer""" | ||||
| 
 | ||||
|     outpost_set = ListField(child=CharField(), read_only=True, source="outpost_set.all") | ||||
							
								
								
									
										14
									
								
								authentik/providers/rac/apps.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								authentik/providers/rac/apps.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | ||||
| """RAC app config""" | ||||
|  | ||||
| from django.apps import AppConfig | ||||
|  | ||||
|  | ||||
| class AuthentikProviderRAC(AppConfig): | ||||
|     """authentik rac app config""" | ||||
|  | ||||
|     name = "authentik.providers.rac" | ||||
|     label = "authentik_providers_rac" | ||||
|     verbose_name = "authentik Providers.RAC" | ||||
|     default = True | ||||
|     mountpoint = "" | ||||
|     ws_mountpoint = "authentik.providers.rac.urls" | ||||
| @ -7,22 +7,22 @@ from channels.generic.websocket import AsyncWebsocketConsumer | ||||
| from django.http.request import QueryDict | ||||
| from structlog.stdlib import BoundLogger, get_logger | ||||
| 
 | ||||
| from authentik.enterprise.providers.rac.models import ConnectionToken, RACProvider | ||||
| from authentik.outposts.consumer import OUTPOST_GROUP_INSTANCE | ||||
| from authentik.outposts.models import Outpost, OutpostState, OutpostType | ||||
| from authentik.providers.rac.models import ConnectionToken, RACProvider | ||||
| 
 | ||||
| # Global broadcast group, which messages are sent to when the outpost connects back | ||||
| # to authentik for a specific connection | ||||
| # The `RACClientConsumer` consumer adds itself to this group on connection, | ||||
| # and removes itself once it has been assigned a specific outpost channel | ||||
| RAC_CLIENT_GROUP = "group_enterprise_rac_client" | ||||
| RAC_CLIENT_GROUP = "group_rac_client" | ||||
| # A group for all connections in a given authentik session ID | ||||
| # A disconnect message is sent to this group when the session expires/is deleted | ||||
| RAC_CLIENT_GROUP_SESSION = "group_enterprise_rac_client_%(session)s" | ||||
| RAC_CLIENT_GROUP_SESSION = "group_rac_client_%(session)s" | ||||
| # A group for all connections with a specific token, which in almost all cases | ||||
| # is just one connection, however this is used to disconnect the connection | ||||
| # when the token is deleted | ||||
| RAC_CLIENT_GROUP_TOKEN = "group_enterprise_rac_token_%(token)s"  # nosec | ||||
| RAC_CLIENT_GROUP_TOKEN = "group_rac_token_%(token)s"  # nosec | ||||
| 
 | ||||
| # Step 1: Client connects to this websocket endpoint | ||||
| # Step 2: We prepare all the connection args for Guac | ||||
| @ -3,7 +3,7 @@ | ||||
| from channels.exceptions import ChannelFull | ||||
| from channels.generic.websocket import AsyncWebsocketConsumer | ||||
| 
 | ||||
| from authentik.enterprise.providers.rac.consumer_client import RAC_CLIENT_GROUP | ||||
| from authentik.providers.rac.consumer_client import RAC_CLIENT_GROUP | ||||
| 
 | ||||
| 
 | ||||
| class RACOutpostConsumer(AsyncWebsocketConsumer): | ||||
| @ -74,7 +74,7 @@ class RACProvider(Provider): | ||||
| 
 | ||||
|     @property | ||||
|     def serializer(self) -> type[Serializer]: | ||||
|         from authentik.enterprise.providers.rac.api.providers import RACProviderSerializer | ||||
|         from authentik.providers.rac.api.providers import RACProviderSerializer | ||||
| 
 | ||||
|         return RACProviderSerializer | ||||
| 
 | ||||
| @ -100,7 +100,7 @@ class Endpoint(SerializerModel, PolicyBindingModel): | ||||
| 
 | ||||
|     @property | ||||
|     def serializer(self) -> type[Serializer]: | ||||
|         from authentik.enterprise.providers.rac.api.endpoints import EndpointSerializer | ||||
|         from authentik.providers.rac.api.endpoints import EndpointSerializer | ||||
| 
 | ||||
|         return EndpointSerializer | ||||
| 
 | ||||
| @ -129,7 +129,7 @@ class RACPropertyMapping(PropertyMapping): | ||||
| 
 | ||||
|     @property | ||||
|     def serializer(self) -> type[Serializer]: | ||||
|         from authentik.enterprise.providers.rac.api.property_mappings import ( | ||||
|         from authentik.providers.rac.api.property_mappings import ( | ||||
|             RACPropertyMappingSerializer, | ||||
|         ) | ||||
| 
 | ||||
| @ -10,12 +10,12 @@ from django.dispatch import receiver | ||||
| from django.http import HttpRequest | ||||
| 
 | ||||
| from authentik.core.models import User | ||||
| from authentik.enterprise.providers.rac.api.endpoints import user_endpoint_cache_key | ||||
| from authentik.enterprise.providers.rac.consumer_client import ( | ||||
| from authentik.providers.rac.api.endpoints import user_endpoint_cache_key | ||||
| from authentik.providers.rac.consumer_client import ( | ||||
|     RAC_CLIENT_GROUP_SESSION, | ||||
|     RAC_CLIENT_GROUP_TOKEN, | ||||
| ) | ||||
| from authentik.enterprise.providers.rac.models import ConnectionToken, Endpoint | ||||
| from authentik.providers.rac.models import ConnectionToken, Endpoint | ||||
| 
 | ||||
| 
 | ||||
| @receiver(user_logged_out) | ||||
| @ -3,7 +3,7 @@ | ||||
| {% load authentik_core %} | ||||
| 
 | ||||
| {% block head %} | ||||
| <script src="{% versioned_script 'dist/enterprise/rac/index-%v.js' %}" type="module"></script> | ||||
| <script src="{% versioned_script 'dist/rac/index-%v.js' %}" type="module"></script> | ||||
| <meta name="theme-color" content="#18191a" media="(prefers-color-scheme: dark)"> | ||||
| <meta name="theme-color" content="#ffffff" media="(prefers-color-scheme: light)"> | ||||
| <link rel="icon" href="{{ tenant.branding_favicon_url }}"> | ||||
| @ -1,16 +1,9 @@ | ||||
| """Test RAC Provider""" | ||||
| 
 | ||||
| from datetime import timedelta | ||||
| from time import mktime | ||||
| from unittest.mock import MagicMock, patch | ||||
| 
 | ||||
| from django.urls import reverse | ||||
| from django.utils.timezone import now | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from authentik.core.tests.utils import create_test_admin_user, create_test_flow | ||||
| from authentik.enterprise.license import LicenseKey | ||||
| from authentik.enterprise.models import License | ||||
| from authentik.lib.generators import generate_id | ||||
| 
 | ||||
| 
 | ||||
| @ -20,21 +13,8 @@ class TestAPI(APITestCase): | ||||
|     def setUp(self) -> None: | ||||
|         self.user = create_test_admin_user() | ||||
| 
 | ||||
|     @patch( | ||||
|         "authentik.enterprise.license.LicenseKey.validate", | ||||
|         MagicMock( | ||||
|             return_value=LicenseKey( | ||||
|                 aud="", | ||||
|                 exp=int(mktime((now() + timedelta(days=3000)).timetuple())), | ||||
|                 name=generate_id(), | ||||
|                 internal_users=100, | ||||
|                 external_users=100, | ||||
|             ) | ||||
|         ), | ||||
|     ) | ||||
|     def test_create(self): | ||||
|         """Test creation of RAC Provider""" | ||||
|         License.objects.create(key=generate_id()) | ||||
|         self.client.force_login(self.user) | ||||
|         response = self.client.post( | ||||
|             reverse("authentik_api:racprovider-list"), | ||||
| @ -5,10 +5,10 @@ from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from authentik.core.models import Application | ||||
| from authentik.core.tests.utils import create_test_admin_user | ||||
| from authentik.enterprise.providers.rac.models import Endpoint, Protocols, RACProvider | ||||
| from authentik.lib.generators import generate_id | ||||
| from authentik.policies.dummy.models import DummyPolicy | ||||
| from authentik.policies.models import PolicyBinding | ||||
| from authentik.providers.rac.models import Endpoint, Protocols, RACProvider | ||||
| 
 | ||||
| 
 | ||||
| class TestEndpointsAPI(APITestCase): | ||||
| @ -4,14 +4,14 @@ from django.test import TransactionTestCase | ||||
| 
 | ||||
| from authentik.core.models import Application, AuthenticatedSession | ||||
| from authentik.core.tests.utils import create_test_admin_user | ||||
| from authentik.enterprise.providers.rac.models import ( | ||||
| from authentik.lib.generators import generate_id | ||||
| from authentik.providers.rac.models import ( | ||||
|     ConnectionToken, | ||||
|     Endpoint, | ||||
|     Protocols, | ||||
|     RACPropertyMapping, | ||||
|     RACProvider, | ||||
| ) | ||||
| from authentik.lib.generators import generate_id | ||||
| 
 | ||||
| 
 | ||||
| class TestModels(TransactionTestCase): | ||||
| @ -1,23 +1,17 @@ | ||||
| """RAC Views tests""" | ||||
| 
 | ||||
| from datetime import timedelta | ||||
| from json import loads | ||||
| from time import mktime | ||||
| from unittest.mock import MagicMock, patch | ||||
| 
 | ||||
| from django.urls import reverse | ||||
| from django.utils.timezone import now | ||||
| from rest_framework.test import APITestCase | ||||
| 
 | ||||
| from authentik.core.models import Application | ||||
| from authentik.core.tests.utils import create_test_admin_user, create_test_flow | ||||
| from authentik.enterprise.license import LicenseKey | ||||
| from authentik.enterprise.models import License | ||||
| from authentik.enterprise.providers.rac.models import Endpoint, Protocols, RACProvider | ||||
| from authentik.lib.generators import generate_id | ||||
| from authentik.policies.denied import AccessDeniedResponse | ||||
| from authentik.policies.dummy.models import DummyPolicy | ||||
| from authentik.policies.models import PolicyBinding | ||||
| from authentik.providers.rac.models import Endpoint, Protocols, RACProvider | ||||
| 
 | ||||
| 
 | ||||
| class TestRACViews(APITestCase): | ||||
| @ -39,21 +33,8 @@ class TestRACViews(APITestCase): | ||||
|             provider=self.provider, | ||||
|         ) | ||||
| 
 | ||||
|     @patch( | ||||
|         "authentik.enterprise.license.LicenseKey.validate", | ||||
|         MagicMock( | ||||
|             return_value=LicenseKey( | ||||
|                 aud="", | ||||
|                 exp=int(mktime((now() + timedelta(days=3000)).timetuple())), | ||||
|                 name=generate_id(), | ||||
|                 internal_users=100, | ||||
|                 external_users=100, | ||||
|             ) | ||||
|         ), | ||||
|     ) | ||||
|     def test_no_policy(self): | ||||
|         """Test request""" | ||||
|         License.objects.create(key=generate_id()) | ||||
|         self.client.force_login(self.user) | ||||
|         response = self.client.get( | ||||
|             reverse( | ||||
| @ -70,18 +51,6 @@ class TestRACViews(APITestCase): | ||||
|         final_response = self.client.get(next_url) | ||||
|         self.assertEqual(final_response.status_code, 200) | ||||
| 
 | ||||
|     @patch( | ||||
|         "authentik.enterprise.license.LicenseKey.validate", | ||||
|         MagicMock( | ||||
|             return_value=LicenseKey( | ||||
|                 aud="", | ||||
|                 exp=int(mktime((now() + timedelta(days=3000)).timetuple())), | ||||
|                 name=generate_id(), | ||||
|                 internal_users=100, | ||||
|                 external_users=100, | ||||
|             ) | ||||
|         ), | ||||
|     ) | ||||
|     def test_app_deny(self): | ||||
|         """Test request (deny on app level)""" | ||||
|         PolicyBinding.objects.create( | ||||
| @ -89,7 +58,6 @@ class TestRACViews(APITestCase): | ||||
|             policy=DummyPolicy.objects.create(name="deny", result=False, wait_min=1, wait_max=2), | ||||
|             order=0, | ||||
|         ) | ||||
|         License.objects.create(key=generate_id()) | ||||
|         self.client.force_login(self.user) | ||||
|         response = self.client.get( | ||||
|             reverse( | ||||
| @ -99,18 +67,6 @@ class TestRACViews(APITestCase): | ||||
|         ) | ||||
|         self.assertIsInstance(response, AccessDeniedResponse) | ||||
| 
 | ||||
|     @patch( | ||||
|         "authentik.enterprise.license.LicenseKey.validate", | ||||
|         MagicMock( | ||||
|             return_value=LicenseKey( | ||||
|                 aud="", | ||||
|                 exp=int(mktime((now() + timedelta(days=3000)).timetuple())), | ||||
|                 name=generate_id(), | ||||
|                 internal_users=100, | ||||
|                 external_users=100, | ||||
|             ) | ||||
|         ), | ||||
|     ) | ||||
|     def test_endpoint_deny(self): | ||||
|         """Test request (deny on endpoint level)""" | ||||
|         PolicyBinding.objects.create( | ||||
| @ -118,7 +74,6 @@ class TestRACViews(APITestCase): | ||||
|             policy=DummyPolicy.objects.create(name="deny", result=False, wait_min=1, wait_max=2), | ||||
|             order=0, | ||||
|         ) | ||||
|         License.objects.create(key=generate_id()) | ||||
|         self.client.force_login(self.user) | ||||
|         response = self.client.get( | ||||
|             reverse( | ||||
| @ -4,14 +4,14 @@ from channels.auth import AuthMiddleware | ||||
| from channels.sessions import CookieMiddleware | ||||
| from django.urls import path | ||||
| 
 | ||||
| from authentik.enterprise.providers.rac.api.connection_tokens import ConnectionTokenViewSet | ||||
| from authentik.enterprise.providers.rac.api.endpoints import EndpointViewSet | ||||
| from authentik.enterprise.providers.rac.api.property_mappings import RACPropertyMappingViewSet | ||||
| from authentik.enterprise.providers.rac.api.providers import RACProviderViewSet | ||||
| from authentik.enterprise.providers.rac.consumer_client import RACClientConsumer | ||||
| from authentik.enterprise.providers.rac.consumer_outpost import RACOutpostConsumer | ||||
| from authentik.enterprise.providers.rac.views import RACInterface, RACStartView | ||||
| from authentik.outposts.channels import TokenOutpostMiddleware | ||||
| from authentik.providers.rac.api.connection_tokens import ConnectionTokenViewSet | ||||
| from authentik.providers.rac.api.endpoints import EndpointViewSet | ||||
| from authentik.providers.rac.api.property_mappings import RACPropertyMappingViewSet | ||||
| from authentik.providers.rac.api.providers import RACProviderViewSet | ||||
| from authentik.providers.rac.consumer_client import RACClientConsumer | ||||
| from authentik.providers.rac.consumer_outpost import RACOutpostConsumer | ||||
| from authentik.providers.rac.views import RACInterface, RACStartView | ||||
| from authentik.root.asgi_middleware import SessionMiddleware | ||||
| from authentik.root.middleware import ChannelsLoggingMiddleware | ||||
| 
 | ||||
| @ -10,8 +10,6 @@ from django.utils.translation import gettext as _ | ||||
| 
 | ||||
| from authentik.core.models import Application, AuthenticatedSession | ||||
| from authentik.core.views.interface import InterfaceView | ||||
| from authentik.enterprise.policy import EnterprisePolicyAccessView | ||||
| from authentik.enterprise.providers.rac.models import ConnectionToken, Endpoint, RACProvider | ||||
| from authentik.events.models import Event, EventAction | ||||
| from authentik.flows.challenge import RedirectChallenge | ||||
| from authentik.flows.exceptions import FlowNonApplicableException | ||||
| @ -20,9 +18,11 @@ from authentik.flows.planner import PLAN_CONTEXT_APPLICATION, FlowPlanner | ||||
| from authentik.flows.stage import RedirectStage | ||||
| from authentik.lib.utils.time import timedelta_from_string | ||||
| from authentik.policies.engine import PolicyEngine | ||||
| from authentik.policies.views import PolicyAccessView | ||||
| from authentik.providers.rac.models import ConnectionToken, Endpoint, RACProvider | ||||
| 
 | ||||
| 
 | ||||
| class RACStartView(EnterprisePolicyAccessView): | ||||
| class RACStartView(PolicyAccessView): | ||||
|     """Start a RAC connection by checking access and creating a connection token""" | ||||
| 
 | ||||
|     endpoint: Endpoint | ||||
| @ -2,7 +2,7 @@ | ||||
|  | ||||
| from django.apps import apps | ||||
| from django.contrib.auth.models import Permission | ||||
| from django.db.models import Q, QuerySet | ||||
| from django.db.models import QuerySet | ||||
| from django_filters.filters import ModelChoiceFilter | ||||
| from django_filters.filterset import FilterSet | ||||
| from django_filters.rest_framework import DjangoFilterBackend | ||||
| @ -18,7 +18,6 @@ from rest_framework.filters import OrderingFilter, SearchFilter | ||||
| from rest_framework.permissions import IsAuthenticated | ||||
| from rest_framework.viewsets import ReadOnlyModelViewSet | ||||
|  | ||||
| from authentik.blueprints.v1.importer import excluded_models | ||||
| from authentik.core.api.utils import ModelSerializer, PassiveSerializer | ||||
| from authentik.core.models import User | ||||
| from authentik.lib.validators import RequiredTogetherValidator | ||||
| @ -106,13 +105,13 @@ class RBACPermissionViewSet(ReadOnlyModelViewSet): | ||||
|     ] | ||||
|  | ||||
|     def get_queryset(self) -> QuerySet: | ||||
|         query = Q() | ||||
|         for model in excluded_models(): | ||||
|             query |= Q( | ||||
|                 content_type__app_label=model._meta.app_label, | ||||
|                 content_type__model=model._meta.model_name, | ||||
|         return ( | ||||
|             Permission.objects.all() | ||||
|             .select_related("content_type") | ||||
|             .filter( | ||||
|                 content_type__app_label__startswith="authentik", | ||||
|             ) | ||||
|         return Permission.objects.all().select_related("content_type").exclude(query) | ||||
|         ) | ||||
|  | ||||
|  | ||||
| class PermissionAssignSerializer(PassiveSerializer): | ||||
|  | ||||
| @ -87,6 +87,7 @@ TENANT_APPS = [ | ||||
|     "authentik.providers.ldap", | ||||
|     "authentik.providers.oauth2", | ||||
|     "authentik.providers.proxy", | ||||
|     "authentik.providers.rac", | ||||
|     "authentik.providers.radius", | ||||
|     "authentik.providers.saml", | ||||
|     "authentik.providers.scim", | ||||
|  | ||||
| @ -2,6 +2,7 @@ | ||||
|  | ||||
| from typing import Any | ||||
|  | ||||
| from requests import RequestException | ||||
| from structlog.stdlib import get_logger | ||||
|  | ||||
| from authentik.sources.oauth.clients.oauth2 import UserprofileHeaderAuthClient | ||||
| @ -21,10 +22,35 @@ class AzureADOAuthRedirect(OAuthRedirect): | ||||
|         } | ||||
|  | ||||
|  | ||||
| class AzureADClient(UserprofileHeaderAuthClient): | ||||
|     """Fetch AzureAD group information""" | ||||
|  | ||||
|     def get_profile_info(self, token): | ||||
|         profile_data = super().get_profile_info(token) | ||||
|         if "https://graph.microsoft.com/GroupMember.Read.All" not in self.source.additional_scopes: | ||||
|             return profile_data | ||||
|         group_response = self.session.request( | ||||
|             "get", | ||||
|             "https://graph.microsoft.com/v1.0/me/memberOf", | ||||
|             headers={"Authorization": f"{token['token_type']} {token['access_token']}"}, | ||||
|         ) | ||||
|         try: | ||||
|             group_response.raise_for_status() | ||||
|         except RequestException as exc: | ||||
|             LOGGER.warning( | ||||
|                 "Unable to fetch user profile", | ||||
|                 exc=exc, | ||||
|                 response=exc.response.text if exc.response else str(exc), | ||||
|             ) | ||||
|             return None | ||||
|         profile_data["raw_groups"] = group_response.json() | ||||
|         return profile_data | ||||
|  | ||||
|  | ||||
| class AzureADOAuthCallback(OpenIDConnectOAuth2Callback): | ||||
|     """AzureAD OAuth2 Callback""" | ||||
|  | ||||
|     client_class = UserprofileHeaderAuthClient | ||||
|     client_class = AzureADClient | ||||
|  | ||||
|     def get_user_id(self, info: dict[str, str]) -> str: | ||||
|         # Default try to get `id` for the Graph API endpoint | ||||
| @ -53,8 +79,24 @@ class AzureADType(SourceType): | ||||
|  | ||||
|     def get_base_user_properties(self, info: dict[str, Any], **kwargs) -> dict[str, Any]: | ||||
|         mail = info.get("mail", None) or info.get("otherMails", [None])[0] | ||||
|         # Format group info | ||||
|         groups = [] | ||||
|         group_id_dict = {} | ||||
|         for group in info.get("raw_groups", {}).get("value", []): | ||||
|             if group["@odata.type"] != "#microsoft.graph.group": | ||||
|                 continue | ||||
|             groups.append(group["id"]) | ||||
|             group_id_dict[group["id"]] = group | ||||
|         info["raw_groups"] = group_id_dict | ||||
|         return { | ||||
|             "username": info.get("userPrincipalName"), | ||||
|             "email": mail, | ||||
|             "name": info.get("displayName"), | ||||
|             "groups": groups, | ||||
|         } | ||||
|  | ||||
|     def get_base_group_properties(self, source, group_id, **kwargs): | ||||
|         raw_group = kwargs["info"]["raw_groups"][group_id] | ||||
|         return { | ||||
|             "name": raw_group["displayName"], | ||||
|         } | ||||
|  | ||||
| @ -300,9 +300,11 @@ class TestAuthenticatorEmailStage(FlowTestCase): | ||||
|             ) | ||||
|             self.assertEqual(response.status_code, 200) | ||||
|             self.assertTrue(device.confirmed) | ||||
|             # Session key should be removed after device is saved | ||||
|             device.save() | ||||
|             self.assertNotIn(SESSION_KEY_EMAIL_DEVICE, self.client.session) | ||||
|             # Get a fresh session to check if the key was removed | ||||
|             session = self.client.session | ||||
|             session.save() | ||||
|             session.load() | ||||
|             self.assertNotIn(SESSION_KEY_EMAIL_DEVICE, session) | ||||
|  | ||||
|     def test_model_properties_and_methods(self): | ||||
|         """Test model properties""" | ||||
|  | ||||
| @ -12,6 +12,7 @@ from structlog.stdlib import get_logger | ||||
|  | ||||
| from authentik.events.models import Event, EventAction, TaskStatus | ||||
| from authentik.events.system_tasks import SystemTask | ||||
| from authentik.lib.utils.reflection import class_to_path, path_to_class | ||||
| from authentik.root.celery import CELERY_APP | ||||
| from authentik.stages.authenticator_email.models import AuthenticatorEmailStage | ||||
| from authentik.stages.email.models import EmailStage | ||||
| @ -32,9 +33,10 @@ def send_mails( | ||||
|         Celery group promise for the email sending tasks | ||||
|     """ | ||||
|     tasks = [] | ||||
|     stage_class = stage.__class__ | ||||
|     # Use the class path instead of the class itself for serialization | ||||
|     stage_class_path = class_to_path(stage.__class__) | ||||
|     for message in messages: | ||||
|         tasks.append(send_mail.s(message.__dict__, stage_class, str(stage.pk))) | ||||
|         tasks.append(send_mail.s(message.__dict__, stage_class_path, str(stage.pk))) | ||||
|     lazy_group = group(*tasks) | ||||
|     promise = lazy_group() | ||||
|     return promise | ||||
| @ -61,7 +63,7 @@ def get_email_body(email: EmailMultiAlternatives) -> str: | ||||
| def send_mail( | ||||
|     self: SystemTask, | ||||
|     message: dict[Any, Any], | ||||
|     stage_class: EmailStage | AuthenticatorEmailStage = EmailStage, | ||||
|     stage_class_path: str | None = None, | ||||
|     email_stage_pk: str | None = None, | ||||
| ): | ||||
|     """Send Email for Email Stage. Retries are scheduled automatically.""" | ||||
| @ -69,9 +71,10 @@ def send_mail( | ||||
|     message_id = make_msgid(domain=DNS_NAME) | ||||
|     self.set_uid(slugify(message_id.replace(".", "_").replace("@", "_"))) | ||||
|     try: | ||||
|         if not email_stage_pk: | ||||
|             stage: EmailStage | AuthenticatorEmailStage = stage_class(use_global_settings=True) | ||||
|         if not stage_class_path or not email_stage_pk: | ||||
|             stage = EmailStage(use_global_settings=True) | ||||
|         else: | ||||
|             stage_class = path_to_class(stage_class_path) | ||||
|             stages = stage_class.objects.filter(pk=email_stage_pk) | ||||
|             if not stages.exists(): | ||||
|                 self.set_status( | ||||
|  | ||||
							
								
								
									
										58
									
								
								authentik/stages/email/tests/test_tasks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								authentik/stages/email/tests/test_tasks.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,58 @@ | ||||
| """Test email stage tasks""" | ||||
|  | ||||
| from unittest.mock import patch | ||||
|  | ||||
| from django.core.mail import EmailMultiAlternatives | ||||
| from django.test import TestCase | ||||
|  | ||||
| from authentik.core.tests.utils import create_test_admin_user | ||||
| from authentik.lib.utils.reflection import class_to_path | ||||
| from authentik.stages.authenticator_email.models import AuthenticatorEmailStage | ||||
| from authentik.stages.email.models import EmailStage | ||||
| from authentik.stages.email.tasks import get_email_body, send_mails | ||||
|  | ||||
|  | ||||
| class TestEmailTasks(TestCase): | ||||
|     """Test email stage tasks""" | ||||
|  | ||||
|     def setUp(self): | ||||
|         self.user = create_test_admin_user() | ||||
|         self.stage = EmailStage.objects.create( | ||||
|             name="test-email", | ||||
|             use_global_settings=True, | ||||
|         ) | ||||
|         self.auth_stage = AuthenticatorEmailStage.objects.create( | ||||
|             name="test-auth-email", | ||||
|             use_global_settings=True, | ||||
|         ) | ||||
|  | ||||
|     def test_get_email_body_html(self): | ||||
|         """Test get_email_body with HTML alternative""" | ||||
|         message = EmailMultiAlternatives() | ||||
|         message.body = "plain text" | ||||
|         message.attach_alternative("<p>html content</p>", "text/html") | ||||
|         self.assertEqual(get_email_body(message), "<p>html content</p>") | ||||
|  | ||||
|     def test_get_email_body_plain(self): | ||||
|         """Test get_email_body with plain text only""" | ||||
|         message = EmailMultiAlternatives() | ||||
|         message.body = "plain text" | ||||
|         self.assertEqual(get_email_body(message), "plain text") | ||||
|  | ||||
|     def test_send_mails_email_stage(self): | ||||
|         """Test send_mails with EmailStage""" | ||||
|         message = EmailMultiAlternatives() | ||||
|         with patch("authentik.stages.email.tasks.send_mail") as mock_send: | ||||
|             send_mails(self.stage, message) | ||||
|             mock_send.s.assert_called_once_with( | ||||
|                 message.__dict__, class_to_path(EmailStage), str(self.stage.pk) | ||||
|             ) | ||||
|  | ||||
|     def test_send_mails_authenticator_stage(self): | ||||
|         """Test send_mails with AuthenticatorEmailStage""" | ||||
|         message = EmailMultiAlternatives() | ||||
|         with patch("authentik.stages.email.tasks.send_mail") as mock_send: | ||||
|             send_mails(self.auth_stage, message) | ||||
|             mock_send.s.assert_called_once_with( | ||||
|                 message.__dict__, class_to_path(AuthenticatorEmailStage), str(self.auth_stage.pk) | ||||
|             ) | ||||
| @ -2,7 +2,7 @@ | ||||
|     "$schema": "http://json-schema.org/draft-07/schema", | ||||
|     "$id": "https://goauthentik.io/blueprints/schema.json", | ||||
|     "type": "object", | ||||
|     "title": "authentik 2024.12.3 Blueprint schema", | ||||
|     "title": "authentik 2025.2.1 Blueprint schema", | ||||
|     "required": [ | ||||
|         "version", | ||||
|         "entries" | ||||
| @ -801,6 +801,126 @@ | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
|                             "model", | ||||
|                             "identifiers" | ||||
|                         ], | ||||
|                         "properties": { | ||||
|                             "model": { | ||||
|                                 "const": "authentik_providers_rac.racprovider" | ||||
|                             }, | ||||
|                             "id": { | ||||
|                                 "type": "string" | ||||
|                             }, | ||||
|                             "state": { | ||||
|                                 "type": "string", | ||||
|                                 "enum": [ | ||||
|                                     "absent", | ||||
|                                     "present", | ||||
|                                     "created", | ||||
|                                     "must_created" | ||||
|                                 ], | ||||
|                                 "default": "present" | ||||
|                             }, | ||||
|                             "conditions": { | ||||
|                                 "type": "array", | ||||
|                                 "items": { | ||||
|                                     "type": "boolean" | ||||
|                                 } | ||||
|                             }, | ||||
|                             "permissions": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racprovider_permissions" | ||||
|                             }, | ||||
|                             "attrs": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racprovider" | ||||
|                             }, | ||||
|                             "identifiers": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racprovider" | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
|                             "model", | ||||
|                             "identifiers" | ||||
|                         ], | ||||
|                         "properties": { | ||||
|                             "model": { | ||||
|                                 "const": "authentik_providers_rac.endpoint" | ||||
|                             }, | ||||
|                             "id": { | ||||
|                                 "type": "string" | ||||
|                             }, | ||||
|                             "state": { | ||||
|                                 "type": "string", | ||||
|                                 "enum": [ | ||||
|                                     "absent", | ||||
|                                     "present", | ||||
|                                     "created", | ||||
|                                     "must_created" | ||||
|                                 ], | ||||
|                                 "default": "present" | ||||
|                             }, | ||||
|                             "conditions": { | ||||
|                                 "type": "array", | ||||
|                                 "items": { | ||||
|                                     "type": "boolean" | ||||
|                                 } | ||||
|                             }, | ||||
|                             "permissions": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.endpoint_permissions" | ||||
|                             }, | ||||
|                             "attrs": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.endpoint" | ||||
|                             }, | ||||
|                             "identifiers": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.endpoint" | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
|                             "model", | ||||
|                             "identifiers" | ||||
|                         ], | ||||
|                         "properties": { | ||||
|                             "model": { | ||||
|                                 "const": "authentik_providers_rac.racpropertymapping" | ||||
|                             }, | ||||
|                             "id": { | ||||
|                                 "type": "string" | ||||
|                             }, | ||||
|                             "state": { | ||||
|                                 "type": "string", | ||||
|                                 "enum": [ | ||||
|                                     "absent", | ||||
|                                     "present", | ||||
|                                     "created", | ||||
|                                     "must_created" | ||||
|                                 ], | ||||
|                                 "default": "present" | ||||
|                             }, | ||||
|                             "conditions": { | ||||
|                                 "type": "array", | ||||
|                                 "items": { | ||||
|                                     "type": "boolean" | ||||
|                                 } | ||||
|                             }, | ||||
|                             "permissions": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racpropertymapping_permissions" | ||||
|                             }, | ||||
|                             "attrs": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racpropertymapping" | ||||
|                             }, | ||||
|                             "identifiers": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racpropertymapping" | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
| @ -3561,126 +3681,6 @@ | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
|                             "model", | ||||
|                             "identifiers" | ||||
|                         ], | ||||
|                         "properties": { | ||||
|                             "model": { | ||||
|                                 "const": "authentik_providers_rac.racprovider" | ||||
|                             }, | ||||
|                             "id": { | ||||
|                                 "type": "string" | ||||
|                             }, | ||||
|                             "state": { | ||||
|                                 "type": "string", | ||||
|                                 "enum": [ | ||||
|                                     "absent", | ||||
|                                     "present", | ||||
|                                     "created", | ||||
|                                     "must_created" | ||||
|                                 ], | ||||
|                                 "default": "present" | ||||
|                             }, | ||||
|                             "conditions": { | ||||
|                                 "type": "array", | ||||
|                                 "items": { | ||||
|                                     "type": "boolean" | ||||
|                                 } | ||||
|                             }, | ||||
|                             "permissions": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racprovider_permissions" | ||||
|                             }, | ||||
|                             "attrs": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racprovider" | ||||
|                             }, | ||||
|                             "identifiers": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racprovider" | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
|                             "model", | ||||
|                             "identifiers" | ||||
|                         ], | ||||
|                         "properties": { | ||||
|                             "model": { | ||||
|                                 "const": "authentik_providers_rac.endpoint" | ||||
|                             }, | ||||
|                             "id": { | ||||
|                                 "type": "string" | ||||
|                             }, | ||||
|                             "state": { | ||||
|                                 "type": "string", | ||||
|                                 "enum": [ | ||||
|                                     "absent", | ||||
|                                     "present", | ||||
|                                     "created", | ||||
|                                     "must_created" | ||||
|                                 ], | ||||
|                                 "default": "present" | ||||
|                             }, | ||||
|                             "conditions": { | ||||
|                                 "type": "array", | ||||
|                                 "items": { | ||||
|                                     "type": "boolean" | ||||
|                                 } | ||||
|                             }, | ||||
|                             "permissions": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.endpoint_permissions" | ||||
|                             }, | ||||
|                             "attrs": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.endpoint" | ||||
|                             }, | ||||
|                             "identifiers": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.endpoint" | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
|                             "model", | ||||
|                             "identifiers" | ||||
|                         ], | ||||
|                         "properties": { | ||||
|                             "model": { | ||||
|                                 "const": "authentik_providers_rac.racpropertymapping" | ||||
|                             }, | ||||
|                             "id": { | ||||
|                                 "type": "string" | ||||
|                             }, | ||||
|                             "state": { | ||||
|                                 "type": "string", | ||||
|                                 "enum": [ | ||||
|                                     "absent", | ||||
|                                     "present", | ||||
|                                     "created", | ||||
|                                     "must_created" | ||||
|                                 ], | ||||
|                                 "default": "present" | ||||
|                             }, | ||||
|                             "conditions": { | ||||
|                                 "type": "array", | ||||
|                                 "items": { | ||||
|                                     "type": "boolean" | ||||
|                                 } | ||||
|                             }, | ||||
|                             "permissions": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racpropertymapping_permissions" | ||||
|                             }, | ||||
|                             "attrs": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racpropertymapping" | ||||
|                             }, | ||||
|                             "identifiers": { | ||||
|                                 "$ref": "#/$defs/model_authentik_providers_rac.racpropertymapping" | ||||
|                             } | ||||
|                         } | ||||
|                     }, | ||||
|                     { | ||||
|                         "type": "object", | ||||
|                         "required": [ | ||||
| @ -4663,6 +4663,7 @@ | ||||
|                         "authentik.providers.ldap", | ||||
|                         "authentik.providers.oauth2", | ||||
|                         "authentik.providers.proxy", | ||||
|                         "authentik.providers.rac", | ||||
|                         "authentik.providers.radius", | ||||
|                         "authentik.providers.saml", | ||||
|                         "authentik.providers.scim", | ||||
| @ -4703,7 +4704,6 @@ | ||||
|                         "authentik.enterprise.audit", | ||||
|                         "authentik.enterprise.providers.google_workspace", | ||||
|                         "authentik.enterprise.providers.microsoft_entra", | ||||
|                         "authentik.enterprise.providers.rac", | ||||
|                         "authentik.enterprise.providers.ssf", | ||||
|                         "authentik.enterprise.stages.authenticator_endpoint_gdtc", | ||||
|                         "authentik.enterprise.stages.source", | ||||
| @ -4738,6 +4738,9 @@ | ||||
|                         "authentik_providers_oauth2.scopemapping", | ||||
|                         "authentik_providers_oauth2.oauth2provider", | ||||
|                         "authentik_providers_proxy.proxyprovider", | ||||
|                         "authentik_providers_rac.racprovider", | ||||
|                         "authentik_providers_rac.endpoint", | ||||
|                         "authentik_providers_rac.racpropertymapping", | ||||
|                         "authentik_providers_radius.radiusprovider", | ||||
|                         "authentik_providers_radius.radiusproviderpropertymapping", | ||||
|                         "authentik_providers_saml.samlprovider", | ||||
| @ -4807,9 +4810,6 @@ | ||||
|                         "authentik_providers_google_workspace.googleworkspaceprovidermapping", | ||||
|                         "authentik_providers_microsoft_entra.microsoftentraprovider", | ||||
|                         "authentik_providers_microsoft_entra.microsoftentraprovidermapping", | ||||
|                         "authentik_providers_rac.racprovider", | ||||
|                         "authentik_providers_rac.endpoint", | ||||
|                         "authentik_providers_rac.racpropertymapping", | ||||
|                         "authentik_providers_ssf.ssfprovider", | ||||
|                         "authentik_stages_authenticator_endpoint_gdtc.authenticatorendpointgdtcstage", | ||||
|                         "authentik_stages_source.sourcestage", | ||||
| @ -6046,6 +6046,216 @@ | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racprovider": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "name": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Name" | ||||
|                 }, | ||||
|                 "authentication_flow": { | ||||
|                     "type": "string", | ||||
|                     "format": "uuid", | ||||
|                     "title": "Authentication flow", | ||||
|                     "description": "Flow used for authentication when the associated application is accessed by an un-authenticated user." | ||||
|                 }, | ||||
|                 "authorization_flow": { | ||||
|                     "type": "string", | ||||
|                     "format": "uuid", | ||||
|                     "title": "Authorization flow", | ||||
|                     "description": "Flow used when authorizing this provider." | ||||
|                 }, | ||||
|                 "property_mappings": { | ||||
|                     "type": "array", | ||||
|                     "items": { | ||||
|                         "type": "string", | ||||
|                         "format": "uuid" | ||||
|                     }, | ||||
|                     "title": "Property mappings" | ||||
|                 }, | ||||
|                 "settings": { | ||||
|                     "type": "object", | ||||
|                     "additionalProperties": true, | ||||
|                     "title": "Settings" | ||||
|                 }, | ||||
|                 "connection_expiry": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Connection expiry", | ||||
|                     "description": "Determines how long a session lasts. Default of 0 means that the sessions lasts until the browser is closed. (Format: hours=-1;minutes=-2;seconds=-3)" | ||||
|                 }, | ||||
|                 "delete_token_on_disconnect": { | ||||
|                     "type": "boolean", | ||||
|                     "title": "Delete token on disconnect", | ||||
|                     "description": "When set to true, connection tokens will be deleted upon disconnect." | ||||
|                 } | ||||
|             }, | ||||
|             "required": [] | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racprovider_permissions": { | ||||
|             "type": "array", | ||||
|             "items": { | ||||
|                 "type": "object", | ||||
|                 "required": [ | ||||
|                     "permission" | ||||
|                 ], | ||||
|                 "properties": { | ||||
|                     "permission": { | ||||
|                         "type": "string", | ||||
|                         "enum": [ | ||||
|                             "add_racprovider", | ||||
|                             "change_racprovider", | ||||
|                             "delete_racprovider", | ||||
|                             "view_racprovider" | ||||
|                         ] | ||||
|                     }, | ||||
|                     "user": { | ||||
|                         "type": "integer" | ||||
|                     }, | ||||
|                     "role": { | ||||
|                         "type": "string" | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_rac.endpoint": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "name": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Name" | ||||
|                 }, | ||||
|                 "provider": { | ||||
|                     "type": "integer", | ||||
|                     "title": "Provider" | ||||
|                 }, | ||||
|                 "protocol": { | ||||
|                     "type": "string", | ||||
|                     "enum": [ | ||||
|                         "rdp", | ||||
|                         "vnc", | ||||
|                         "ssh" | ||||
|                     ], | ||||
|                     "title": "Protocol" | ||||
|                 }, | ||||
|                 "host": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Host" | ||||
|                 }, | ||||
|                 "settings": { | ||||
|                     "type": "object", | ||||
|                     "additionalProperties": true, | ||||
|                     "title": "Settings" | ||||
|                 }, | ||||
|                 "property_mappings": { | ||||
|                     "type": "array", | ||||
|                     "items": { | ||||
|                         "type": "string", | ||||
|                         "format": "uuid" | ||||
|                     }, | ||||
|                     "title": "Property mappings" | ||||
|                 }, | ||||
|                 "auth_mode": { | ||||
|                     "type": "string", | ||||
|                     "enum": [ | ||||
|                         "static", | ||||
|                         "prompt" | ||||
|                     ], | ||||
|                     "title": "Auth mode" | ||||
|                 }, | ||||
|                 "maximum_connections": { | ||||
|                     "type": "integer", | ||||
|                     "minimum": -2147483648, | ||||
|                     "maximum": 2147483647, | ||||
|                     "title": "Maximum connections" | ||||
|                 } | ||||
|             }, | ||||
|             "required": [] | ||||
|         }, | ||||
|         "model_authentik_providers_rac.endpoint_permissions": { | ||||
|             "type": "array", | ||||
|             "items": { | ||||
|                 "type": "object", | ||||
|                 "required": [ | ||||
|                     "permission" | ||||
|                 ], | ||||
|                 "properties": { | ||||
|                     "permission": { | ||||
|                         "type": "string", | ||||
|                         "enum": [ | ||||
|                             "add_endpoint", | ||||
|                             "change_endpoint", | ||||
|                             "delete_endpoint", | ||||
|                             "view_endpoint" | ||||
|                         ] | ||||
|                     }, | ||||
|                     "user": { | ||||
|                         "type": "integer" | ||||
|                     }, | ||||
|                     "role": { | ||||
|                         "type": "string" | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racpropertymapping": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "managed": { | ||||
|                     "type": [ | ||||
|                         "string", | ||||
|                         "null" | ||||
|                     ], | ||||
|                     "minLength": 1, | ||||
|                     "title": "Managed by authentik", | ||||
|                     "description": "Objects that are managed by authentik. These objects are created and updated automatically. This flag only indicates that an object can be overwritten by migrations. You can still modify the objects via the API, but expect changes to be overwritten in a later update." | ||||
|                 }, | ||||
|                 "name": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Name" | ||||
|                 }, | ||||
|                 "expression": { | ||||
|                     "type": "string", | ||||
|                     "title": "Expression" | ||||
|                 }, | ||||
|                 "static_settings": { | ||||
|                     "type": "object", | ||||
|                     "additionalProperties": true, | ||||
|                     "title": "Static settings" | ||||
|                 } | ||||
|             }, | ||||
|             "required": [] | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racpropertymapping_permissions": { | ||||
|             "type": "array", | ||||
|             "items": { | ||||
|                 "type": "object", | ||||
|                 "required": [ | ||||
|                     "permission" | ||||
|                 ], | ||||
|                 "properties": { | ||||
|                     "permission": { | ||||
|                         "type": "string", | ||||
|                         "enum": [ | ||||
|                             "add_racpropertymapping", | ||||
|                             "change_racpropertymapping", | ||||
|                             "delete_racpropertymapping", | ||||
|                             "view_racpropertymapping" | ||||
|                         ] | ||||
|                     }, | ||||
|                     "user": { | ||||
|                         "type": "integer" | ||||
|                     }, | ||||
|                     "role": { | ||||
|                         "type": "string" | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_radius.radiusprovider": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
| @ -14215,216 +14425,6 @@ | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racprovider": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "name": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Name" | ||||
|                 }, | ||||
|                 "authentication_flow": { | ||||
|                     "type": "string", | ||||
|                     "format": "uuid", | ||||
|                     "title": "Authentication flow", | ||||
|                     "description": "Flow used for authentication when the associated application is accessed by an un-authenticated user." | ||||
|                 }, | ||||
|                 "authorization_flow": { | ||||
|                     "type": "string", | ||||
|                     "format": "uuid", | ||||
|                     "title": "Authorization flow", | ||||
|                     "description": "Flow used when authorizing this provider." | ||||
|                 }, | ||||
|                 "property_mappings": { | ||||
|                     "type": "array", | ||||
|                     "items": { | ||||
|                         "type": "string", | ||||
|                         "format": "uuid" | ||||
|                     }, | ||||
|                     "title": "Property mappings" | ||||
|                 }, | ||||
|                 "settings": { | ||||
|                     "type": "object", | ||||
|                     "additionalProperties": true, | ||||
|                     "title": "Settings" | ||||
|                 }, | ||||
|                 "connection_expiry": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Connection expiry", | ||||
|                     "description": "Determines how long a session lasts. Default of 0 means that the sessions lasts until the browser is closed. (Format: hours=-1;minutes=-2;seconds=-3)" | ||||
|                 }, | ||||
|                 "delete_token_on_disconnect": { | ||||
|                     "type": "boolean", | ||||
|                     "title": "Delete token on disconnect", | ||||
|                     "description": "When set to true, connection tokens will be deleted upon disconnect." | ||||
|                 } | ||||
|             }, | ||||
|             "required": [] | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racprovider_permissions": { | ||||
|             "type": "array", | ||||
|             "items": { | ||||
|                 "type": "object", | ||||
|                 "required": [ | ||||
|                     "permission" | ||||
|                 ], | ||||
|                 "properties": { | ||||
|                     "permission": { | ||||
|                         "type": "string", | ||||
|                         "enum": [ | ||||
|                             "add_racprovider", | ||||
|                             "change_racprovider", | ||||
|                             "delete_racprovider", | ||||
|                             "view_racprovider" | ||||
|                         ] | ||||
|                     }, | ||||
|                     "user": { | ||||
|                         "type": "integer" | ||||
|                     }, | ||||
|                     "role": { | ||||
|                         "type": "string" | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_rac.endpoint": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "name": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Name" | ||||
|                 }, | ||||
|                 "provider": { | ||||
|                     "type": "integer", | ||||
|                     "title": "Provider" | ||||
|                 }, | ||||
|                 "protocol": { | ||||
|                     "type": "string", | ||||
|                     "enum": [ | ||||
|                         "rdp", | ||||
|                         "vnc", | ||||
|                         "ssh" | ||||
|                     ], | ||||
|                     "title": "Protocol" | ||||
|                 }, | ||||
|                 "host": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Host" | ||||
|                 }, | ||||
|                 "settings": { | ||||
|                     "type": "object", | ||||
|                     "additionalProperties": true, | ||||
|                     "title": "Settings" | ||||
|                 }, | ||||
|                 "property_mappings": { | ||||
|                     "type": "array", | ||||
|                     "items": { | ||||
|                         "type": "string", | ||||
|                         "format": "uuid" | ||||
|                     }, | ||||
|                     "title": "Property mappings" | ||||
|                 }, | ||||
|                 "auth_mode": { | ||||
|                     "type": "string", | ||||
|                     "enum": [ | ||||
|                         "static", | ||||
|                         "prompt" | ||||
|                     ], | ||||
|                     "title": "Auth mode" | ||||
|                 }, | ||||
|                 "maximum_connections": { | ||||
|                     "type": "integer", | ||||
|                     "minimum": -2147483648, | ||||
|                     "maximum": 2147483647, | ||||
|                     "title": "Maximum connections" | ||||
|                 } | ||||
|             }, | ||||
|             "required": [] | ||||
|         }, | ||||
|         "model_authentik_providers_rac.endpoint_permissions": { | ||||
|             "type": "array", | ||||
|             "items": { | ||||
|                 "type": "object", | ||||
|                 "required": [ | ||||
|                     "permission" | ||||
|                 ], | ||||
|                 "properties": { | ||||
|                     "permission": { | ||||
|                         "type": "string", | ||||
|                         "enum": [ | ||||
|                             "add_endpoint", | ||||
|                             "change_endpoint", | ||||
|                             "delete_endpoint", | ||||
|                             "view_endpoint" | ||||
|                         ] | ||||
|                     }, | ||||
|                     "user": { | ||||
|                         "type": "integer" | ||||
|                     }, | ||||
|                     "role": { | ||||
|                         "type": "string" | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racpropertymapping": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|                 "managed": { | ||||
|                     "type": [ | ||||
|                         "string", | ||||
|                         "null" | ||||
|                     ], | ||||
|                     "minLength": 1, | ||||
|                     "title": "Managed by authentik", | ||||
|                     "description": "Objects that are managed by authentik. These objects are created and updated automatically. This flag only indicates that an object can be overwritten by migrations. You can still modify the objects via the API, but expect changes to be overwritten in a later update." | ||||
|                 }, | ||||
|                 "name": { | ||||
|                     "type": "string", | ||||
|                     "minLength": 1, | ||||
|                     "title": "Name" | ||||
|                 }, | ||||
|                 "expression": { | ||||
|                     "type": "string", | ||||
|                     "title": "Expression" | ||||
|                 }, | ||||
|                 "static_settings": { | ||||
|                     "type": "object", | ||||
|                     "additionalProperties": true, | ||||
|                     "title": "Static settings" | ||||
|                 } | ||||
|             }, | ||||
|             "required": [] | ||||
|         }, | ||||
|         "model_authentik_providers_rac.racpropertymapping_permissions": { | ||||
|             "type": "array", | ||||
|             "items": { | ||||
|                 "type": "object", | ||||
|                 "required": [ | ||||
|                     "permission" | ||||
|                 ], | ||||
|                 "properties": { | ||||
|                     "permission": { | ||||
|                         "type": "string", | ||||
|                         "enum": [ | ||||
|                             "add_racpropertymapping", | ||||
|                             "change_racpropertymapping", | ||||
|                             "delete_racpropertymapping", | ||||
|                             "view_racpropertymapping" | ||||
|                         ] | ||||
|                     }, | ||||
|                     "user": { | ||||
|                         "type": "integer" | ||||
|                     }, | ||||
|                     "role": { | ||||
|                         "type": "string" | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         }, | ||||
|         "model_authentik_providers_ssf.ssfprovider": { | ||||
|             "type": "object", | ||||
|             "properties": { | ||||
|  | ||||
| @ -10,6 +10,7 @@ import ( | ||||
|  | ||||
| 	"goauthentik.io/internal/common" | ||||
| 	"goauthentik.io/internal/config" | ||||
| 	"goauthentik.io/internal/constants" | ||||
| 	"goauthentik.io/internal/debug" | ||||
| 	"goauthentik.io/internal/outpost/ak" | ||||
| 	"goauthentik.io/internal/outpost/ak/healthcheck" | ||||
| @ -24,7 +25,8 @@ Required environment variables: | ||||
| - AUTHENTIK_INSECURE: Skip SSL Certificate verification` | ||||
|  | ||||
| var rootCmd = &cobra.Command{ | ||||
| 	Long: helpMessage, | ||||
| 	Long:    helpMessage, | ||||
| 	Version: constants.FullVersion(), | ||||
| 	PersistentPreRun: func(cmd *cobra.Command, args []string) { | ||||
| 		log.SetLevel(log.DebugLevel) | ||||
| 		log.SetFormatter(&log.JSONFormatter{ | ||||
|  | ||||
| @ -10,6 +10,7 @@ import ( | ||||
|  | ||||
| 	"goauthentik.io/internal/common" | ||||
| 	"goauthentik.io/internal/config" | ||||
| 	"goauthentik.io/internal/constants" | ||||
| 	"goauthentik.io/internal/debug" | ||||
| 	"goauthentik.io/internal/outpost/ak" | ||||
| 	"goauthentik.io/internal/outpost/ak/healthcheck" | ||||
| @ -27,7 +28,8 @@ Optionally, you can set these: | ||||
| - AUTHENTIK_HOST_BROWSER: URL to use in the browser, when it differs from AUTHENTIK_HOST` | ||||
|  | ||||
| var rootCmd = &cobra.Command{ | ||||
| 	Long: helpMessage, | ||||
| 	Long:    helpMessage, | ||||
| 	Version: constants.FullVersion(), | ||||
| 	PersistentPreRun: func(cmd *cobra.Command, args []string) { | ||||
| 		log.SetLevel(log.DebugLevel) | ||||
| 		log.SetFormatter(&log.JSONFormatter{ | ||||
|  | ||||
| @ -9,6 +9,7 @@ import ( | ||||
| 	"github.com/spf13/cobra" | ||||
|  | ||||
| 	"goauthentik.io/internal/common" | ||||
| 	"goauthentik.io/internal/constants" | ||||
| 	"goauthentik.io/internal/debug" | ||||
| 	"goauthentik.io/internal/outpost/ak" | ||||
| 	"goauthentik.io/internal/outpost/ak/healthcheck" | ||||
| @ -23,7 +24,8 @@ Required environment variables: | ||||
| - AUTHENTIK_INSECURE: Skip SSL Certificate verification` | ||||
|  | ||||
| var rootCmd = &cobra.Command{ | ||||
| 	Long: helpMessage, | ||||
| 	Long:    helpMessage, | ||||
| 	Version: constants.FullVersion(), | ||||
| 	PersistentPreRun: func(cmd *cobra.Command, args []string) { | ||||
| 		log.SetLevel(log.DebugLevel) | ||||
| 		log.SetFormatter(&log.JSONFormatter{ | ||||
|  | ||||
| @ -9,6 +9,7 @@ import ( | ||||
| 	"github.com/spf13/cobra" | ||||
|  | ||||
| 	"goauthentik.io/internal/common" | ||||
| 	"goauthentik.io/internal/constants" | ||||
| 	"goauthentik.io/internal/debug" | ||||
| 	"goauthentik.io/internal/outpost/ak" | ||||
| 	"goauthentik.io/internal/outpost/ak/healthcheck" | ||||
| @ -23,7 +24,8 @@ Required environment variables: | ||||
| - AUTHENTIK_INSECURE: Skip SSL Certificate verification` | ||||
|  | ||||
| var rootCmd = &cobra.Command{ | ||||
| 	Long: helpMessage, | ||||
| 	Long:    helpMessage, | ||||
| 	Version: constants.FullVersion(), | ||||
| 	PersistentPreRun: func(cmd *cobra.Command, args []string) { | ||||
| 		log.SetLevel(log.DebugLevel) | ||||
| 		log.SetFormatter(&log.JSONFormatter{ | ||||
|  | ||||
| @ -31,7 +31,7 @@ services: | ||||
|     volumes: | ||||
|       - redis:/data | ||||
|   server: | ||||
|     image: ${AUTHENTIK_IMAGE:-ghcr.io/goauthentik/server}:${AUTHENTIK_TAG:-2024.12.3} | ||||
|     image: ${AUTHENTIK_IMAGE:-ghcr.io/goauthentik/server}:${AUTHENTIK_TAG:-2025.2.1} | ||||
|     restart: unless-stopped | ||||
|     command: server | ||||
|     environment: | ||||
| @ -54,7 +54,7 @@ services: | ||||
|       redis: | ||||
|         condition: service_healthy | ||||
|   worker: | ||||
|     image: ${AUTHENTIK_IMAGE:-ghcr.io/goauthentik/server}:${AUTHENTIK_TAG:-2024.12.3} | ||||
|     image: ${AUTHENTIK_IMAGE:-ghcr.io/goauthentik/server}:${AUTHENTIK_TAG:-2025.2.1} | ||||
|     restart: unless-stopped | ||||
|     command: worker | ||||
|     environment: | ||||
|  | ||||
| @ -29,4 +29,4 @@ func UserAgent() string { | ||||
| 	return fmt.Sprintf("authentik@%s", FullVersion()) | ||||
| } | ||||
|  | ||||
| const VERSION = "2024.12.3" | ||||
| const VERSION = "2025.2.1" | ||||
|  | ||||
| @ -26,7 +26,7 @@ Parameters: | ||||
|     Description: authentik Docker image | ||||
|   AuthentikVersion: | ||||
|     Type: String | ||||
|     Default: 2024.12.3 | ||||
|     Default: 2025.2.1 | ||||
|     Description: authentik Docker image tag | ||||
|   AuthentikServerCPU: | ||||
|     Type: Number | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| { | ||||
|     "name": "@goauthentik/authentik", | ||||
|     "version": "2024.12.3", | ||||
|     "version": "2025.2.1", | ||||
|     "private": true | ||||
| } | ||||
|  | ||||
| @ -1,6 +1,6 @@ | ||||
| [tool.poetry] | ||||
| name = "authentik" | ||||
| version = "2024.12.3" | ||||
| version = "2025.2.1" | ||||
| description = "" | ||||
| authors = ["authentik Team <hello@goauthentik.io>"] | ||||
|  | ||||
|  | ||||
							
								
								
									
										10
									
								
								schema.yml
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								schema.yml
									
									
									
									
									
								
							| @ -1,7 +1,7 @@ | ||||
| openapi: 3.0.3 | ||||
| info: | ||||
|   title: authentik | ||||
|   version: 2024.12.3 | ||||
|   version: 2025.2.1 | ||||
|   description: Making authentication simple. | ||||
|   contact: | ||||
|     email: hello@goauthentik.io | ||||
| @ -39482,6 +39482,7 @@ components: | ||||
|       - authentik.providers.ldap | ||||
|       - authentik.providers.oauth2 | ||||
|       - authentik.providers.proxy | ||||
|       - authentik.providers.rac | ||||
|       - authentik.providers.radius | ||||
|       - authentik.providers.saml | ||||
|       - authentik.providers.scim | ||||
| @ -39522,7 +39523,6 @@ components: | ||||
|       - authentik.enterprise.audit | ||||
|       - authentik.enterprise.providers.google_workspace | ||||
|       - authentik.enterprise.providers.microsoft_entra | ||||
|       - authentik.enterprise.providers.rac | ||||
|       - authentik.enterprise.providers.ssf | ||||
|       - authentik.enterprise.stages.authenticator_endpoint_gdtc | ||||
|       - authentik.enterprise.stages.source | ||||
| @ -46625,6 +46625,9 @@ components: | ||||
|       - authentik_providers_oauth2.scopemapping | ||||
|       - authentik_providers_oauth2.oauth2provider | ||||
|       - authentik_providers_proxy.proxyprovider | ||||
|       - authentik_providers_rac.racprovider | ||||
|       - authentik_providers_rac.endpoint | ||||
|       - authentik_providers_rac.racpropertymapping | ||||
|       - authentik_providers_radius.radiusprovider | ||||
|       - authentik_providers_radius.radiusproviderpropertymapping | ||||
|       - authentik_providers_saml.samlprovider | ||||
| @ -46694,9 +46697,6 @@ components: | ||||
|       - authentik_providers_google_workspace.googleworkspaceprovidermapping | ||||
|       - authentik_providers_microsoft_entra.microsoftentraprovider | ||||
|       - authentik_providers_microsoft_entra.microsoftentraprovidermapping | ||||
|       - authentik_providers_rac.racprovider | ||||
|       - authentik_providers_rac.endpoint | ||||
|       - authentik_providers_rac.racpropertymapping | ||||
|       - authentik_providers_ssf.ssfprovider | ||||
|       - authentik_stages_authenticator_endpoint_gdtc.authenticatorendpointgdtcstage | ||||
|       - authentik_stages_source.sourcestage | ||||
|  | ||||
| @ -74,7 +74,7 @@ const interfaces = [ | ||||
|     ["user/UserInterface.ts", "user"], | ||||
|     ["flow/FlowInterface.ts", "flow"], | ||||
|     ["standalone/api-browser/index.ts", "standalone/api-browser"], | ||||
|     ["enterprise/rac/index.ts", "enterprise/rac"], | ||||
|     ["rac/index.ts", "rac"], | ||||
|     ["standalone/loading/index.ts", "standalone/loading"], | ||||
|     ["polyfill/poly.ts", "."], | ||||
| ]; | ||||
|  | ||||
							
								
								
									
										8
									
								
								web/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										8
									
								
								web/package-lock.json
									
									
									
										generated
									
									
									
								
							| @ -23,7 +23,7 @@ | ||||
|                 "@floating-ui/dom": "^1.6.11", | ||||
|                 "@formatjs/intl-listformat": "^7.5.7", | ||||
|                 "@fortawesome/fontawesome-free": "^6.6.0", | ||||
|                 "@goauthentik/api": "^2024.12.3-1739814462", | ||||
|                 "@goauthentik/api": "^2024.12.3-1739965710", | ||||
|                 "@lit-labs/ssr": "^3.2.2", | ||||
|                 "@lit/context": "^1.1.2", | ||||
|                 "@lit/localize": "^0.12.2", | ||||
| @ -1814,9 +1814,9 @@ | ||||
|             } | ||||
|         }, | ||||
|         "node_modules/@goauthentik/api": { | ||||
|             "version": "2024.12.3-1739814462", | ||||
|             "resolved": "https://registry.npmjs.org/@goauthentik/api/-/api-2024.12.3-1739814462.tgz", | ||||
|             "integrity": "sha512-qWGsq7zP0rG1PfjZA+iimaX4cVkd1n2JA/WceTOKgBmqnomQSI7SJNkdSpD+Qdy76PI0UuQWN73PInq/3rmm5Q==" | ||||
|             "version": "2024.12.3-1739965710", | ||||
|             "resolved": "https://registry.npmjs.org/@goauthentik/api/-/api-2024.12.3-1739965710.tgz", | ||||
|             "integrity": "sha512-16zoQWeJhAFSwttvqLRoXoQA43tMW1ZXDEihW6r8rtWtlxqPh7n36RtcWYraYiLcjmJskI90zdgz6k1kmY5AXw==" | ||||
|         }, | ||||
|         "node_modules/@goauthentik/web": { | ||||
|             "resolved": "", | ||||
|  | ||||
| @ -11,7 +11,7 @@ | ||||
|         "@floating-ui/dom": "^1.6.11", | ||||
|         "@formatjs/intl-listformat": "^7.5.7", | ||||
|         "@fortawesome/fontawesome-free": "^6.6.0", | ||||
|         "@goauthentik/api": "^2024.12.3-1739814462", | ||||
|         "@goauthentik/api": "^2024.12.3-1739965710", | ||||
|         "@lit-labs/ssr": "^3.2.2", | ||||
|         "@lit/context": "^1.1.2", | ||||
|         "@lit/localize": "^0.12.2", | ||||
|  | ||||
| @ -6,7 +6,7 @@ const config: KnipConfig = { | ||||
|         "./src/user/UserInterface.ts", | ||||
|         "./src/flow/FlowInterface.ts", | ||||
|         "./src/standalone/api-browser/index.ts", | ||||
|         "./src/enterprise/rac/index.ts", | ||||
|         "./src/rac/index.ts", | ||||
|         "./src/standalone/loading/index.ts", | ||||
|         "./src/polyfill/poly.ts", | ||||
|     ], | ||||
|  | ||||
| @ -7,6 +7,7 @@ import "@goauthentik/components/ak-radio-input"; | ||||
| import "@goauthentik/components/ak-switch-input"; | ||||
| import "@goauthentik/components/ak-text-input"; | ||||
| import "@goauthentik/components/ak-textarea-input"; | ||||
| import "@goauthentik/elements/Alert.js"; | ||||
| import { | ||||
|     CapabilitiesEnum, | ||||
|     WithCapabilitiesConfig, | ||||
| @ -21,7 +22,7 @@ import "@goauthentik/elements/forms/SearchSelect"; | ||||
| import "@patternfly/elements/pf-tooltip/pf-tooltip.js"; | ||||
|  | ||||
| import { msg } from "@lit/localize"; | ||||
| import { TemplateResult, html } from "lit"; | ||||
| import { TemplateResult, html, nothing } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators.js"; | ||||
| import { ifDefined } from "lit/directives/if-defined.js"; | ||||
|  | ||||
| @ -120,7 +121,12 @@ export class ApplicationForm extends WithCapabilitiesConfig(ModelForm<Applicatio | ||||
|     } | ||||
|  | ||||
|     renderForm(): TemplateResult { | ||||
|         const alertMsg = msg( | ||||
|             "Using this form will only create an Application. In order to authenticate with the application, you will have to manually pair it with a Provider.", | ||||
|         ); | ||||
|  | ||||
|         return html`<form class="pf-c-form pf-m-horizontal"> | ||||
|             ${this.instance ? nothing : html`<ak-alert level="pf-m-info">${alertMsg}</ak-alert>`} | ||||
|             <ak-text-input | ||||
|                 name="name" | ||||
|                 value=${ifDefined(this.instance?.name)} | ||||
|  | ||||
| @ -50,7 +50,7 @@ export class ApplicationListPage extends WithBrandConfig(TablePage<Application>) | ||||
|     } | ||||
|     pageDescription(): string { | ||||
|         return msg( | ||||
|             str`External applications that use ${this.brand.brandingTitle || "authentik"} as an identity provider via protocols like OAuth2 and SAML. All applications are shown here, even ones you cannot access.`, | ||||
|             str`External applications that use ${this.brand?.brandingTitle ?? "authentik"} as an identity provider via protocols like OAuth2 and SAML. All applications are shown here, even ones you cannot access.`, | ||||
|         ); | ||||
|     } | ||||
|     pageIcon(): string { | ||||
| @ -85,10 +85,6 @@ export class ApplicationListPage extends WithBrandConfig(TablePage<Application>) | ||||
|         ]; | ||||
|     } | ||||
|  | ||||
|     renderSectionBefore(): TemplateResult { | ||||
|         return html`<ak-application-wizard-hint></ak-application-wizard-hint>`; | ||||
|     } | ||||
|  | ||||
|     renderSidebarAfter(): TemplateResult { | ||||
|         return html`<div class="pf-c-sidebar__panel pf-m-width-25"> | ||||
|             <div class="pf-c-card"> | ||||
| @ -160,12 +156,21 @@ export class ApplicationListPage extends WithBrandConfig(TablePage<Application>) | ||||
|     } | ||||
|  | ||||
|     renderObjectCreate(): TemplateResult { | ||||
|         return html`<ak-forms-modal .open=${getURLParam("createForm", false)}> | ||||
|             <span slot="submit"> ${msg("Create")} </span> | ||||
|             <span slot="header"> ${msg("Create Application")} </span> | ||||
|             <ak-application-form slot="form"> </ak-application-form> | ||||
|             <button slot="trigger" class="pf-c-button pf-m-primary">${msg("Create")}</button> | ||||
|         </ak-forms-modal>`; | ||||
|         return html` <ak-application-wizard .open=${getURLParam("createWizard", false)}> | ||||
|                 <button | ||||
|                     slot="trigger" | ||||
|                     class="pf-c-button pf-m-primary" | ||||
|                     data-ouia-component-id="start-application-wizard" | ||||
|                 > | ||||
|                     ${msg("Create with Provider")} | ||||
|                 </button> | ||||
|             </ak-application-wizard> | ||||
|             <ak-forms-modal .open=${getURLParam("createForm", false)}> | ||||
|                 <span slot="submit"> ${msg("Create")} </span> | ||||
|                 <span slot="header"> ${msg("Create Application")} </span> | ||||
|                 <ak-application-form slot="form"> </ak-application-form> | ||||
|                 <button slot="trigger" class="pf-c-button pf-m-primary">${msg("Create")}</button> | ||||
|             </ak-forms-modal>`; | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| @ -30,7 +30,7 @@ export class ApplicationWizardStep extends WizardStep { | ||||
|     // As recommended in [WizardStep](../../../components/ak-wizard/WizardStep.ts), we override | ||||
|     // these fields and provide them to all the child classes. | ||||
|     wizardTitle = msg("New application"); | ||||
|     wizardDescription = msg("Create a new application"); | ||||
|     wizardDescription = msg("Create a new application and configure a provider for it."); | ||||
|     canCancel = true; | ||||
|  | ||||
|     // This should be overridden in the children for more precise targeting. | ||||
|  | ||||
| @ -31,9 +31,9 @@ export class BoundPoliciesList extends Table<PolicyBinding> { | ||||
|  | ||||
|     @property({ type: Array }) | ||||
|     allowedTypes: PolicyBindingCheckTarget[] = [ | ||||
|         PolicyBindingCheckTarget.policy, | ||||
|         PolicyBindingCheckTarget.group, | ||||
|         PolicyBindingCheckTarget.user, | ||||
|         PolicyBindingCheckTarget.policy, | ||||
|     ]; | ||||
|  | ||||
|     @property({ type: Array }) | ||||
|  | ||||
| @ -58,9 +58,9 @@ export class PolicyBindingForm extends ModelForm<PolicyBinding, string> { | ||||
|  | ||||
|     @property({ type: Array }) | ||||
|     allowedTypes: PolicyBindingCheckTarget[] = [ | ||||
|         PolicyBindingCheckTarget.policy, | ||||
|         PolicyBindingCheckTarget.group, | ||||
|         PolicyBindingCheckTarget.user, | ||||
|         PolicyBindingCheckTarget.policy, | ||||
|     ]; | ||||
|  | ||||
|     @property({ type: Array }) | ||||
|  | ||||
| @ -105,6 +105,22 @@ export class GeoIPPolicyForm extends BasePolicyForm<GeoIPPolicy> { | ||||
|                             )} | ||||
|                         </p> | ||||
|                     </ak-form-element-horizontal> | ||||
|                     <ak-form-element-horizontal | ||||
|                         label=${msg("Maximum distance")} | ||||
|                         name="historyMaxDistanceKm" | ||||
|                     > | ||||
|                         <input | ||||
|                             type="number" | ||||
|                             min="1" | ||||
|                             value="${first(this.instance?.historyMaxDistanceKm, 100)}" | ||||
|                             class="pf-c-form-control" | ||||
|                         /> | ||||
|                         <p class="pf-c-form__helper-text"> | ||||
|                             ${msg( | ||||
|                                 "Maximum distance a login attempt is allowed from in kilometers.", | ||||
|                             )} | ||||
|                         </p> | ||||
|                     </ak-form-element-horizontal> | ||||
|                     <ak-form-element-horizontal | ||||
|                         label=${msg("Distance tolerance")} | ||||
|                         name="distanceToleranceKm" | ||||
| @ -133,27 +149,6 @@ export class GeoIPPolicyForm extends BasePolicyForm<GeoIPPolicy> { | ||||
|                             ${msg("Amount of previous login events to check against.")} | ||||
|                         </p> | ||||
|                     </ak-form-element-horizontal> | ||||
|                     <ak-form-element-horizontal | ||||
|                         label=${msg("Maximum distance")} | ||||
|                         name="historyMaxDistanceKm" | ||||
|                     > | ||||
|                         <input | ||||
|                             type="number" | ||||
|                             min="1" | ||||
|                             value="${first(this.instance?.historyMaxDistanceKm, 100)}" | ||||
|                             class="pf-c-form-control" | ||||
|                         /> | ||||
|                         <p class="pf-c-form__helper-text"> | ||||
|                             ${msg( | ||||
|                                 "Maximum distance a login attempt is allowed from in kilometers.", | ||||
|                             )} | ||||
|                         </p> | ||||
|                     </ak-form-element-horizontal> | ||||
|                 </div> | ||||
|             </ak-form-group> | ||||
|             <ak-form-group> | ||||
|                 <span slot="header"> ${msg("Distance settings (Impossible travel)")} </span> | ||||
|                 <div slot="body" class="pf-c-form"> | ||||
|                     <ak-form-element-horizontal name="checkImpossibleTravel"> | ||||
|                         <label class="pf-c-switch"> | ||||
|                             <input | ||||
|  | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 772 KiB After Width: | Height: | Size: 628 KiB | 
| @ -3,7 +3,7 @@ export const SUCCESS_CLASS = "pf-m-success"; | ||||
| export const ERROR_CLASS = "pf-m-danger"; | ||||
| export const PROGRESS_CLASS = "pf-m-in-progress"; | ||||
| export const CURRENT_CLASS = "pf-m-current"; | ||||
| export const VERSION = "2024.12.3"; | ||||
| export const VERSION = "2025.2.1"; | ||||
| export const TITLE_DEFAULT = "authentik"; | ||||
| export const ROUTE_SEPARATOR = ";"; | ||||
|  | ||||
|  | ||||
| @ -13,6 +13,7 @@ export interface GlobalAuthentik { | ||||
|     build: string; | ||||
|     api: { | ||||
|         base: string; | ||||
|         relBase: string; | ||||
|     }; | ||||
| } | ||||
|  | ||||
| @ -27,6 +28,7 @@ export function globalAK(): GlobalAuthentik { | ||||
|         ak.brand = CurrentBrandFromJSON(ak.brand); | ||||
|         ak.config = ConfigFromJSON(ak.config); | ||||
|     } | ||||
|     const apiBase = new URL(process.env.AK_API_BASE_PATH || window.location.origin); | ||||
|     if (!ak) { | ||||
|         return { | ||||
|             config: ConfigFromJSON({ | ||||
| @ -39,7 +41,8 @@ export function globalAK(): GlobalAuthentik { | ||||
|             versionSubdomain: "", | ||||
|             build: "", | ||||
|             api: { | ||||
|                 base: process.env.AK_API_BASE_PATH || window.location.origin, | ||||
|                 base: apiBase.toString(), | ||||
|                 relBase: apiBase.pathname, | ||||
|             }, | ||||
|         }; | ||||
|     } | ||||
|  | ||||
| @ -45,6 +45,8 @@ html > form > input { | ||||
|     left: -2000px; | ||||
| } | ||||
|  | ||||
| /*#region Icons*/ | ||||
|  | ||||
| .pf-icon { | ||||
|     display: inline-block; | ||||
|     font-style: normal; | ||||
| @ -54,6 +56,18 @@ html > form > input { | ||||
|     vertical-align: middle; | ||||
| } | ||||
|  | ||||
| .pf-c-form-control { | ||||
|     --pf-c-form-control--m-caps-lock--BackgroundUrl: url("data:image/svg+xml;charset=utf8,%3Csvg fill='%23aaabac' viewBox='0 0 56 56' xmlns='http://www.w3.org/2000/svg'%3E%3Cpath d='M 20.7812 37.6211 L 35.2421 37.6211 C 38.5233 37.6211 40.2577 35.6992 40.2577 32.6055 L 40.2577 28.4570 L 49.1404 28.4570 C 51.0859 28.4570 52.6329 27.3086 52.6329 25.5039 C 52.6329 24.4024 52.0703 23.5351 51.0158 22.6211 L 30.9062 4.8789 C 29.9452 4.0351 29.0546 3.4727 27.9999 3.4727 C 26.9687 3.4727 26.0780 4.0351 25.1171 4.8789 L 4.9843 22.6445 C 3.8828 23.6055 3.3671 24.4024 3.3671 25.5039 C 3.3671 27.3086 4.9140 28.4570 6.8828 28.4570 L 15.7421 28.4570 L 15.7421 32.6055 C 15.7421 35.6992 17.4999 37.6211 20.7812 37.6211 Z M 21.1562 34.0820 C 20.2655 34.0820 19.6562 33.4961 19.6562 32.6055 L 19.6562 25.7149 C 19.6562 25.1524 19.4452 24.9180 18.8828 24.9180 L 8.6640 24.9180 C 8.4999 24.9180 8.4296 24.8476 8.4296 24.7305 C 8.4296 24.6367 8.4530 24.5430 8.5702 24.4492 L 27.5077 7.9961 C 27.7187 7.8086 27.8359 7.7383 27.9999 7.7383 C 28.1640 7.7383 28.3046 7.8086 28.4921 7.9961 L 47.4532 24.4492 C 47.5703 24.5430 47.5939 24.6367 47.5939 24.7305 C 47.5939 24.8476 47.4998 24.9180 47.3356 24.9180 L 37.1406 24.9180 C 36.5780 24.9180 36.3671 25.1524 36.3671 25.7149 L 36.3671 32.6055 C 36.3671 33.4727 35.7109 34.0820 34.8671 34.0820 Z M 19.7733 52.5273 L 36.0624 52.5273 C 38.7577 52.5273 40.3046 51.0273 40.3046 48.3086 L 40.3046 44.9336 C 40.3046 42.2148 38.7577 40.6680 36.0624 40.6680 L 19.7733 40.6680 C 17.0546 40.6680 15.5077 42.2383 15.5077 44.9336 L 15.5077 48.3086 C 15.5077 51.0039 17.0546 52.5273 19.7733 52.5273 Z M 20.3124 49.2227 C 19.4921 49.2227 19.0468 48.8008 19.0468 47.9805 L 19.0468 45.2617 C 19.0468 44.4414 19.4921 43.9727 20.3124 43.9727 L 35.5233 43.9727 C 36.3202 43.9727 36.7655 44.4414 36.7655 45.2617 L 36.7655 47.9805 C 36.7655 48.8008 36.3202 49.2227 35.5233 49.2227 Z'/%3E%3C/svg%3E"); | ||||
| } | ||||
|  | ||||
| .pf-c-form-control.pf-m-icon.pf-m-caps-lock { | ||||
|     --pf-c-form-control--m-icon--BackgroundUrl: var( | ||||
|         --pf-c-form-control--m-caps-lock--BackgroundUrl | ||||
|     ); | ||||
| } | ||||
|  | ||||
| /*#endregion*/ | ||||
|  | ||||
| .pf-c-page__header { | ||||
|     z-index: 0; | ||||
|     background-color: var(--ak-dark-background-light); | ||||
|  | ||||
| @ -3,6 +3,7 @@ import type { AbstractConstructor } from "@goauthentik/elements/types.js"; | ||||
|  | ||||
| import { consume } from "@lit/context"; | ||||
| import type { LitElement } from "lit"; | ||||
| import { state } from "lit/decorators.js"; | ||||
|  | ||||
| import type { CurrentBrand } from "@goauthentik/api"; | ||||
|  | ||||
| @ -12,6 +13,7 @@ export function WithBrandConfig<T extends AbstractConstructor<LitElement>>( | ||||
| ) { | ||||
|     abstract class WithBrandProvider extends superclass { | ||||
|         @consume({ context: authentikBrandContext, subscribe }) | ||||
|         @state() | ||||
|         public brand!: CurrentBrand; | ||||
|     } | ||||
|     return WithBrandProvider; | ||||
|  | ||||
							
								
								
									
										27
									
								
								web/src/elements/utils/focus.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								web/src/elements/utils/focus.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | ||||
| /** | ||||
|  * @fileoverview Utilities for DOM element interaction, focus management, and event handling. | ||||
|  */ | ||||
|  | ||||
| /** | ||||
|  * Recursively check if the target element or any of its children are active (i.e. "focused"). | ||||
|  * | ||||
|  * @param targetElement The element to check if it is active. | ||||
|  * @param containerElement The container element to check if the target element is active within. | ||||
|  */ | ||||
| export function isActiveElement( | ||||
|     targetElement: Element | null, | ||||
|     containerElement: Element | null, | ||||
| ): boolean { | ||||
|     // Does the container element even exist? | ||||
|     if (!containerElement) return false; | ||||
|  | ||||
|     // Does the container element have a shadow root? | ||||
|     if (!("shadowRoot" in containerElement)) return false; | ||||
|     if (containerElement.shadowRoot === null) return false; | ||||
|  | ||||
|     // Is the target element the active element? | ||||
|     if (containerElement.shadowRoot.activeElement === targetElement) return true; | ||||
|  | ||||
|     // Let's check the children of the container element... | ||||
|     return isActiveElement(containerElement.shadowRoot.activeElement, containerElement); | ||||
| } | ||||
| @ -1,36 +1,93 @@ | ||||
| import { AKElement } from "@goauthentik/elements/Base.js"; | ||||
| import { bound } from "@goauthentik/elements/decorators/bound"; | ||||
| import "@goauthentik/elements/forms/FormElement"; | ||||
| import { isActiveElement } from "@goauthentik/elements/utils/focus"; | ||||
|  | ||||
| import { msg } from "@lit/localize"; | ||||
| import { html, nothing, render } from "lit"; | ||||
| import { customElement, property } from "lit/decorators.js"; | ||||
| import { html, nothing } from "lit"; | ||||
| import { customElement, property, state } from "lit/decorators.js"; | ||||
| import { classMap } from "lit/directives/class-map.js"; | ||||
| import { ifDefined } from "lit/directives/if-defined.js"; | ||||
| import { Ref, createRef, ref } from "lit/directives/ref.js"; | ||||
|  | ||||
| import PFButton from "@patternfly/patternfly/components/Button/button.css"; | ||||
| import PFFormControl from "@patternfly/patternfly/components/FormControl/form-control.css"; | ||||
| import PFInputGroup from "@patternfly/patternfly/components/InputGroup/input-group.css"; | ||||
| import PFBase from "@patternfly/patternfly/patternfly-base.css"; | ||||
|  | ||||
| /** | ||||
|  * A configuration object for the visibility states of the password input. | ||||
|  */ | ||||
| interface VisibilityProps { | ||||
|     icon: string; | ||||
|     label: string; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Enum-like object for the visibility states of the password input. | ||||
|  */ | ||||
| const Visibility = { | ||||
|     Reveal: { | ||||
|         icon: "fa-eye", | ||||
|         label: msg("Show password"), | ||||
|     }, | ||||
|     Mask: { | ||||
|         icon: "fa-eye-slash", | ||||
|         label: msg("Hide password"), | ||||
|     }, | ||||
| } as const satisfies Record<string, VisibilityProps>; | ||||
|  | ||||
| @customElement("ak-flow-input-password") | ||||
| export class InputPassword extends AKElement { | ||||
|     static get styles() { | ||||
|         return [PFBase, PFInputGroup, PFFormControl, PFButton]; | ||||
|     } | ||||
|  | ||||
|     //#region Properties | ||||
|  | ||||
|     /** | ||||
|      * The ID of the input field. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: String, attribute: "input-id" }) | ||||
|     inputId = "ak-stage-password-input"; | ||||
|  | ||||
|     /** | ||||
|      * The name of the input field. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: String }) | ||||
|     name = "password"; | ||||
|  | ||||
|     /** | ||||
|      * The label for the input field. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: String }) | ||||
|     label = msg("Password"); | ||||
|  | ||||
|     /** | ||||
|      * The placeholder text for the input field. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: String }) | ||||
|     placeholder = msg("Please enter your password"); | ||||
|  | ||||
|     /** | ||||
|      * The initial value of the input field. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: String, attribute: "prefill" }) | ||||
|     passwordPrefill = ""; | ||||
|     initialValue = ""; | ||||
|  | ||||
|     /** | ||||
|      * The errors for the input field. | ||||
|      */ | ||||
|     @property({ type: Object }) | ||||
|     errors: Record<string, string> = {}; | ||||
|  | ||||
| @ -41,113 +98,220 @@ export class InputPassword extends AKElement { | ||||
|     @property({ type: String }) | ||||
|     invalid?: string; | ||||
|  | ||||
|     /** | ||||
|      * Whether to allow the user to toggle the visibility of the password. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: Boolean, attribute: "allow-show-password" }) | ||||
|     allowShowPassword = false; | ||||
|  | ||||
|     /** | ||||
|      * Whether the password is currently visible. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: Boolean, attribute: "password-visible" }) | ||||
|     passwordVisible = false; | ||||
|  | ||||
|     /** | ||||
|      * Automatically grab focus after rendering. | ||||
|      * | ||||
|      * @attr | ||||
|      */ | ||||
|     @property({ type: Boolean, attribute: "grab-focus" }) | ||||
|     grabFocus = false; | ||||
|  | ||||
|     timer?: number; | ||||
|     //#endregion | ||||
|  | ||||
|     input?: HTMLInputElement; | ||||
|     //#region Refs | ||||
|  | ||||
|     cleanup(): void { | ||||
|         if (this.timer) { | ||||
|             console.debug("authentik/stages/password: cleared focus timer"); | ||||
|             window.clearInterval(this.timer); | ||||
|             this.timer = undefined; | ||||
|     inputRef: Ref<HTMLInputElement> = createRef(); | ||||
|  | ||||
|     toggleVisibilityRef: Ref<HTMLButtonElement> = createRef(); | ||||
|  | ||||
|     //#endregion | ||||
|  | ||||
|     //#region State | ||||
|  | ||||
|     /** | ||||
|      * Whether the caps lock key is enabled. | ||||
|      */ | ||||
|     @state() | ||||
|     capsLock = false; | ||||
|  | ||||
|     //#endregion | ||||
|  | ||||
|     //#region Listeners | ||||
|  | ||||
|     /** | ||||
|      * Toggle the visibility of the password field. | ||||
|      * | ||||
|      * Directly affects the DOM, so no `.requestUpdate()` required. Effect is immediately visible. | ||||
|      * | ||||
|      * @param event The event that triggered the visibility toggle. | ||||
|      */ | ||||
|     @bound | ||||
|     togglePasswordVisibility(event?: PointerEvent) { | ||||
|         event?.stopPropagation(); | ||||
|         event?.preventDefault(); | ||||
|  | ||||
|         const input = this.inputRef.value; | ||||
|  | ||||
|         if (!input) { | ||||
|             console.warn("ak-flow-password-input: unable to identify input field"); | ||||
|  | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         input.type = input.type === "password" ? "text" : "password"; | ||||
|  | ||||
|         this.syncVisibilityToggle(input); | ||||
|     } | ||||
|  | ||||
|     // Must support both older browsers and shadyDom; we'll keep using this in-line, but it'll still | ||||
|     // be in the scope of the parent element, not an independent shadowDOM. | ||||
|     /** | ||||
|      * Listen for key events, synchronizing the caps lock indicators. | ||||
|      */ | ||||
|     @bound | ||||
|     capsLockListener(event: KeyboardEvent) { | ||||
|         this.capsLock = event.getModifierState("CapsLock"); | ||||
|     } | ||||
|  | ||||
|     //#region Lifecycle | ||||
|  | ||||
|     /** | ||||
|      * Interval ID for the focus observer. | ||||
|      * | ||||
|      * @see {@linkcode observeInputFocus} | ||||
|      */ | ||||
|     inputFocusIntervalID?: ReturnType<typeof setInterval>; | ||||
|  | ||||
|     /** | ||||
|      * Periodically attempt to focus the input field until it is focused. | ||||
|      * | ||||
|      * This is some-what of a crude way to get autofocus, but in most cases | ||||
|      * the `autofocus` attribute isn't enough, due to timing within shadow doms and such. | ||||
|      */ | ||||
|     observeInputFocus(): void { | ||||
|         if (!this.grabFocus) { | ||||
|             return; | ||||
|         } | ||||
|         this.inputFocusIntervalID = setInterval(() => { | ||||
|             const input = this.inputRef.value; | ||||
|  | ||||
|             if (!input) return; | ||||
|  | ||||
|             if (isActiveElement(input, document.activeElement)) { | ||||
|                 console.debug("authentik/stages/password: cleared focus observer"); | ||||
|                 clearInterval(this.inputFocusIntervalID); | ||||
|             } | ||||
|  | ||||
|             input.focus(); | ||||
|         }, 10); | ||||
|  | ||||
|         console.debug("authentik/stages/password: started focus observer"); | ||||
|     } | ||||
|  | ||||
|     connectedCallback() { | ||||
|         super.connectedCallback(); | ||||
|  | ||||
|         this.observeInputFocus(); | ||||
|  | ||||
|         addEventListener("keydown", this.capsLockListener); | ||||
|         addEventListener("keyup", this.capsLockListener); | ||||
|     } | ||||
|  | ||||
|     disconnectedCallback() { | ||||
|         if (this.inputFocusIntervalID) { | ||||
|             clearInterval(this.inputFocusIntervalID); | ||||
|         } | ||||
|  | ||||
|         super.disconnectedCallback(); | ||||
|  | ||||
|         removeEventListener("keydown", this.capsLockListener); | ||||
|         removeEventListener("keyup", this.capsLockListener); | ||||
|     } | ||||
|  | ||||
|     //#endregion | ||||
|  | ||||
|     //#region Render | ||||
|  | ||||
|     /** | ||||
|      * Create the render root for the password input. | ||||
|      * | ||||
|      * Must support both older browsers and shadyDom; we'll keep using this in-line, | ||||
|      * but it'll still be in the scope of the parent element, not an independent shadowDOM. | ||||
|      */ | ||||
|     createRenderRoot() { | ||||
|         return this; | ||||
|     } | ||||
|  | ||||
|     // State is saved in the DOM, and read from the DOM. Directly affects the DOM, | ||||
|     // so no `.requestUpdate()` required. Effect is immediately visible. | ||||
|     togglePasswordVisibility(ev: PointerEvent) { | ||||
|         const passwordField = this.renderRoot.querySelector(`#${this.inputId}`) as HTMLInputElement; | ||||
|         ev.stopPropagation(); | ||||
|         ev.preventDefault(); | ||||
|     /** | ||||
|      * Render the password visibility toggle button. | ||||
|      * | ||||
|      * In the unlikely event that we want to make "show password" the _default_ behavior, | ||||
|      * this effect handler is broken out into its own method. | ||||
|      * | ||||
|      * The current behavior in the main {@linkcode render} method assumes the field is of type "password." | ||||
|      * | ||||
|      * To have this effect, er, take effect, call it in an {@linkcode updated} method. | ||||
|      * | ||||
|      * @param input The password field to render the visibility features for. | ||||
|      */ | ||||
|     syncVisibilityToggle(input: HTMLInputElement | undefined = this.inputRef.value): void { | ||||
|         if (!input) return; | ||||
|  | ||||
|         if (!passwordField) { | ||||
|             throw new Error("ak-flow-password-input: unable to identify input field"); | ||||
|         } | ||||
|         const toggleElement = this.toggleVisibilityRef.value; | ||||
|  | ||||
|         passwordField.type = passwordField.type === "password" ? "text" : "password"; | ||||
|         this.renderPasswordVisibilityFeatures(passwordField); | ||||
|     } | ||||
|         if (!toggleElement) return; | ||||
|  | ||||
|     // In the unlikely event that we want to make "show password" the _default_ behavior, this | ||||
|     // effect handler is broken out into its own method. The current behavior in the main | ||||
|     // `.render()` method assumes the field is of type "password." To have this effect, er, take | ||||
|     // effect, call it in an `.updated()` method. | ||||
|     renderPasswordVisibilityFeatures(passwordField: HTMLInputElement) { | ||||
|         const toggleId = `#${this.inputId}-visibility-toggle`; | ||||
|         const visibilityToggle = this.renderRoot.querySelector(toggleId) as HTMLButtonElement; | ||||
|         if (!visibilityToggle) { | ||||
|             return; | ||||
|         } | ||||
|         const show = passwordField.type === "password"; | ||||
|         visibilityToggle?.setAttribute( | ||||
|         const masked = input.type === "password"; | ||||
|  | ||||
|         toggleElement.setAttribute( | ||||
|             "aria-label", | ||||
|             show ? msg("Show password") : msg("Hide password"), | ||||
|         ); | ||||
|         visibilityToggle?.querySelector("i")?.remove(); | ||||
|         render( | ||||
|             show | ||||
|                 ? html`<i class="fas fa-eye" aria-hidden="true"></i>` | ||||
|                 : html`<i class="fas fa-eye-slash" aria-hidden="true"></i>`, | ||||
|             visibilityToggle, | ||||
|             masked ? Visibility.Reveal.label : Visibility.Mask.label, | ||||
|         ); | ||||
|  | ||||
|         const iconElement = toggleElement.querySelector("i")!; | ||||
|  | ||||
|         iconElement.classList.remove(Visibility.Mask.icon, Visibility.Reveal.icon); | ||||
|         iconElement.classList.add(masked ? Visibility.Reveal.icon : Visibility.Mask.icon); | ||||
|     } | ||||
|  | ||||
|     renderInput(): HTMLInputElement { | ||||
|         this.input = document.createElement("input"); | ||||
|         this.input.id = `${this.inputId}`; | ||||
|         this.input.type = "password"; | ||||
|         this.input.name = this.name; | ||||
|         this.input.placeholder = this.placeholder; | ||||
|         this.input.autofocus = this.grabFocus; | ||||
|         this.input.autocomplete = "current-password"; | ||||
|         this.input.classList.add("pf-c-form-control"); | ||||
|         this.input.required = true; | ||||
|         this.input.value = this.passwordPrefill ?? ""; | ||||
|         if (this.invalid) { | ||||
|             this.input.setAttribute("aria-invalid", this.invalid); | ||||
|         } | ||||
|         // This is somewhat of a crude way to get autofocus, but in most cases the `autofocus` attribute | ||||
|         // isn't enough, due to timing within shadow doms and such. | ||||
|     renderVisibilityToggle() { | ||||
|         if (!this.allowShowPassword) return nothing; | ||||
|  | ||||
|         if (this.grabFocus) { | ||||
|             this.timer = window.setInterval(() => { | ||||
|                 if (!this.input) { | ||||
|                     return; | ||||
|                 } | ||||
|                 // Because activeElement behaves differently with shadow dom | ||||
|                 // we need to recursively check | ||||
|                 const rootEl = document.activeElement; | ||||
|                 const isActive = (el: Element | null): boolean => { | ||||
|                     if (!rootEl) return false; | ||||
|                     if (!("shadowRoot" in rootEl)) return false; | ||||
|                     if (rootEl.shadowRoot === null) return false; | ||||
|                     if (rootEl.shadowRoot.activeElement === el) return true; | ||||
|                     return isActive(rootEl.shadowRoot.activeElement); | ||||
|                 }; | ||||
|                 if (isActive(this.input)) { | ||||
|                     this.cleanup(); | ||||
|                 } | ||||
|                 this.input.focus(); | ||||
|             }, 10); | ||||
|             console.debug("authentik/stages/password: started focus timer"); | ||||
|         } | ||||
|         return this.input; | ||||
|         const { label, icon } = this.passwordVisible ? Visibility.Mask : Visibility.Reveal; | ||||
|  | ||||
|         return html`<button | ||||
|             ${ref(this.toggleVisibilityRef)} | ||||
|             aria-label=${label} | ||||
|             @click=${this.togglePasswordVisibility} | ||||
|             class="pf-c-button pf-m-control" | ||||
|             type="button" | ||||
|         > | ||||
|             <i class="fas ${icon}" aria-hidden="true"></i> | ||||
|         </button>`; | ||||
|     } | ||||
|  | ||||
|     renderHelperText() { | ||||
|         if (!this.capsLock) return nothing; | ||||
|  | ||||
|         return html`<div | ||||
|             class="pf-c-form__helper-text" | ||||
|             id="helper-text-form-caps-lock-helper" | ||||
|             aria-live="polite" | ||||
|         > | ||||
|             <div class="pf-c-helper-text"> | ||||
|                 <div class="pf-c-helper-text__item pf-m-warning"> | ||||
|                     <span class="pf-c-helper-text__item-icon"> | ||||
|                         <i class="fas fa-fw fa-exclamation-triangle" aria-hidden="true"></i> | ||||
|                     </span> | ||||
|  | ||||
|                     <span class="pf-c-helper-text__item-text">${msg("Caps Lock is enabled.")}</span> | ||||
|                 </div> | ||||
|             </div> | ||||
|         </div>`; | ||||
|     } | ||||
|  | ||||
|     render() { | ||||
| @ -157,22 +321,34 @@ export class InputPassword extends AKElement { | ||||
|             class="pf-c-form__group" | ||||
|             .errors=${this.errors} | ||||
|         > | ||||
|             <div class="pf-c-input-group"> | ||||
|                 ${this.renderInput()} | ||||
|                 ${this.allowShowPassword | ||||
|                     ? html` <button | ||||
|                           id="${this.inputId}-visibility-toggle" | ||||
|                           class="pf-c-button pf-m-control ak-stage-password-toggle-visibility" | ||||
|                           type="button" | ||||
|                           aria-label=${msg("Show password")} | ||||
|                           @click=${(ev: PointerEvent) => this.togglePasswordVisibility(ev)} | ||||
|                       > | ||||
|                           <i class="fas fa-eye" aria-hidden="true"></i> | ||||
|                       </button>` | ||||
|                     : nothing} | ||||
|             <div class="pf-c-form__group-control"> | ||||
|                 <div class="pf-c-input-group"> | ||||
|                     <input | ||||
|                         type=${this.passwordVisible ? "text" : "password"} | ||||
|                         id=${this.inputId} | ||||
|                         name=${this.name} | ||||
|                         placeholder=${this.placeholder} | ||||
|                         autocomplete="current-password" | ||||
|                         class="${classMap({ | ||||
|                             "pf-c-form-control": true, | ||||
|                             "pf-m-icon": true, | ||||
|                             "pf-m-caps-lock": this.capsLock, | ||||
|                         })}" | ||||
|                         required | ||||
|                         aria-invalid=${ifDefined(this.invalid)} | ||||
|                         value=${this.initialValue} | ||||
|                         ${ref(this.inputRef)} | ||||
|                     /> | ||||
|  | ||||
|                     ${this.renderVisibilityToggle()} | ||||
|                 </div> | ||||
|  | ||||
|                 ${this.renderHelperText()} | ||||
|             </div> | ||||
|         </ak-form-element>`; | ||||
|     } | ||||
|  | ||||
|     //#endregion | ||||
| } | ||||
|  | ||||
| declare global { | ||||
|  | ||||
| @ -3,7 +3,7 @@ import "@goauthentik/elements/forms/FormElement"; | ||||
| import { BaseDeviceStage } from "@goauthentik/flow/stages/authenticator_validate/base"; | ||||
| import { PasswordManagerPrefill } from "@goauthentik/flow/stages/identification/IdentificationStage"; | ||||
|  | ||||
| import { msg } from "@lit/localize"; | ||||
| import { msg, str } from "@lit/localize"; | ||||
| import { CSSResult, TemplateResult, css, html } from "lit"; | ||||
| import { customElement } from "lit/decorators.js"; | ||||
|  | ||||
| @ -35,7 +35,7 @@ export class AuthenticatorValidateStageWebCode extends BaseDeviceStage< | ||||
|         switch (this.deviceChallenge?.deviceClass) { | ||||
|             case DeviceClassesEnum.Email: { | ||||
|                 const email = this.deviceChallenge.challenge?.email; | ||||
|                 return msg(`A code has been sent to you via email${email ? ` ${email}` : ""}`); | ||||
|                 return msg(str`A code has been sent to you via email${email ? ` ${email}` : ""}`); | ||||
|             } | ||||
|             case DeviceClassesEnum.Sms: | ||||
|                 return msg("A code has been sent to you via SMS."); | ||||
|  | ||||
| @ -161,7 +161,7 @@ export class CaptchaStage extends BaseStage<CaptchaChallenge, CaptchaChallengeRe | ||||
|         super.disconnectedCallback(); | ||||
|     } | ||||
|  | ||||
|     get captchaDocumentContainer() { | ||||
|     get captchaDocumentContainer(): HTMLDivElement { | ||||
|         if (this._captchaDocumentContainer) { | ||||
|             return this._captchaDocumentContainer; | ||||
|         } | ||||
| @ -170,7 +170,7 @@ export class CaptchaStage extends BaseStage<CaptchaChallenge, CaptchaChallengeRe | ||||
|         return this._captchaDocumentContainer; | ||||
|     } | ||||
|  | ||||
|     get captchaFrame() { | ||||
|     get captchaFrame(): HTMLIFrameElement { | ||||
|         if (this._captchaFrame) { | ||||
|             return this._captchaFrame; | ||||
|         } | ||||
| @ -326,7 +326,7 @@ export class CaptchaStage extends BaseStage<CaptchaChallenge, CaptchaChallengeRe | ||||
|             .exhaustive(); | ||||
|     } | ||||
|  | ||||
|     updated(changedProperties: PropertyValues<this>) { | ||||
|     firstUpdated(changedProperties: PropertyValues<this>) { | ||||
|         if (!(changedProperties.has("challenge") && this.challenge !== undefined)) { | ||||
|             return; | ||||
|         } | ||||
|  | ||||
| @ -97,8 +97,19 @@ export class LibraryApplication extends AKElement { | ||||
|             return html``; | ||||
|         } | ||||
|         if (this.application?.launchUrl === "goauthentik.io://providers/rac/launch") { | ||||
|             return html`<ak-library-rac-endpoint-launch .app=${this.application}> | ||||
|                 </ak-library-rac-endpoint-launch> | ||||
|             return html`<div class="pf-c-card__header"> | ||||
|                     <a | ||||
|                         @click=${() => { | ||||
|                             this.racEndpointLaunch?.onClick(); | ||||
|                         }} | ||||
|                     > | ||||
|                         <ak-app-icon | ||||
|                             size=${PFSize.Large} | ||||
|                             name=${this.application.name} | ||||
|                             icon=${ifDefined(this.application.metaIcon || undefined)} | ||||
|                         ></ak-app-icon> | ||||
|                     </a> | ||||
|                 </div> | ||||
|                 <div class="pf-c-card__title"> | ||||
|                     <a | ||||
|                         @click=${() => { | ||||
| @ -107,15 +118,29 @@ export class LibraryApplication extends AKElement { | ||||
|                     > | ||||
|                         ${this.application.name} | ||||
|                     </a> | ||||
|                 </div>`; | ||||
|                 </div> | ||||
|                 <ak-library-rac-endpoint-launch .app=${this.application}> | ||||
|                 </ak-library-rac-endpoint-launch>`; | ||||
|         } | ||||
|         return html`<div class="pf-c-card__title"> | ||||
|             <a | ||||
|                 href="${ifDefined(this.application.launchUrl ?? "")}" | ||||
|                 target="${ifDefined(this.application.openInNewTab ? "_blank" : undefined)}" | ||||
|                 >${this.application.name}</a | ||||
|             > | ||||
|         </div>`; | ||||
|         return html`<div class="pf-c-card__header"> | ||||
|                 <a | ||||
|                     href="${ifDefined(this.application.launchUrl ?? "")}" | ||||
|                     target="${ifDefined(this.application.openInNewTab ? "_blank" : undefined)}" | ||||
|                 > | ||||
|                     <ak-app-icon | ||||
|                         size=${PFSize.Large} | ||||
|                         name=${this.application.name} | ||||
|                         icon=${ifDefined(this.application.metaIcon || undefined)} | ||||
|                     ></ak-app-icon> | ||||
|                 </a> | ||||
|             </div> | ||||
|             <div class="pf-c-card__title"> | ||||
|                 <a | ||||
|                     href="${ifDefined(this.application.launchUrl ?? "")}" | ||||
|                     target="${ifDefined(this.application.openInNewTab ? "_blank" : undefined)}" | ||||
|                     >${this.application.name}</a | ||||
|                 > | ||||
|             </div>`; | ||||
|     } | ||||
|  | ||||
|     render(): TemplateResult { | ||||
| @ -135,18 +160,6 @@ export class LibraryApplication extends AKElement { | ||||
|             class="pf-c-card pf-m-hoverable pf-m-compact ${classMap(classes)}" | ||||
|             style=${styleMap(styles)} | ||||
|         > | ||||
|             <div class="pf-c-card__header"> | ||||
|                 <a | ||||
|                     href="${ifDefined(this.application.launchUrl ?? "")}" | ||||
|                     target="${ifDefined(this.application.openInNewTab ? "_blank" : undefined)}" | ||||
|                 > | ||||
|                     <ak-app-icon | ||||
|                         size=${PFSize.Large} | ||||
|                         name=${this.application.name} | ||||
|                         icon=${ifDefined(this.application.metaIcon || undefined)} | ||||
|                     ></ak-app-icon> | ||||
|                 </a> | ||||
|             </div> | ||||
|             ${this.renderLaunch()} | ||||
|             <div class="expander"></div> | ||||
|             ${expandable ? this.renderExpansion(this.application) : nothing} | ||||
|  | ||||
| @ -42,7 +42,7 @@ export class LibraryPageApplicationEmptyList extends AKElement { | ||||
|  | ||||
|     renderNewAppButton() { | ||||
|         const href = paramURL("/core/applications", { | ||||
|             createForm: true, | ||||
|             createWizard: true, | ||||
|         }); | ||||
|         return html` | ||||
|             <div class="pf-u-pt-lg"> | ||||
|  | ||||
| @ -116,8 +116,13 @@ export class LibraryPage extends AKElement { | ||||
|     @bound | ||||
|     launchRequest(event: LibraryPageSearchSelected) { | ||||
|         event.stopPropagation(); | ||||
|         if (this.selectedApp?.launchUrl) { | ||||
|         if (!this.selectedApp?.launchUrl) { | ||||
|             return; | ||||
|         } | ||||
|         if (!this.selectedApp.openInNewTab) { | ||||
|             window.location.assign(this.selectedApp?.launchUrl); | ||||
|         } else { | ||||
|             window.open(this.selectedApp.launchUrl); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
| @ -32,7 +32,7 @@ export class UserSettingsPassword extends AKElement { | ||||
|             <div class="pf-c-card__body"> | ||||
|                 <a | ||||
|                     href="${ifDefined(this.configureUrl)}${AndNext( | ||||
|                         `${globalAK().api.base}if/user/#/settings;${JSON.stringify({ page: "page-details" })}`, | ||||
|                         `${globalAK().api.relBase}if/user/#/settings;${JSON.stringify({ page: "page-details" })}`, | ||||
|                     )}" | ||||
|                     class="pf-c-button pf-m-primary" | ||||
|                 > | ||||
|  | ||||
| @ -10,7 +10,7 @@ import { StageHost } from "@goauthentik/flow/stages/base"; | ||||
| import "@goauthentik/user/user-settings/details/stages/prompt/PromptStage"; | ||||
|  | ||||
| import { msg } from "@lit/localize"; | ||||
| import { CSSResult, TemplateResult, html } from "lit"; | ||||
| import { CSSResult, PropertyValues, TemplateResult, html } from "lit"; | ||||
| import { customElement, property } from "lit/decorators.js"; | ||||
| import { unsafeHTML } from "lit/directives/unsafe-html.js"; | ||||
|  | ||||
| @ -83,12 +83,14 @@ export class UserSettingsFlowExecutor | ||||
|             }); | ||||
|     } | ||||
|  | ||||
|     firstUpdated(): void { | ||||
|         this.flowSlug = this.brand?.flowUserSettings; | ||||
|         if (!this.flowSlug) { | ||||
|             return; | ||||
|     updated(changedProperties: PropertyValues<this>): void { | ||||
|         if (changedProperties.has("brand") && this.brand) { | ||||
|             this.flowSlug = this.brand?.flowUserSettings; | ||||
|             if (!this.flowSlug) { | ||||
|                 return; | ||||
|             } | ||||
|             this.nextChallenge(); | ||||
|         } | ||||
|         this.nextChallenge(); | ||||
|     } | ||||
|  | ||||
|     async nextChallenge(): Promise<void> { | ||||
| @ -161,7 +163,7 @@ export class UserSettingsFlowExecutor | ||||
|                 // Flow has finished, so let's load while in the background we can restart the flow | ||||
|                 this.loading = true; | ||||
|                 console.debug("authentik/user/flows: redirect to '/', restarting flow."); | ||||
|                 this.firstUpdated(); | ||||
|                 this.nextChallenge(); | ||||
|                 this.globalRefresh(); | ||||
|                 showMessage({ | ||||
|                     level: MessageLevel.success, | ||||
|  | ||||
| @ -74,7 +74,7 @@ export class MFADevicesPage extends Table<Device> { | ||||
|                         return html`<li> | ||||
|                             <a | ||||
|                                 href="${ifDefined(stage.configureUrl)}${AndNext( | ||||
|                                     `${globalAK().api.base}if/user/#/settings;${JSON.stringify({ | ||||
|                                     `${globalAK().api.relBase}if/user/#/settings;${JSON.stringify({ | ||||
|                                         page: "page-mfa", | ||||
|                                     })}`, | ||||
|                                 )}" | ||||
|  | ||||
| @ -89,7 +89,7 @@ export async function findWizardTitle() { | ||||
| async function passByPoliciesAndCommit() { | ||||
|     const title = await findWizardTitle(); | ||||
|     // Expect to be on the Bindings panel | ||||
|     await expect(await title.getText()).toEqual("Configure Policy Bindings"); | ||||
|     await expect(await title.getText()).toEqual("Configure Policy/User/Group Bindings"); | ||||
|     await (await ApplicationWizardView.nextButton()).click(); | ||||
|     await ApplicationWizardView.pause(); | ||||
|     await (await ApplicationWizardView.submitPage()).waitForDisplayed(); | ||||
|  | ||||
| @ -4,29 +4,21 @@ title: Manage applications | ||||
|  | ||||
| Managing the applications that your team uses involves several tasks, from initially adding the application and provider, to controlling access and visibility of the application, to providing access URLs. | ||||
|  | ||||
| ## Add new applications | ||||
|  | ||||
| Learn how to add new applications from our video or follow the instructions below. | ||||
|  | ||||
| ### Video | ||||
|  | ||||
| <iframe width="560" height="315" src="https://www.youtube.com/embed/broUAWrIWDI;start=22" title="YouTube video player" frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share" allowfullscreen></iframe> | ||||
|  | ||||
| ### Instructions | ||||
|  | ||||
| To add an application to authentik and have it display on users' **My applications** page, you can use the Application Wizard, which creates both the new application and the required provider at the same time. | ||||
| To add an application to authentik and have it display on users' **My applications** page, follow these steps: | ||||
|  | ||||
| 1. Log into authentik as an admin, and navigate to **Applications --> Applications**. | ||||
| 1. Log in to authentik as an admin, and open the authentik Admin interface. | ||||
|  | ||||
| 2. Click **Create with Wizard**. (Alternatively, use our legacy process and click **Create**. The legacy process requires that the application and its authentication provider be configured separately.) | ||||
| 2. Navigate to **Applications -> Applications** and click **Create with Provider** to create an application and provider pair. (Alternatively you can create only an application, without a provider, by clicking **Create.)** | ||||
|  | ||||
| 3. In the **New application** wizard, define the application details, the provider type, bindings for the application. | ||||
| 3. In the **New application** box, define the application details, the provider type and configuration settings, and bindings for the application. | ||||
|  | ||||
|     - **Application**: provide a name, an optional group for the type of application, the policy engine mode, and optional UI settings. | ||||
|  | ||||
|     - **Choose a Provider**: select the provider types for this application. | ||||
|  | ||||
|     - **Configure a Provider**: provide a name (or accept the auto-provided name), the authorization flow to use for this provider, and any additional required configurations. | ||||
|     - **Configure the Provider**: provide a name (or accept the auto-provided name), the authorization flow to use for this provider, and any additional required configurations. | ||||
|  | ||||
|     - **Configure Bindings**: to manage the listing and access to applications on a user's **My applications** page, you can optionally create a [binding](../flows-stages/bindings/index.md) between the application and a specific policy, group, or user. Note that if you do not define any bindings, then all users have access to the application. For more information about user access, refer to our documentation about [authorization](#policy-driven-authorization) and [hiding an application](#hide-applications). | ||||
|  | ||||
| @ -83,8 +75,8 @@ return { | ||||
| 3. Click the **Application entitlements** tab at the top of the page, and then click **Create entitlement**. Provide a name for the entitlement, enter any optional **Attributes**, and then click **Create**. | ||||
| 4. In the list locate the entitlement to which you want to bind a user or group, and then **click the caret (>) to expand the entitlement details.** | ||||
| 5. In the expanded area, click **Bind existing Group/User**. | ||||
| 6. In the **Create Binding** modal box, select either the tab for **Group** or **User**, and then in the drop-down list, select the group or user. | ||||
| 7. Optionally, configure additional settings for the binding, and then click **Create** to create the binding and close the modal box. | ||||
| 6. In the **Create Binding** box, select either the tab for **Group** or **User**, and then in the drop-down list, select the group or user. | ||||
| 7. Optionally, configure additional settings for the binding, and then click **Create** to create the binding and close the box. | ||||
|  | ||||
| ## Hide applications | ||||
|  | ||||
|  | ||||
| @ -9,5 +9,5 @@ For instructions to create a binding, refer to the documentation for the specifi | ||||
| - [Bind a stage to a flow](../stages/index.md#bind-a-stage-to-a-flow) | ||||
| - [Bind a policy to a flow or stage](../../../customize/policies/working_with_policies#bind-a-policy-to-a-flow-or-stage) | ||||
| - [Bind users or groups to a specific application with an Application Entitlement](../../applications/manage_apps.md#application-entitlements) | ||||
| - [Bind a policy to a specific application when you create a new app using the Wizard](../../applications/manage_apps.md#instructions) | ||||
| - [Bind a policy to a specific application when you create a new application and provider](../../applications/manage_apps.md#instructions) | ||||
| - [Bind users and groups to a stage binding, to define whether or not that stage is shown](../stages/index.md#bind-users-and-groups-to-a-flows-stage-binding) | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: Duo authenticator setup stage | ||||
| title: Duo Authenticator Setup stage | ||||
| --- | ||||
|  | ||||
| This stage configures a Duo authenticator. To get the API Credentials for this stage, open your Duo Admin dashboard. | ||||
|  | ||||
| @ -0,0 +1,48 @@ | ||||
| --- | ||||
| title: Email Authenticator Setup stage | ||||
| --- | ||||
|  | ||||
| <span class="badge badge--version">authentik 2025.2+</span> | ||||
|  | ||||
| This stage configures an email-based authenticator that sends a one-time code to a user's email address for authentication. | ||||
|  | ||||
| When a user goes through a flow that includes this stage, they are prompted for their email address (if not already set). The user then receives an email with a one-time code, which they enter into the authentik Login panel. | ||||
|  | ||||
| The email address will be saved and can be used with the [Authenticator validation](../authenticator_validate/index.md) stage for future authentications. | ||||
|  | ||||
| ## Flow integration | ||||
|  | ||||
| To use the Email Authenticator Setup stage in a flow, follow these steps: | ||||
|  | ||||
| 1. [Create](../../flow/index.md#create-a-custom-flow) a new flow or edit an existing one. | ||||
| 2. On the flow's **Stage Bindings** tab, click **Create and bind stage** to create and add the Email Authenticator Setup stage. (If the stage already exists, click **Bind existing stage**.) | ||||
| 3. Configure the stage settings as described below. | ||||
|  | ||||
|     - **Name**: provide a descriptive name, such as Email Authenticator Setup. | ||||
|     - **Authenticator type name**: define the display name for this stage. | ||||
|     - **Use global connection settings**: the stage can be configured in two ways: global settings or stage-specific settings. | ||||
|  | ||||
|         - Enable (toggle on) the **Use global connection settings** option to use authentik's global email configuration. Note that you must already have configured your environment variables to use the global settings. See instructions for [Docker Compose](../../../../install-config/install/docker-compose#email-configuration-optional-but-recommended) and for [Kubernetes](../../../../install-config/install/kubernetes#optional-step-configure-global-email-credentials). | ||||
|  | ||||
|         - If you need different email settings for this stage, disable (toggle off) **Use global connection settings** and configure the following options: | ||||
|  | ||||
|         - **Connection settings**: | ||||
|  | ||||
|             - **SMTP Host**: SMTP server hostname (default: localhost) | ||||
|             - **SMTP Port**: SMTP server port number(default: 25) | ||||
|             - **SMTP Username**: SMTP authentication username (optional) | ||||
|             - **SMTP Password**: SMTP authentication password (optional) | ||||
|                 - **Use TLS**: Enable TLS encryption | ||||
|                 - **Use SSL**: Enable SSL encryption | ||||
|             - **Timeout**: Connection timeout in seconds (default: 10) | ||||
|             - **From Address**: Email address that messages are sent from (default: system@authentik.local) | ||||
|  | ||||
|         - **Stage-specific settings**: | ||||
|  | ||||
|             - **Subject**: Email subject line (default: "authentik Sign-in code") | ||||
|             - **Token Expiration**: Time in minutes that the sent token is valid (default: 30) | ||||
|             - **Configuration flow**: select the flow to which you are binding this stage. | ||||
|  | ||||
| 4. Click **Update** to complete the creation and binding of the stage to the flow. | ||||
|  | ||||
| The new Email Authenticator Setup stage now appears on the **Stage Bindings** tab for the flow. | ||||
| @ -28,7 +28,7 @@ For detailed instructions, refer to Google documentation. | ||||
| ### Create a Google cloud project | ||||
|  | ||||
| 1. Open the Google Cloud Console (https://cloud.google.com/cloud-console). | ||||
| 2. In upper left, click the drop-down box to open the **Select a project** modal box, and then select **New Project**. | ||||
| 2. In upper left, click the drop-down box to open the **Select a project** box, and then select **New Project**. | ||||
| 3. Create a new project and give it a name like "authentik GWS". | ||||
| 4. Use the search bar at the top of your new project page to search for "API Library". | ||||
| 5. On the **API Library** page, use the search bar again to find "Chrome Verified Access API". | ||||
| @ -49,7 +49,7 @@ For detailed instructions, refer to Google documentation. | ||||
|  | ||||
| 1. On the **Service accounts** page, click the account that you just created. | ||||
| 2. Click the **Keys** tab at top of the page, the click **Add Key -> Create new key**. | ||||
| 3. In the Create modal box, select JSON as the key type, and then click **Create**. | ||||
| 3. In the Create box, select JSON as the key type, and then click **Create**. | ||||
|    A pop-up displays with the private key, and the key is saved to your computer as a JSON file. | ||||
|    Later, when you create the stage in authentik, you will add this key in the **Credentials** field. | ||||
| 4. On the service account page, click the **Details** tab, and expand the **Advanced settings** area. | ||||
| @ -66,7 +66,7 @@ For detailed instructions, refer to Google documentation. | ||||
|  | ||||
| 2. In the Admin interface, navigate to **Flows -> Stages**. | ||||
|  | ||||
| 3. Click **Create**, and select **Endpoint Authenticator Google Device Trust Connector Stage**, and in the **New stage** modal box, define the following fields: | ||||
| 3. Click **Create**, and select **Endpoint Authenticator Google Device Trust Connector Stage**, and in the **New stage** box, define the following fields: | ||||
|  | ||||
|     - **Name**: define a descriptive name, such as "chrome-device-trust". | ||||
|  | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: SMS authenticator setup stage | ||||
| title: SMS Authenticator Setup stage | ||||
| --- | ||||
|  | ||||
| This stage configures an SMS-based authenticator using either Twilio, or a generic HTTP endpoint. | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: Static authenticator setup stage | ||||
| title: Static Authenticator Setup stage | ||||
| --- | ||||
|  | ||||
| This stage configures static Tokens, which can be used as a backup method to time-based OTP tokens. | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: TOTP authenticator setup stage | ||||
| title: TOTP Authenticator Setup stage | ||||
| --- | ||||
|  | ||||
| This stage configures a time-based OTP Device, such as Google Authenticator or Authy. | ||||
|  | ||||
| @ -1,10 +1,11 @@ | ||||
| --- | ||||
| title: Authenticator validation stage | ||||
| title: Authenticator Validation stage | ||||
| --- | ||||
|  | ||||
| This stage validates an already configured Authenticator Device. This device has to be configured using any of the other authenticator stages: | ||||
|  | ||||
| - [Duo authenticator stage](../authenticator_duo/index.md) | ||||
| - [Email authenticator stage](../authenticator_email/index.md) | ||||
| - [SMS authenticator stage](../authenticator_sms/index.md) | ||||
| - [Static authenticator stage](../authenticator_static/index.md) | ||||
| - [TOTP authenticator stage](../authenticator_totp/index.md) | ||||
|  | ||||
| @ -1,5 +1,5 @@ | ||||
| --- | ||||
| title: WebAuthn authenticator setup stage | ||||
| title: WebAuthn Authenticator Setup stage | ||||
| --- | ||||
|  | ||||
| This stage configures a WebAuthn-based Authenticator. This can either be a browser, biometrics or a Security stick like a YubiKey. | ||||
|  | ||||
| @ -70,8 +70,8 @@ To bind a user or a group to a stage binding for a specific flow, follow these s | ||||
|  | ||||
|  | ||||
| 6. In the expanded area, click **Bind existing policy/group/user**. | ||||
| 7. In the **Create Binding** modal box, select either the tab for **Group** or **User**. | ||||
| 7. In the **Create Binding** box, select either the tab for **Group** or **User**. | ||||
| 8. In the drop-down list, select the group or user. | ||||
| 9. Optionally, configure additional settings for the binding, and then click **Create** to create the binding and close the modal box. | ||||
| 9. Optionally, configure additional settings for the binding, and then click **Create** to create the binding and close the box. | ||||
|  | ||||
| Learn more about [bindings](../bindings/index.md) and [working with them](../bindings/work_with_bindings.md). | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	