wip: rename to authentik (#361)
* root: initial rename * web: rename custom element prefix * root: rename external functions with pb_ prefix * root: fix formatting * root: replace domain with goauthentik.io * proxy: update path * root: rename remaining prefixes * flows: rename file extension * root: pbadmin -> akadmin * docs: fix image filenames * lifecycle: ignore migration files * ci: copy default config from current source before loading last tagged * *: new sentry dsn * tests: fix missing python3.9-dev package * root: add additional migrations for service accounts created by outposts * core: mark system-created service accounts with attribute * policies/expression: fix pb_ replacement not working * web: fix last linting errors, add lit-analyse * policies/expressions: fix lint errors * web: fix sidebar display on screens where not all items fit * proxy: attempt to fix proxy pipeline * proxy: use go env GOPATH to get gopath * lib: fix user_default naming inconsistency * docs: add upgrade docs * docs: update screenshots to use authentik * admin: fix create button on empty-state of outpost * web: fix modal submit not refreshing SiteShell and Table * web: fix height of app-card and height of generic icon * web: fix rendering of subtext * admin: fix version check error not being caught * web: fix worker count not being shown * docs: update screenshots * root: new icon * web: fix lint error * admin: fix linting error * root: migrate coverage config to pyproject
This commit is contained in:
parent
810a7ab50b
commit
1cfe1aff13
|
@ -27,7 +27,7 @@ values =
|
||||||
|
|
||||||
[bumpversion:file:.github/workflows/release.yml]
|
[bumpversion:file:.github/workflows/release.yml]
|
||||||
|
|
||||||
[bumpversion:file:passbook/__init__.py]
|
[bumpversion:file:authentik/__init__.py]
|
||||||
|
|
||||||
[bumpversion:file:proxy/pkg/version.go]
|
[bumpversion:file:proxy/pkg/version.go]
|
||||||
|
|
||||||
|
|
33
.coveragerc
33
.coveragerc
|
@ -1,33 +0,0 @@
|
||||||
[run]
|
|
||||||
source = passbook
|
|
||||||
relative_files = true
|
|
||||||
omit =
|
|
||||||
*/asgi.py
|
|
||||||
manage.py
|
|
||||||
*/migrations/*
|
|
||||||
*/apps.py
|
|
||||||
website/
|
|
||||||
|
|
||||||
[report]
|
|
||||||
sort = Cover
|
|
||||||
skip_covered = True
|
|
||||||
precision = 2
|
|
||||||
exclude_lines =
|
|
||||||
pragma: no cover
|
|
||||||
|
|
||||||
# Don't complain about missing debug-only code:
|
|
||||||
def __unicode__
|
|
||||||
def __str__
|
|
||||||
def __repr__
|
|
||||||
if self\.debug
|
|
||||||
if TYPE_CHECKING
|
|
||||||
|
|
||||||
# Don't complain if tests don't hit defensive assertion code:
|
|
||||||
raise AssertionError
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
# Don't complain if non-runnable code isn't run:
|
|
||||||
if 0:
|
|
||||||
if __name__ == .__main__.:
|
|
||||||
|
|
||||||
show_missing = True
|
|
|
@ -27,7 +27,7 @@ If applicable, add screenshots to help explain your problem.
|
||||||
Output of docker-compose logs or kubectl logs respectively
|
Output of docker-compose logs or kubectl logs respectively
|
||||||
|
|
||||||
**Version and Deployment (please complete the following information):**
|
**Version and Deployment (please complete the following information):**
|
||||||
- passbook version: [e.g. 0.10.0-stable]
|
- authentik version: [e.g. 0.10.0-stable]
|
||||||
- Deployment: [e.g. docker-compose, helm]
|
- Deployment: [e.g. docker-compose, helm]
|
||||||
|
|
||||||
**Additional context**
|
**Additional context**
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
name: passbook-on-release
|
name: authentik-on-release
|
||||||
|
|
||||||
on:
|
on:
|
||||||
release:
|
release:
|
||||||
|
@ -18,13 +18,13 @@ jobs:
|
||||||
- name: Building Docker Image
|
- name: Building Docker Image
|
||||||
run: docker build
|
run: docker build
|
||||||
--no-cache
|
--no-cache
|
||||||
-t beryju/passbook:0.12.11-stable
|
-t beryju/authentik:0.12.11-stable
|
||||||
-t beryju/passbook:latest
|
-t beryju/authentik:latest
|
||||||
-f Dockerfile .
|
-f Dockerfile .
|
||||||
- name: Push Docker Container to Registry (versioned)
|
- name: Push Docker Container to Registry (versioned)
|
||||||
run: docker push beryju/passbook:0.12.11-stable
|
run: docker push beryju/authentik:0.12.11-stable
|
||||||
- name: Push Docker Container to Registry (latest)
|
- name: Push Docker Container to Registry (latest)
|
||||||
run: docker push beryju/passbook:latest
|
run: docker push beryju/authentik:latest
|
||||||
build-proxy:
|
build-proxy:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
|
@ -36,7 +36,7 @@ jobs:
|
||||||
run: |
|
run: |
|
||||||
cd proxy
|
cd proxy
|
||||||
go get -u github.com/go-swagger/go-swagger/cmd/swagger
|
go get -u github.com/go-swagger/go-swagger/cmd/swagger
|
||||||
swagger generate client -f ../swagger.yaml -A passbook -t pkg/
|
swagger generate client -f ../swagger.yaml -A authentik -t pkg/
|
||||||
go build -v .
|
go build -v .
|
||||||
- name: Docker Login Registry
|
- name: Docker Login Registry
|
||||||
env:
|
env:
|
||||||
|
@ -48,13 +48,13 @@ jobs:
|
||||||
cd proxy/
|
cd proxy/
|
||||||
docker build \
|
docker build \
|
||||||
--no-cache \
|
--no-cache \
|
||||||
-t beryju/passbook-proxy:0.12.11-stable \
|
-t beryju/authentik-proxy:0.12.11-stable \
|
||||||
-t beryju/passbook-proxy:latest \
|
-t beryju/authentik-proxy:latest \
|
||||||
-f Dockerfile .
|
-f Dockerfile .
|
||||||
- name: Push Docker Container to Registry (versioned)
|
- name: Push Docker Container to Registry (versioned)
|
||||||
run: docker push beryju/passbook-proxy:0.12.11-stable
|
run: docker push beryju/authentik-proxy:0.12.11-stable
|
||||||
- name: Push Docker Container to Registry (latest)
|
- name: Push Docker Container to Registry (latest)
|
||||||
run: docker push beryju/passbook-proxy:latest
|
run: docker push beryju/authentik-proxy:latest
|
||||||
build-static:
|
build-static:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
|
@ -69,13 +69,13 @@ jobs:
|
||||||
cd web/
|
cd web/
|
||||||
docker build \
|
docker build \
|
||||||
--no-cache \
|
--no-cache \
|
||||||
-t beryju/passbook-static:0.12.11-stable \
|
-t beryju/authentik-static:0.12.11-stable \
|
||||||
-t beryju/passbook-static:latest \
|
-t beryju/authentik-static:latest \
|
||||||
-f Dockerfile .
|
-f Dockerfile .
|
||||||
- name: Push Docker Container to Registry (versioned)
|
- name: Push Docker Container to Registry (versioned)
|
||||||
run: docker push beryju/passbook-static:0.12.11-stable
|
run: docker push beryju/authentik-static:0.12.11-stable
|
||||||
- name: Push Docker Container to Registry (latest)
|
- name: Push Docker Container to Registry (latest)
|
||||||
run: docker push beryju/passbook-static:latest
|
run: docker push beryju/authentik-static:latest
|
||||||
test-release:
|
test-release:
|
||||||
needs:
|
needs:
|
||||||
- build-server
|
- build-server
|
||||||
|
@ -87,11 +87,11 @@ jobs:
|
||||||
run: |
|
run: |
|
||||||
sudo apt-get install -y pwgen
|
sudo apt-get install -y pwgen
|
||||||
echo "PG_PASS=$(pwgen 40 1)" >> .env
|
echo "PG_PASS=$(pwgen 40 1)" >> .env
|
||||||
echo "PASSBOOK_SECRET_KEY=$(pwgen 50 1)" >> .env
|
echo "AUTHENTIK_SECRET_KEY=$(pwgen 50 1)" >> .env
|
||||||
docker-compose pull -q
|
docker-compose pull -q
|
||||||
docker-compose up --no-start
|
docker-compose up --no-start
|
||||||
docker-compose start postgresql redis
|
docker-compose start postgresql redis
|
||||||
docker-compose run -u root --entrypoint /bin/bash server -c "pip install --no-cache -r requirements-dev.txt && ./manage.py test passbook"
|
docker-compose run -u root --entrypoint /bin/bash server -c "pip install --no-cache -r requirements-dev.txt && ./manage.py test authentik"
|
||||||
sentry-release:
|
sentry-release:
|
||||||
needs:
|
needs:
|
||||||
- test-release
|
- test-release
|
||||||
|
@ -103,7 +103,7 @@ jobs:
|
||||||
env:
|
env:
|
||||||
SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }}
|
SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }}
|
||||||
SENTRY_ORG: beryjuorg
|
SENTRY_ORG: beryjuorg
|
||||||
SENTRY_PROJECT: passbook
|
SENTRY_PROJECT: authentik
|
||||||
SENTRY_URL: https://sentry.beryju.org
|
SENTRY_URL: https://sentry.beryju.org
|
||||||
with:
|
with:
|
||||||
tagName: 0.12.11-stable
|
tagName: 0.12.11-stable
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
name: passbook-on-tag
|
name: authentik-on-tag
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
|
@ -14,17 +14,17 @@ jobs:
|
||||||
- name: Pre-release test
|
- name: Pre-release test
|
||||||
run: |
|
run: |
|
||||||
sudo apt-get install -y pwgen
|
sudo apt-get install -y pwgen
|
||||||
echo "PASSBOOK_TAG=latest" >> .env
|
echo "AUTHENTIK_TAG=latest" >> .env
|
||||||
echo "PG_PASS=$(pwgen 40 1)" >> .env
|
echo "PG_PASS=$(pwgen 40 1)" >> .env
|
||||||
echo "PASSBOOK_SECRET_KEY=$(pwgen 50 1)" >> .env
|
echo "AUTHENTIK_SECRET_KEY=$(pwgen 50 1)" >> .env
|
||||||
docker-compose pull -q
|
docker-compose pull -q
|
||||||
docker build \
|
docker build \
|
||||||
--no-cache \
|
--no-cache \
|
||||||
-t beryju/passbook:latest \
|
-t beryju/authentik:latest \
|
||||||
-f Dockerfile .
|
-f Dockerfile .
|
||||||
docker-compose up --no-start
|
docker-compose up --no-start
|
||||||
docker-compose start postgresql redis
|
docker-compose start postgresql redis
|
||||||
docker-compose run -u root --entrypoint /bin/bash server -c "pip install --no-cache -r requirements-dev.txt && ./manage.py test passbook"
|
docker-compose run -u root --entrypoint /bin/bash server -c "pip install --no-cache -r requirements-dev.txt && ./manage.py test authentik"
|
||||||
- name: Install Helm
|
- name: Install Helm
|
||||||
run: |
|
run: |
|
||||||
apt update && apt install -y curl
|
apt update && apt install -y curl
|
||||||
|
@ -33,7 +33,7 @@ jobs:
|
||||||
run: |
|
run: |
|
||||||
helm dependency update helm/
|
helm dependency update helm/
|
||||||
helm package helm/
|
helm package helm/
|
||||||
mv passbook-*.tgz passbook-chart.tgz
|
mv authentik-*.tgz authentik-chart.tgz
|
||||||
- name: Extract version number
|
- name: Extract version number
|
||||||
id: get_version
|
id: get_version
|
||||||
uses: actions/github-script@0.2.0
|
uses: actions/github-script@0.2.0
|
||||||
|
@ -58,6 +58,6 @@ jobs:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
with:
|
with:
|
||||||
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
upload_url: ${{ steps.create_release.outputs.upload_url }}
|
||||||
asset_path: ./passbook-chart.tgz
|
asset_path: ./authentik-chart.tgz
|
||||||
asset_name: passbook-chart.tgz
|
asset_name: authentik-chart.tgz
|
||||||
asset_content_type: application/gzip
|
asset_content_type: application/gzip
|
||||||
|
|
12
Dockerfile
12
Dockerfile
|
@ -30,18 +30,18 @@ RUN apt-get update && \
|
||||||
# but then we have to drop permmissions later
|
# but then we have to drop permmissions later
|
||||||
groupadd -g 998 docker_998 && \
|
groupadd -g 998 docker_998 && \
|
||||||
groupadd -g 999 docker_999 && \
|
groupadd -g 999 docker_999 && \
|
||||||
adduser --system --no-create-home --uid 1000 --group --home /passbook passbook && \
|
adduser --system --no-create-home --uid 1000 --group --home /authentik authentik && \
|
||||||
usermod -a -G docker_998 passbook && \
|
usermod -a -G docker_998 authentik && \
|
||||||
usermod -a -G docker_999 passbook && \
|
usermod -a -G docker_999 authentik && \
|
||||||
mkdir /backups && \
|
mkdir /backups && \
|
||||||
chown passbook:passbook /backups
|
chown authentik:authentik /backups
|
||||||
|
|
||||||
COPY ./passbook/ /passbook
|
COPY ./authentik/ /authentik
|
||||||
COPY ./pytest.ini /
|
COPY ./pytest.ini /
|
||||||
COPY ./manage.py /
|
COPY ./manage.py /
|
||||||
COPY ./lifecycle/ /lifecycle
|
COPY ./lifecycle/ /lifecycle
|
||||||
|
|
||||||
USER passbook
|
USER authentik
|
||||||
STOPSIGNAL SIGINT
|
STOPSIGNAL SIGINT
|
||||||
ENV TMPDIR /dev/shm/
|
ENV TMPDIR /dev/shm/
|
||||||
ENTRYPOINT [ "/lifecycle/bootstrap.sh" ]
|
ENTRYPOINT [ "/lifecycle/bootstrap.sh" ]
|
||||||
|
|
18
Makefile
18
Makefile
|
@ -9,30 +9,30 @@ test-e2e:
|
||||||
coverage run manage.py test --failfast -v 3 tests/e2e
|
coverage run manage.py test --failfast -v 3 tests/e2e
|
||||||
|
|
||||||
coverage:
|
coverage:
|
||||||
coverage run manage.py test --failfast -v 3 passbook
|
coverage run manage.py test --failfast -v 3 authentik
|
||||||
coverage html
|
coverage html
|
||||||
coverage report
|
coverage report
|
||||||
|
|
||||||
lint-fix:
|
lint-fix:
|
||||||
isort -rc .
|
isort -rc authentik tests lifecycle
|
||||||
black passbook tests lifecycle
|
black authentik tests lifecycle
|
||||||
|
|
||||||
lint:
|
lint:
|
||||||
pyright passbook tests lifecycle
|
pyright authentik tests lifecycle
|
||||||
bandit -r passbook tests lifecycle -x node_modules
|
bandit -r authentik tests lifecycle -x node_modules
|
||||||
pylint passbook tests lifecycle
|
pylint authentik tests lifecycle
|
||||||
prospector
|
prospector
|
||||||
|
|
||||||
gen: coverage
|
gen: coverage
|
||||||
./manage.py generate_swagger -o swagger.yaml -f yaml
|
./manage.py generate_swagger -o swagger.yaml -f yaml
|
||||||
|
|
||||||
local-stack:
|
local-stack:
|
||||||
export PASSBOOK_TAG=testing
|
export AUTHENTIK_TAG=testing
|
||||||
docker build -t beryju/passbook:testng .
|
docker build -t beryju/authentik:testng .
|
||||||
docker-compose up -d
|
docker-compose up -d
|
||||||
docker-compose run --rm server migrate
|
docker-compose run --rm server migrate
|
||||||
|
|
||||||
build-static:
|
build-static:
|
||||||
docker-compose -f scripts/ci.docker-compose.yml up -d
|
docker-compose -f scripts/ci.docker-compose.yml up -d
|
||||||
docker build -t beryju/passbook-static -f static.Dockerfile --network=scripts_default .
|
docker build -t beryju/authentik-static -f static.Dockerfile --network=scripts_default .
|
||||||
docker-compose -f scripts/ci.docker-compose.yml down -v
|
docker-compose -f scripts/ci.docker-compose.yml down -v
|
||||||
|
|
26
README.md
26
README.md
|
@ -1,21 +1,23 @@
|
||||||
<img src="website/static/img/logo.svg" height="50" alt="passbook logo"><img src="website/static/img/brand_inverted.svg" height="50" alt="passbook">
|
<img src="icons/icon_top_brand.svg" height="250" alt="authentik logo">
|
||||||
|
|
||||||
[![CI Build status](https://img.shields.io/azure-devops/build/beryjuorg/passbook/1?style=flat-square)](https://dev.azure.com/beryjuorg/passbook/_build?definitionId=1)
|
---
|
||||||
![Tests](https://img.shields.io/azure-devops/tests/beryjuorg/passbook/1?compact_message&style=flat-square)
|
|
||||||
[![Code Coverage](https://img.shields.io/codecov/c/gh/beryju/passbook?style=flat-square)](https://codecov.io/gh/BeryJu/passbook)
|
|
||||||
![Docker pulls](https://img.shields.io/docker/pulls/beryju/passbook.svg?style=flat-square)
|
|
||||||
![Latest version](https://img.shields.io/docker/v/beryju/passbook?sort=semver&style=flat-square)
|
|
||||||
![LGTM Grade](https://img.shields.io/lgtm/grade/python/github/BeryJu/passbook?style=flat-square)
|
|
||||||
|
|
||||||
## What is passbook?
|
[![CI Build status](https://img.shields.io/azure-devops/build/beryjuorg/authentik/1?style=flat-square)](https://dev.azure.com/beryjuorg/authentik/_build?definitionId=1)
|
||||||
|
[![Tests](https://img.shields.io/azure-devops/tests/beryjuorg/authentik/1?compact_message&style=flat-square)](https://dev.azure.com/beryjuorg/authentik/_build?definitionId=1)
|
||||||
|
[![Code Coverage](https://img.shields.io/codecov/c/gh/beryju/authentik?style=flat-square)](https://codecov.io/gh/BeryJu/authentik)
|
||||||
|
![Docker pulls](https://img.shields.io/docker/pulls/beryju/authentik.svg?style=flat-square)
|
||||||
|
![Latest version](https://img.shields.io/docker/v/beryju/authentik?sort=semver&style=flat-square)
|
||||||
|
![LGTM Grade](https://img.shields.io/lgtm/grade/python/github/BeryJu/authentik?style=flat-square)
|
||||||
|
|
||||||
passbook is an open-source Identity Provider focused on flexibility and versatility. You can use passbook in an existing environment to add support for new protocols. passbook is also a great solution for implementing signup/recovery/etc in your application, so you don't have to deal with it.
|
## What is authentik?
|
||||||
|
|
||||||
|
authentik is an open-source Identity Provider focused on flexibility and versatility. You can use authentik in an existing environment to add support for new protocols. authentik is also a great solution for implementing signup/recovery/etc in your application, so you don't have to deal with it.
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
For small/test setups it is recommended to use docker-compose, see the [documentation](https://passbook.beryju.org/docs/installation/docker-compose/)
|
For small/test setups it is recommended to use docker-compose, see the [documentation](https://goauthentik.io/docs/installation/docker-compose/)
|
||||||
|
|
||||||
For bigger setups, there is a Helm Chart in the `helm/` directory. This is documented [here](https://passbook.beryju.org/docs/installation/kubernetes/)
|
For bigger setups, there is a Helm Chart in the `helm/` directory. This is documented [here](https://goauthentik.io/docs/installation/kubernetes/)
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||
|
|
||||||
|
@ -24,7 +26,7 @@ For bigger setups, there is a Helm Chart in the `helm/` directory. This is docum
|
||||||
|
|
||||||
## Development
|
## Development
|
||||||
|
|
||||||
See [Development Documentation](https://passbook.beryju.org/docs/development/local-dev-environment)
|
See [Development Documentation](https://goauthentik.io/docs/development/local-dev-environment)
|
||||||
|
|
||||||
## Security
|
## Security
|
||||||
|
|
||||||
|
|
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
## Supported Versions
|
## Supported Versions
|
||||||
|
|
||||||
As passbook is currently in a pre-stable, only the latest "stable" version is supported. After passbook 1.0, this will change.
|
As authentik is currently in a pre-stable, only the latest "stable" version is supported. After authentik 1.0, this will change.
|
||||||
|
|
||||||
| Version | Supported |
|
| Version | Supported |
|
||||||
| -------- | ------------------ |
|
| -------- | ------------------ |
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
"""authentik"""
|
||||||
|
__version__ = "0.12.11-stable"
|
|
@ -0,0 +1,79 @@
|
||||||
|
"""authentik administration overview"""
|
||||||
|
from django.core.cache import cache
|
||||||
|
from drf_yasg2.utils import swagger_auto_schema
|
||||||
|
from rest_framework.fields import SerializerMethodField
|
||||||
|
from rest_framework.permissions import IsAdminUser
|
||||||
|
from rest_framework.request import Request
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework.serializers import Serializer
|
||||||
|
from rest_framework.viewsets import ViewSet
|
||||||
|
|
||||||
|
from authentik import __version__
|
||||||
|
from authentik.admin.tasks import VERSION_CACHE_KEY, update_latest_version
|
||||||
|
from authentik.core.models import Provider
|
||||||
|
from authentik.policies.models import Policy
|
||||||
|
from authentik.root.celery import CELERY_APP
|
||||||
|
|
||||||
|
|
||||||
|
class AdministrationOverviewSerializer(Serializer):
|
||||||
|
"""Overview View"""
|
||||||
|
|
||||||
|
version = SerializerMethodField()
|
||||||
|
version_latest = SerializerMethodField()
|
||||||
|
worker_count = SerializerMethodField()
|
||||||
|
providers_without_application = SerializerMethodField()
|
||||||
|
policies_without_binding = SerializerMethodField()
|
||||||
|
cached_policies = SerializerMethodField()
|
||||||
|
cached_flows = SerializerMethodField()
|
||||||
|
|
||||||
|
def get_version(self, _) -> str:
|
||||||
|
"""Get current version"""
|
||||||
|
return __version__
|
||||||
|
|
||||||
|
def get_version_latest(self, _) -> str:
|
||||||
|
"""Get latest version from cache"""
|
||||||
|
version_in_cache = cache.get(VERSION_CACHE_KEY)
|
||||||
|
if not version_in_cache:
|
||||||
|
update_latest_version.delay()
|
||||||
|
return __version__
|
||||||
|
return version_in_cache
|
||||||
|
|
||||||
|
def get_worker_count(self, _) -> int:
|
||||||
|
"""Ping workers"""
|
||||||
|
return len(CELERY_APP.control.ping(timeout=0.5))
|
||||||
|
|
||||||
|
def get_providers_without_application(self, _) -> int:
|
||||||
|
"""Count of providers without application"""
|
||||||
|
return len(Provider.objects.filter(application=None))
|
||||||
|
|
||||||
|
def get_policies_without_binding(self, _) -> int:
|
||||||
|
"""Count of policies not bound or use in prompt stages"""
|
||||||
|
return len(
|
||||||
|
Policy.objects.filter(bindings__isnull=True, promptstage__isnull=True)
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_cached_policies(self, _) -> int:
|
||||||
|
"""Get cached policy count"""
|
||||||
|
return len(cache.keys("policy_*"))
|
||||||
|
|
||||||
|
def get_cached_flows(self, _) -> int:
|
||||||
|
"""Get cached flow count"""
|
||||||
|
return len(cache.keys("flow_*"))
|
||||||
|
|
||||||
|
def create(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def update(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
|
class AdministrationOverviewViewSet(ViewSet):
|
||||||
|
"""Return single instance of AdministrationOverviewSerializer"""
|
||||||
|
|
||||||
|
permission_classes = [IsAdminUser]
|
||||||
|
|
||||||
|
@swagger_auto_schema(responses={200: AdministrationOverviewSerializer(many=True)})
|
||||||
|
def list(self, request: Request) -> Response:
|
||||||
|
"""Return single instance of AdministrationOverviewSerializer"""
|
||||||
|
serializer = AdministrationOverviewSerializer(True)
|
||||||
|
return Response(serializer.data)
|
|
@ -0,0 +1,79 @@
|
||||||
|
"""authentik administration overview"""
|
||||||
|
import time
|
||||||
|
from collections import Counter
|
||||||
|
from datetime import timedelta
|
||||||
|
from typing import Dict, List
|
||||||
|
|
||||||
|
from django.db.models import Count, ExpressionWrapper, F
|
||||||
|
from django.db.models.fields import DurationField
|
||||||
|
from django.db.models.functions import ExtractHour
|
||||||
|
from django.http import response
|
||||||
|
from django.utils.timezone import now
|
||||||
|
from drf_yasg2.utils import swagger_auto_schema
|
||||||
|
from rest_framework.fields import SerializerMethodField
|
||||||
|
from rest_framework.permissions import IsAdminUser
|
||||||
|
from rest_framework.request import Request
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework.serializers import Serializer
|
||||||
|
from rest_framework.viewsets import ViewSet
|
||||||
|
|
||||||
|
from authentik.audit.models import Event, EventAction
|
||||||
|
|
||||||
|
|
||||||
|
def get_events_per_1h(**filter_kwargs) -> List[Dict[str, int]]:
|
||||||
|
"""Get event count by hour in the last day, fill with zeros"""
|
||||||
|
date_from = now() - timedelta(days=1)
|
||||||
|
result = (
|
||||||
|
Event.objects.filter(created__gte=date_from, **filter_kwargs)
|
||||||
|
.annotate(
|
||||||
|
age=ExpressionWrapper(now() - F("created"), output_field=DurationField())
|
||||||
|
)
|
||||||
|
.annotate(age_hours=ExtractHour("age"))
|
||||||
|
.values("age_hours")
|
||||||
|
.annotate(count=Count("pk"))
|
||||||
|
.order_by("age_hours")
|
||||||
|
)
|
||||||
|
data = Counter({d["age_hours"]: d["count"] for d in result})
|
||||||
|
results = []
|
||||||
|
_now = now()
|
||||||
|
for hour in range(0, -24, -1):
|
||||||
|
results.append(
|
||||||
|
{
|
||||||
|
"x": time.mktime((_now + timedelta(hours=hour)).timetuple()) * 1000,
|
||||||
|
"y": data[hour * -1],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
class AdministrationMetricsSerializer(Serializer):
|
||||||
|
"""Overview View"""
|
||||||
|
|
||||||
|
logins_per_1h = SerializerMethodField()
|
||||||
|
logins_failed_per_1h = SerializerMethodField()
|
||||||
|
|
||||||
|
def get_logins_per_1h(self, _):
|
||||||
|
"""Get successful logins per hour for the last 24 hours"""
|
||||||
|
return get_events_per_1h(action=EventAction.LOGIN)
|
||||||
|
|
||||||
|
def get_logins_failed_per_1h(self, _):
|
||||||
|
"""Get failed logins per hour for the last 24 hours"""
|
||||||
|
return get_events_per_1h(action=EventAction.LOGIN_FAILED)
|
||||||
|
|
||||||
|
def create(self, request: Request) -> response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def update(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
|
class AdministrationMetricsViewSet(ViewSet):
|
||||||
|
"""Return single instance of AdministrationMetricsSerializer"""
|
||||||
|
|
||||||
|
permission_classes = [IsAdminUser]
|
||||||
|
|
||||||
|
@swagger_auto_schema(responses={200: AdministrationMetricsSerializer(many=True)})
|
||||||
|
def list(self, request: Request) -> Response:
|
||||||
|
"""Return single instance of AdministrationMetricsSerializer"""
|
||||||
|
serializer = AdministrationMetricsSerializer(True)
|
||||||
|
return Response(serializer.data)
|
|
@ -0,0 +1,72 @@
|
||||||
|
"""Tasks API"""
|
||||||
|
from importlib import import_module
|
||||||
|
|
||||||
|
from django.contrib import messages
|
||||||
|
from django.http.response import Http404
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from drf_yasg2.utils import swagger_auto_schema
|
||||||
|
from rest_framework.decorators import action
|
||||||
|
from rest_framework.fields import CharField, DateTimeField, IntegerField, ListField
|
||||||
|
from rest_framework.permissions import IsAdminUser
|
||||||
|
from rest_framework.request import Request
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework.serializers import Serializer
|
||||||
|
from rest_framework.viewsets import ViewSet
|
||||||
|
|
||||||
|
from authentik.lib.tasks import TaskInfo
|
||||||
|
|
||||||
|
|
||||||
|
class TaskSerializer(Serializer):
|
||||||
|
"""Serialize TaskInfo and TaskResult"""
|
||||||
|
|
||||||
|
task_name = CharField()
|
||||||
|
task_description = CharField()
|
||||||
|
task_finish_timestamp = DateTimeField(source="finish_timestamp")
|
||||||
|
|
||||||
|
status = IntegerField(source="result.status.value")
|
||||||
|
messages = ListField(source="result.messages")
|
||||||
|
|
||||||
|
def create(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def update(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
|
class TaskViewSet(ViewSet):
|
||||||
|
"""Read-only view set that returns all background tasks"""
|
||||||
|
|
||||||
|
permission_classes = [IsAdminUser]
|
||||||
|
|
||||||
|
@swagger_auto_schema(responses={200: TaskSerializer(many=True)})
|
||||||
|
def list(self, request: Request) -> Response:
|
||||||
|
"""List current messages and pass into Serializer"""
|
||||||
|
return Response(TaskSerializer(TaskInfo.all().values(), many=True).data)
|
||||||
|
|
||||||
|
@action(detail=True, methods=["post"])
|
||||||
|
# pylint: disable=invalid-name
|
||||||
|
def retry(self, request: Request, pk=None) -> Response:
|
||||||
|
"""Retry task"""
|
||||||
|
task = TaskInfo.by_name(pk)
|
||||||
|
if not task:
|
||||||
|
raise Http404
|
||||||
|
try:
|
||||||
|
task_module = import_module(task.task_call_module)
|
||||||
|
task_func = getattr(task_module, task.task_call_func)
|
||||||
|
task_func.delay(*task.task_call_args, **task.task_call_kwargs)
|
||||||
|
messages.success(
|
||||||
|
self.request,
|
||||||
|
_(
|
||||||
|
"Successfully re-scheduled Task %(name)s!"
|
||||||
|
% {"name": task.task_name}
|
||||||
|
),
|
||||||
|
)
|
||||||
|
return Response(
|
||||||
|
{
|
||||||
|
"successful": True,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
except ImportError:
|
||||||
|
# if we get an import error, the module path has probably changed
|
||||||
|
task.delete()
|
||||||
|
return Response({"successful": False})
|
|
@ -0,0 +1,11 @@
|
||||||
|
"""authentik admin app config"""
|
||||||
|
from django.apps import AppConfig
|
||||||
|
|
||||||
|
|
||||||
|
class AuthentikAdminConfig(AppConfig):
|
||||||
|
"""authentik admin app config"""
|
||||||
|
|
||||||
|
name = "authentik.admin"
|
||||||
|
label = "authentik_admin"
|
||||||
|
mountpoint = "administration/"
|
||||||
|
verbose_name = "authentik Admin"
|
|
@ -0,0 +1,12 @@
|
||||||
|
"""authentik administration forms"""
|
||||||
|
from django import forms
|
||||||
|
|
||||||
|
from authentik.admin.fields import CodeMirrorWidget, YAMLField
|
||||||
|
from authentik.core.models import User
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyTestForm(forms.Form):
|
||||||
|
"""Form to test policies against user"""
|
||||||
|
|
||||||
|
user = forms.ModelChoiceField(queryset=User.objects.all())
|
||||||
|
context = YAMLField(widget=CodeMirrorWidget(), required=False, initial=dict)
|
|
@ -0,0 +1,17 @@
|
||||||
|
"""authentik core source form fields"""
|
||||||
|
|
||||||
|
SOURCE_FORM_FIELDS = [
|
||||||
|
"name",
|
||||||
|
"slug",
|
||||||
|
"enabled",
|
||||||
|
"authentication_flow",
|
||||||
|
"enrollment_flow",
|
||||||
|
]
|
||||||
|
SOURCE_SERIALIZER_FIELDS = [
|
||||||
|
"pk",
|
||||||
|
"name",
|
||||||
|
"slug",
|
||||||
|
"enabled",
|
||||||
|
"authentication_flow",
|
||||||
|
"enrollment_flow",
|
||||||
|
]
|
|
@ -0,0 +1,22 @@
|
||||||
|
"""authentik administrative user forms"""
|
||||||
|
|
||||||
|
from django import forms
|
||||||
|
|
||||||
|
from authentik.admin.fields import CodeMirrorWidget, YAMLField
|
||||||
|
from authentik.core.models import User
|
||||||
|
|
||||||
|
|
||||||
|
class UserForm(forms.ModelForm):
|
||||||
|
"""Update User Details"""
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
|
||||||
|
model = User
|
||||||
|
fields = ["username", "name", "email", "is_active", "attributes"]
|
||||||
|
widgets = {
|
||||||
|
"name": forms.TextInput,
|
||||||
|
"attributes": CodeMirrorWidget,
|
||||||
|
}
|
||||||
|
field_classes = {
|
||||||
|
"attributes": YAMLField,
|
||||||
|
}
|
|
@ -0,0 +1,9 @@
|
||||||
|
"""authentik admin mixins"""
|
||||||
|
from django.contrib.auth.mixins import UserPassesTestMixin
|
||||||
|
|
||||||
|
|
||||||
|
class AdminRequiredMixin(UserPassesTestMixin):
|
||||||
|
"""Make sure user is administrator"""
|
||||||
|
|
||||||
|
def test_func(self):
|
||||||
|
return self.request.user.is_superuser
|
|
@ -0,0 +1,10 @@
|
||||||
|
"""authentik admin settings"""
|
||||||
|
from celery.schedules import crontab
|
||||||
|
|
||||||
|
CELERY_BEAT_SCHEDULE = {
|
||||||
|
"admin_latest_version": {
|
||||||
|
"task": "authentik.admin.tasks.update_latest_version",
|
||||||
|
"schedule": crontab(minute=0), # Run every hour
|
||||||
|
"options": {"queue": "authentik_scheduled"},
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
"""authentik admin tasks"""
|
||||||
|
from django.core.cache import cache
|
||||||
|
from requests import RequestException, get
|
||||||
|
from structlog import get_logger
|
||||||
|
|
||||||
|
from authentik.lib.tasks import MonitoredTask, TaskResult, TaskResultStatus
|
||||||
|
from authentik.root.celery import CELERY_APP
|
||||||
|
|
||||||
|
LOGGER = get_logger()
|
||||||
|
VERSION_CACHE_KEY = "authentik_latest_version"
|
||||||
|
VERSION_CACHE_TIMEOUT = 2 * 60 * 60 # 2 hours
|
||||||
|
|
||||||
|
|
||||||
|
@CELERY_APP.task(bind=True, base=MonitoredTask)
|
||||||
|
def update_latest_version(self: MonitoredTask):
|
||||||
|
"""Update latest version info"""
|
||||||
|
try:
|
||||||
|
response = get("https://api.github.com/repos/beryju/authentik/releases/latest")
|
||||||
|
response.raise_for_status()
|
||||||
|
data = response.json()
|
||||||
|
tag_name = data.get("tag_name")
|
||||||
|
cache.set(VERSION_CACHE_KEY, tag_name.split("/")[1], VERSION_CACHE_TIMEOUT)
|
||||||
|
self.set_status(
|
||||||
|
TaskResult(
|
||||||
|
TaskResultStatus.SUCCESSFUL, ["Successfully updated latest Version"]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
except (RequestException, IndexError) as exc:
|
||||||
|
cache.set(VERSION_CACHE_KEY, "0.0.0", VERSION_CACHE_TIMEOUT)
|
||||||
|
self.set_status(TaskResult(TaskResultStatus.ERROR).with_error(exc))
|
|
@ -0,0 +1,121 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-applications"></i>
|
||||||
|
{% trans 'Applications' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "External Applications which use authentik as Identity-Provider, utilizing protocols like OAuth2 and SAML." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:application-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Slug' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Provider' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Provider Type' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for application in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<a href="/applications/{{ application.slug }}/">
|
||||||
|
<div>{{ application.name }}</div>
|
||||||
|
{% if application.meta_publisher %}
|
||||||
|
<small>{{ application.meta_publisher }}</small>
|
||||||
|
{% endif %}
|
||||||
|
</a>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<code>{{ application.slug }}</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ application.get_provider }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ application.get_provider|verbose_name }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:application-update' pk=application.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:application-delete' pk=application.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-applications pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Applications.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any application." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no applications exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:application-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,116 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-key"></i>
|
||||||
|
{% trans 'Certificate-Key Pairs' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Import certificates of external providers or create certificates to sign requests with." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:certificatekeypair-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Private Key available' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Fingerprint' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for kp in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ kp.name }}</div>
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{% if kp.key_data is not None %}
|
||||||
|
{% trans 'Yes' %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'No' %}
|
||||||
|
{% endif %}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<code>{{ kp.fingerprint }}</code>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:certificatekeypair-update' pk=kp.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:certificatekeypair-delete' pk=kp.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-key pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Certificates.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any certificates." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no certificates exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:certificatekeypair-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,135 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-process-automation"></i>
|
||||||
|
{% trans 'Flows' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Flows describe a chain of Stages to authenticate, enroll or recover a user. Stages are chosen based on policies applied to them." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:flow-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:flow-import' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Import' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Identifier' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Designation' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Stages' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Policies' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for flow in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div><code>{{ flow.slug }}</code></div>
|
||||||
|
<small>{{ flow.name }}</small>
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ flow.designation }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ flow.stages.all|length }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ flow.policies.all|length }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:flow-update' pk=flow.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:flow-delete' pk=flow.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<a class="pf-c-button pf-m-secondary ak-root-link" href="{% url 'authentik_admin:flow-execute' pk=flow.pk %}?next={{ request.get_full_path }}">{% trans 'Execute' %}</a>
|
||||||
|
<a class="pf-c-button pf-m-secondary ak-root-link" href="{% url 'authentik_admin:flow-export' pk=flow.pk %}?next={{ request.get_full_path }}">{% trans 'Export' %}</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-process-automation pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Flows.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any flows." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no flows exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:flow-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:flow-import' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Import' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,114 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-users"></i>
|
||||||
|
{% trans 'Groups' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Group users together and give them permissions based on the membership." %}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:group-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Parent' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Members' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for group in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ group.name }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ group.parent }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ group.users.all|length }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:group-update' pk=group.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:group-delete' pk=group.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-users pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Groups.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any groups." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no group exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:group-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,149 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load humanize %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load admin_reflection %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-zone"></i>
|
||||||
|
{% trans 'Outposts' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Outposts are deployments of authentik components to support different environments and protocols, like reverse proxies." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Providers' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Health' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Version' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for outpost in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<span>{{ outpost.name }}</span>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ outpost.providers.all.select_subclasses|join:", " }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
{% with states=outpost.state %}
|
||||||
|
{% if states|length > 0 %}
|
||||||
|
<td role="cell">
|
||||||
|
{% for state in states %}
|
||||||
|
<div>
|
||||||
|
{% if state.last_seen %}
|
||||||
|
<i class="fas fa-check pf-m-success"></i> {{ state.last_seen|naturaltime }}
|
||||||
|
{% else %}
|
||||||
|
<i class="fas fa-times pf-m-danger"></i> {% trans 'Unhealthy' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
{% for state in states %}
|
||||||
|
<div>
|
||||||
|
{% if not state.version %}
|
||||||
|
<i class="fas fa-question-circle"></i>
|
||||||
|
{% elif state.version_outdated %}
|
||||||
|
<i class="fas fa-times pf-m-danger"></i> {% blocktrans with is=state.version should=state.version_should %}{{ is }}, should be {{ should }}{% endblocktrans %}
|
||||||
|
{% else %}
|
||||||
|
<i class="fas fa-check pf-m-success"></i> {{ state.version }}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
{% else %}
|
||||||
|
<td role="cell">
|
||||||
|
<i class="fas fa-question-circle"></i>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<i class="fas fa-question-circle"></i>
|
||||||
|
</td>
|
||||||
|
{% endif %}
|
||||||
|
{% endwith %}
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-update' pk=outpost.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-delete' pk=outpost.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% get_htmls outpost as htmls %}
|
||||||
|
{% for html in htmls %}
|
||||||
|
{{ html|safe }}
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="fas fa-map-marker pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Outposts.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any outposts." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no outposts exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,154 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load humanize %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load admin_reflection %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon-integration"></i>
|
||||||
|
{% trans 'Outpost Service-Connections' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Outpost Service-Connections define how authentik connects to external platforms to manage and deploy Outposts." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-service-connection-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Type' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Local?' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Status' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for sc in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<span>{{ sc.name }}</span>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ sc|verbose_name }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ sc.local|yesno:"Yes,No" }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{% if sc.state.healthy %}
|
||||||
|
<i class="fas fa-check pf-m-success"></i> {{ sc.state.version }}
|
||||||
|
{% else %}
|
||||||
|
<i class="fas fa-times pf-m-danger"></i> {% trans 'Unhealthy' %}
|
||||||
|
{% endif %}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-service-connection-update' pk=sc.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-service-connection-delete' pk=sc.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="fas fa-map-marker pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Outpost Service Connections.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any outposts." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no service connections exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:outpost-service-connection-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,230 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load static %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>{% trans 'System Overview' %}</h1>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section">
|
||||||
|
<div class="pf-l-gallery pf-m-gutter">
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-4-col" style="grid-column-end: span 3;grid-row-end: span 2;">
|
||||||
|
<div class="pf-c-card__header">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-server"></i> {% trans 'Logins over the last 24 hours' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
<ak-admin-logins-chart url="{% url 'authentik_api:admin_metrics-list' %}"></ak-admin-logins-chart>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-4-col" style="grid-column-end: span 2;grid-row-end: span 3;">
|
||||||
|
<div class="pf-c-card__header">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-server"></i> {% trans 'Apps with most usage' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
<table class="pf-c-table pf-m-compact" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Application' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Logins' %}</th>
|
||||||
|
<th role="columnheader" scope="col"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for app in most_used_applications %}
|
||||||
|
<tr role="row">
|
||||||
|
<td role="cell">
|
||||||
|
{{ app.application.name }}
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
{{ app.total_logins }}
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<progress value="{{ app.total_logins }}" max="{{ most_used_applications.0.total_logins }}"></progress>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-compact">
|
||||||
|
<div class="pf-c-card__header pf-l-flex pf-m-justify-content-space-between">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-plugged"></i> {% trans 'Providers' %}
|
||||||
|
</div>
|
||||||
|
<a href="{% url 'authentik_admin:providers' %}">
|
||||||
|
<i class="fa fa-external-link-alt"> </i>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
{% if providers_without_application.exists %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-exclamation-triangle"></i> {{ provider_count }}
|
||||||
|
</p>
|
||||||
|
<p>{% trans 'Warning: At least one Provider has no application assigned.' %}</p>
|
||||||
|
{% else %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-check-circle"></i> {{ provider_count }}
|
||||||
|
</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-compact">
|
||||||
|
<div class="pf-c-card__header pf-l-flex pf-m-justify-content-space-between">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-infrastructure"></i> {% trans 'Policies' %}
|
||||||
|
</div>
|
||||||
|
<a href="{% url 'authentik_admin:policies' %}">
|
||||||
|
<i class="fa fa-external-link-alt"> </i>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
{% if policies_without_binding %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-exclamation-triangle"></i> {{ policy_count }}
|
||||||
|
</p>
|
||||||
|
<p>{% trans 'Policies without binding exist.' %}</p>
|
||||||
|
{% else %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-check-circle"></i> {{ policy_count }}
|
||||||
|
</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-compact">
|
||||||
|
<div class="pf-c-card__header pf-l-flex pf-m-justify-content-space-between">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-user"></i> {% trans 'Users' %}
|
||||||
|
</div>
|
||||||
|
<a href="{% url 'authentik_admin:users' %}">
|
||||||
|
<i class="fa fa-external-link-alt"> </i>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-check-circle"></i> {{ user_count }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-compact">
|
||||||
|
<div class="pf-c-card__header pf-l-flex pf-m-justify-content-space-between">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-bundle"></i> {% trans 'Version' %}
|
||||||
|
</div>
|
||||||
|
<a href="https://github.com/BeryJu/authentik/releases" target="_blank">
|
||||||
|
<i class="fa fa-external-link-alt"> </i>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
{% if version >= version_latest %}
|
||||||
|
<i class="fa fa-check-circle"></i> {{ version }}
|
||||||
|
{% else %}
|
||||||
|
<i class="fa fa-exclamation-triangle"></i> {{ version }}
|
||||||
|
{% endif %}
|
||||||
|
</p>
|
||||||
|
{% if version >= version_latest %}
|
||||||
|
{% blocktrans %}
|
||||||
|
Up-to-date!
|
||||||
|
{% endblocktrans %}
|
||||||
|
{% else %}
|
||||||
|
{% blocktrans with latest=version_latest %}
|
||||||
|
{{ latest }} is available!
|
||||||
|
{% endblocktrans %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-compact">
|
||||||
|
<div class="pf-c-card__header">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-server"></i> {% trans 'Workers' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<fetch-fill-slot class="pf-c-card__body" url="{% url 'authentik_api:admin_overview-list' %}" key="worker_count">
|
||||||
|
<div slot="value < 1">
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-exclamation-triangle"></i> <span data-value></span>
|
||||||
|
</p>
|
||||||
|
<p>{% trans 'No workers connected.' %}</p>
|
||||||
|
</div>
|
||||||
|
<div slot="value >= 1">
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-check-circle"></i> <span data-value></span>
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<span class="pf-c-spinner" role="progressbar" aria-valuetext="Loading...">
|
||||||
|
<span class="pf-c-spinner__clipper"></span>
|
||||||
|
<span class="pf-c-spinner__lead-ball"></span>
|
||||||
|
<span class="pf-c-spinner__tail-ball"></span>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</fetch-fill-slot>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-compact">
|
||||||
|
<div class="pf-c-card__header pf-l-flex pf-m-justify-content-space-between">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-server"></i> {% trans 'Cached Policies' %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:overview-clear-policy-cache' %}">
|
||||||
|
<a slot="trigger">
|
||||||
|
<i class="fa fa-trash"> </i>
|
||||||
|
</a>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
{% if cached_policies < 1 %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-exclamation-triangle"></i> {{ cached_policies }}
|
||||||
|
</p>
|
||||||
|
<p>{% trans 'No policies cached. Users may experience slow response times.' %}</p>
|
||||||
|
{% else %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-check-circle"></i> {{ cached_policies }}
|
||||||
|
</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="pf-c-card pf-c-card-aggregate pf-l-gallery__item pf-m-compact">
|
||||||
|
<div class="pf-c-card__header pf-l-flex pf-m-justify-content-space-between">
|
||||||
|
<div class="pf-c-card__header-main">
|
||||||
|
<i class="pf-icon pf-icon-server"></i> {% trans 'Cached Flows' %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:overview-clear-flow-cache' %}">
|
||||||
|
<a slot="trigger">
|
||||||
|
<i class="fa fa-trash"> </i>
|
||||||
|
</a>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
{% if cached_flows < 1 %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<span class="fa fa-exclamation-triangle"></span> {{ cached_flows }}
|
||||||
|
</p>
|
||||||
|
<p>{% trans 'No flows cached.' %}</p>
|
||||||
|
{% else %}
|
||||||
|
<p class="ak-aggregate-card">
|
||||||
|
<i class="fa fa-check-circle"></i> {{ cached_flows }}
|
||||||
|
</p>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,148 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-infrastructure"></i>
|
||||||
|
{% trans 'Policies' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Allow users to use Applications based on properties, enforce Password Criteria and selectively apply Stages." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Type' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for policy in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ policy.name }}</div>
|
||||||
|
{% if not policy.bindings.exists and not policy.promptstage_set.exists %}
|
||||||
|
<i class="pf-icon pf-icon-warning-triangle"></i>
|
||||||
|
<small>{% trans 'Warning: Policy is not assigned.' %}</small>
|
||||||
|
{% else %}
|
||||||
|
<i class="pf-icon pf-icon-ok"></i>
|
||||||
|
<small>{% blocktrans with object_count=policy.bindings.all|length %}Assigned to {{ object_count }} objects.{% endblocktrans %}</small>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ policy|verbose_name }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-update' pk=policy.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-test' pk=policy.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-tertiary">
|
||||||
|
{% trans 'Test' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-delete' pk=policy.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-infrastructure pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Policies.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any policies." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no policies exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,119 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-infrastructure"></i>
|
||||||
|
{% trans 'Policy Bindings' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Bind existing Policies to Models accepting policies." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-binding-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Policy' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Enabled' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Order' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Timeout' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for pbm in object_list %}
|
||||||
|
<tr role="role">
|
||||||
|
<td>
|
||||||
|
{{ pbm }}
|
||||||
|
<small>
|
||||||
|
{{ pbm|fieldtype }}
|
||||||
|
</small>
|
||||||
|
</td>
|
||||||
|
<td></td>
|
||||||
|
<td></td>
|
||||||
|
<td></td>
|
||||||
|
<td></td>
|
||||||
|
</tr>
|
||||||
|
{% for binding in pbm.bindings %}
|
||||||
|
<tr class="row pf-c-table__expandable-row pf-m-expanded">
|
||||||
|
<th role="cell">
|
||||||
|
<div>{{ binding.policy }}</div>
|
||||||
|
<small>
|
||||||
|
{{ binding.policy|fieldtype }}
|
||||||
|
</small>
|
||||||
|
</th>
|
||||||
|
<th role="cell">
|
||||||
|
<div>{{ binding.enabled }}</div>
|
||||||
|
</th>
|
||||||
|
<th role="cell">
|
||||||
|
<div>{{ binding.order }}</div>
|
||||||
|
</th>
|
||||||
|
<th role="cell">
|
||||||
|
<div>{{ binding.timeout }}</div>
|
||||||
|
</th>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-binding-update' pk=binding.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-binding-delete' pk=binding.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="fas fa-cubes pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Policy Bindings.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% trans 'Currently no policy bindings exist. Click the button below to create one.' %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:policy-binding-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,139 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-blueprint"></i>
|
||||||
|
{% trans 'Property Mappings' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Control how authentik exposes and interprets information." %}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:property-mapping-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Type' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for property_mapping in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ property_mapping.name }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ property_mapping|verbose_name }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:property-mapping-update' pk=property_mapping.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:property-mapping-delete' pk=property_mapping.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-blueprint pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Property Mappings.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any property mappings." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no property mappings exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:property-mapping-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,159 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load admin_reflection %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-integration"></i>
|
||||||
|
{% trans 'Providers' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Provide support for protocols like SAML and OAuth to assigned applications." %}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:provider-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Type' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for provider in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ provider.name }}</div>
|
||||||
|
{% if not provider.application %}
|
||||||
|
<i class="pf-icon pf-icon-warning-triangle"></i>
|
||||||
|
<small>{% trans 'Warning: Provider not assigned to any application.' %}</small>
|
||||||
|
{% else %}
|
||||||
|
<i class="pf-icon pf-icon-ok"></i>
|
||||||
|
<small>
|
||||||
|
{% blocktrans with app=provider.application %}
|
||||||
|
Assigned to application {{ app }}.
|
||||||
|
{% endblocktrans %}
|
||||||
|
</small>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ provider|verbose_name }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:provider-update' pk=provider.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:provider-delete' pk=provider.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% get_links provider as links %}
|
||||||
|
{% for name, href in links.items %}
|
||||||
|
<a class="pf-c-button pf-m-tertiary ak-root-link" href="{{ href }}?back={{ request.get_full_path }}">{% trans name %}</a>
|
||||||
|
{% endfor %}
|
||||||
|
{% get_htmls provider as htmls %}
|
||||||
|
{% for html in htmls %}
|
||||||
|
{{ html|safe }}
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon-integration pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Providers.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any providers." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no providers exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:provider-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,153 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load admin_reflection %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-middleware"></i>
|
||||||
|
{% trans 'Source' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "External Sources which can be used to get Identities into authentik, for example Social Providers like Twiter and GitHub or Enterprise Providers like ADFS and LDAP." %}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:source-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Type' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Additional Info' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for source in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ source.name }}</div>
|
||||||
|
{% if not source.enabled %}
|
||||||
|
<small>{% trans 'Disabled' %}</small>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ source|fieldtype }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ source.ui_additional_info|default:""|safe }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:source-update' pk=source.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:source-delete' pk=source.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% get_links source as links %}
|
||||||
|
{% for name, href in links %}
|
||||||
|
<a class="pf-c-button pf-m-tertiary ak-root-link" href="{{ href }}?back={{ request.get_full_path }}">{% trans name %}</a>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-middleware pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Sources.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any sources." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no sources exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:source-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,148 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load admin_reflection %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-plugged"></i>
|
||||||
|
{% trans 'Stages' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Stages are single steps of a Flow that a user is guided through." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Flows' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for stage in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ stage.name }}</div>
|
||||||
|
<small>{{ stage|verbose_name }}</small>
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<ul>
|
||||||
|
{% for flow in stage.flow_set.all %}
|
||||||
|
<li>{{ flow.slug }}<</li>
|
||||||
|
{% empty %}
|
||||||
|
<li>-</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-update' pk=stage.stage_uuid %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-delete' pk=stage.stage_uuid %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% get_links stage as links %}
|
||||||
|
{% for name, href in links.items %}
|
||||||
|
<a class="pf-c-button pf-m-tertiary ak-root-link" href="{{ href }}?back={{ request.get_full_path }}">{% trans name %}</a>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-plugged pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Stages.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any stages." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no stages exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-dropdown class="pf-c-dropdown">
|
||||||
|
<button class="pf-m-primary pf-c-dropdown__toggle" type="button">
|
||||||
|
<span class="pf-c-dropdown__toggle-text">{% trans 'Create' %}</span>
|
||||||
|
<i class="fas fa-caret-down pf-c-dropdown__toggle-icon" aria-hidden="true"></i>
|
||||||
|
</button>
|
||||||
|
<ul class="pf-c-dropdown__menu" hidden>
|
||||||
|
{% for type, name in types.items %}
|
||||||
|
<li>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-create' %}?type={{ type }}">
|
||||||
|
<button slot="trigger" class="pf-c-dropdown__menu-item">
|
||||||
|
{{ name|verbose_name }}<br>
|
||||||
|
<small>
|
||||||
|
{{ name|doc }}
|
||||||
|
</small>
|
||||||
|
</button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</ak-dropdown>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,125 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-infrastructure"></i>
|
||||||
|
{% trans 'Stage Bindings' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Bind existing Stages to Flows." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-binding-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Order' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Stage Type' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% regroup object_list by target as grouped_bindings %}
|
||||||
|
{% for flow in grouped_bindings %}
|
||||||
|
<tr role="role">
|
||||||
|
<td>
|
||||||
|
{% blocktrans with slug=flow.grouper.slug %}
|
||||||
|
Flow {{ slug }}
|
||||||
|
{% endblocktrans %}
|
||||||
|
</td>
|
||||||
|
<td></td>
|
||||||
|
<td></td>
|
||||||
|
<td></td>
|
||||||
|
</tr>
|
||||||
|
{% for binding in flow.list %}
|
||||||
|
<tr class="pf-c-table__expandable-row pf-m-expanded" role="row">
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ binding.order }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ binding.target.slug }}</div>
|
||||||
|
<small>
|
||||||
|
{{ binding.target.name }}
|
||||||
|
</small>
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<div>
|
||||||
|
<div>
|
||||||
|
{{ binding.stage.name }}
|
||||||
|
</div>
|
||||||
|
<small>
|
||||||
|
{{ binding.stage }}
|
||||||
|
</small>
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-binding-update' pk=binding.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Update' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-binding-delete' pk=binding.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="fas fa-cubes pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Flow-Stage Bindings.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% trans 'Currently no flow-stage bindings exist. Click the button below to create one.' %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-binding-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,103 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-migration"></i>
|
||||||
|
{% trans 'Invitations' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Create Invitation Links to enroll Users, and optionally force specific attributes of their account." %}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-invitation-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Expiry' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Link' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for invitation in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ invitation.expiry }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ invitation.Link }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-invitation-delete' pk=invitation.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-migration pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Invitations.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any invitations." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no invitations exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-invitation-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,130 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load admin_reflection %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-plugged"></i>
|
||||||
|
{% trans 'Prompts' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Single Prompts that can be used for Prompt Stages." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-prompt-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Field' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Label' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Type' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Order' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Flows' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for prompt in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ prompt.field_key }}</div>
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<div>
|
||||||
|
{{ prompt.label }}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<div>
|
||||||
|
{{ prompt.type }}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<div>
|
||||||
|
{{ prompt.order }}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<ul>
|
||||||
|
{% for flow in prompt.flow_set.all %}
|
||||||
|
<li>{{ flow.slug }}</li>
|
||||||
|
{% empty %}
|
||||||
|
<li>-</li>
|
||||||
|
{% endfor %}
|
||||||
|
</ul>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-prompt-update' pk=prompt.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Update' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:stage-prompt-delete' pk=prompt.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% get_links prompt as links %}
|
||||||
|
{% for name, href in links.items %}
|
||||||
|
<a class="pf-c-button pf-m-tertiary ak-root-link" href="{{ href }}?back={{ request.get_full_path }}">{% trans name %}</a>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-plugged pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Stage Prompts.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any stage prompts." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no stage prompts exist. Click the button below to create one.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<a href="{% url 'authentik_admin:stage-prompt-create' %}?back={{ request.get_full_path }}" class="pf-c-button pf-m-primary" type="button">{% trans 'Create' %}</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,84 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load humanize %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-automation"></i>
|
||||||
|
{% trans 'System Tasks' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Long-running operations which authentik executes in the background." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Identifier' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Description' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Last Run' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Status' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Messages' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for task in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<pre>{{ task.task_name }}</pre>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ task.task_description }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ task.finish_timestamp|naturaltime }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{% if task.result.status == task_successful %}
|
||||||
|
<i class="fas fa-check pf-m-success"></i> {% trans 'Successful' %}
|
||||||
|
{% elif task.result.status == task_warning %}
|
||||||
|
<i class="fas fa-exclamation-triangle pf-m-warning"></i> {% trans 'Warning' %}
|
||||||
|
{% elif task.result.status == task_error %}
|
||||||
|
<i class="fas fa-times pf-m-danger"></i> {% trans 'Error' %}
|
||||||
|
{% else %}
|
||||||
|
<i class="fas fa-question-circle"></i> {% trans 'Unknown' %}
|
||||||
|
{% endif %}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{% for message in task.result.messages %}
|
||||||
|
<div>
|
||||||
|
{{ message }}
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-action-button url="{% url 'authentik_api:admin_system_tasks-retry' pk=task.task_name %}">
|
||||||
|
{% trans 'Retry Task' %}
|
||||||
|
</ak-action-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,102 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-security"></i>
|
||||||
|
{% trans 'Tokens' %}
|
||||||
|
</h1>
|
||||||
|
<p>{% trans "Tokens are used throughout authentik for Email validation stages, Recovery keys and API access." %}</p>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Identifier' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'User' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Expires?' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Expiry Date' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for token in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>{{ token.identifier }}</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ token.user }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ token.expiring|yesno:"Yes,No" }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{% if not token.expiring %}
|
||||||
|
-
|
||||||
|
{% else %}
|
||||||
|
{{ token.expires }}
|
||||||
|
{% endif %}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:token-delete' pk=token.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-danger">
|
||||||
|
{% trans 'Delete' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<ak-token-copy-button identifier="{{ token.identifier }}">
|
||||||
|
{% trans 'Copy token' %}
|
||||||
|
</ak-token-copy-button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="fas fa-key pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Tokens.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any token." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no tokens exist.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,42 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
{% block above_form %}
|
||||||
|
<h1>
|
||||||
|
{% blocktrans with object_type=object|verbose_name %}
|
||||||
|
Disable {{ object_type }}
|
||||||
|
{% endblocktrans %}
|
||||||
|
</h1>
|
||||||
|
{% endblock %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section">
|
||||||
|
<div class="pf-l-stack">
|
||||||
|
<div class="pf-l-stack__item">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
<form action="" method="post" class="pf-c-form">
|
||||||
|
{% csrf_token %}
|
||||||
|
<p>
|
||||||
|
{% blocktrans with object_type=object|verbose_name name=object %}
|
||||||
|
Are you sure you want to disable {{ object_type }} "{{ object }}"?
|
||||||
|
{% endblocktrans %}
|
||||||
|
</p>
|
||||||
|
<div class="pf-c-form__group pf-m-action">
|
||||||
|
<div class="pf-c-form__actions">
|
||||||
|
<input class="pf-c-button pf-m-danger" type="submit" value="{% trans 'Disable' %}" />
|
||||||
|
<a class="pf-c-button pf-m-secondary" href="{% back %}">{% trans "Back" %}</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,125 @@
|
||||||
|
{% extends "administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
<h1>
|
||||||
|
<i class="pf-icon pf-icon-user"></i>
|
||||||
|
{% trans 'Users' %}
|
||||||
|
</h1>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section pf-m-no-padding-mobile">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
{% if object_list %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
<div class="pf-c-toolbar__bulk-select">
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:user-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
<button role="ak-refresh" class="pf-c-button pf-m-primary">
|
||||||
|
{% trans 'Refresh' %}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table class="pf-c-table pf-m-compact pf-m-grid-xl" role="grid">
|
||||||
|
<thead>
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Name' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Active' %}</th>
|
||||||
|
<th role="columnheader" scope="col">{% trans 'Last Login' %}</th>
|
||||||
|
<th role="cell"></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody role="rowgroup">
|
||||||
|
{% for user in object_list %}
|
||||||
|
<tr role="row">
|
||||||
|
<th role="columnheader">
|
||||||
|
<div>
|
||||||
|
<div>{{ user.username }}</div>
|
||||||
|
<small>{{ user.name }}</small>
|
||||||
|
</div>
|
||||||
|
</th>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ user.is_active }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td role="cell">
|
||||||
|
<span>
|
||||||
|
{{ user.last_login }}
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:user-update' pk=user.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-secondary">
|
||||||
|
{% trans 'Edit' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% if user.is_active %}
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:user-disable' pk=user.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-warning">
|
||||||
|
{% trans 'Disable' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% else %}
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:user-delete' pk=user.pk %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Enable' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
{% endif %}
|
||||||
|
<a class="pf-c-button pf-m-tertiary ak-root-link" href="{% url 'authentik_admin:user-password-reset' pk=user.pk %}?back={{ request.get_full_path }}">{% trans 'Reset Password' %}</a>
|
||||||
|
<a class="pf-c-button pf-m-tertiary ak-root-link" href="{% url 'authentik_core:impersonate-init' user_id=user.pk %}">{% trans 'Impersonate' %}</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div class="pf-c-pagination pf-m-bottom">
|
||||||
|
{% include 'partials/pagination.html' %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="pf-c-toolbar">
|
||||||
|
<div class="pf-c-toolbar__content">
|
||||||
|
{% include 'partials/toolbar_search.html' %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="pf-c-empty-state">
|
||||||
|
<div class="pf-c-empty-state__content">
|
||||||
|
<i class="pf-icon pf-icon-user pf-c-empty-state__icon" aria-hidden="true"></i>
|
||||||
|
<h1 class="pf-c-title pf-m-lg">
|
||||||
|
{% trans 'No Users.' %}
|
||||||
|
</h1>
|
||||||
|
<div class="pf-c-empty-state__body">
|
||||||
|
{% if request.GET.search != "" %}
|
||||||
|
{% trans "Your search query doesn't match any users." %}
|
||||||
|
{% else %}
|
||||||
|
{% trans 'Currently no users exist. How did you even get here.' %}
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<ak-modal-button href="{% url 'authentik_admin:user-create' %}">
|
||||||
|
<ak-spinner-button slot="trigger" class="pf-m-primary">
|
||||||
|
{% trans 'Create' %}
|
||||||
|
</ak-spinner-button>
|
||||||
|
<div slot="modal"></div>
|
||||||
|
</ak-modal-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1 @@
|
||||||
|
<ak-codemirror mode="{{ widget.attrs.mode }}"><textarea class="pf-c-form-control" name="{{ widget.name }}">{% if widget.value %}{{ widget.value }}{% endif %}</textarea></ak-codemirror>
|
|
@ -0,0 +1,18 @@
|
||||||
|
{% extends base_template|default:"generic/form.html" %}
|
||||||
|
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load i18n %}
|
||||||
|
|
||||||
|
{% block above_form %}
|
||||||
|
<h1>
|
||||||
|
{% blocktrans with type=form|form_verbose_name %}
|
||||||
|
Create {{ type }}
|
||||||
|
{% endblocktrans %}
|
||||||
|
</h1>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block action %}
|
||||||
|
{% blocktrans with type=form|form_verbose_name %}
|
||||||
|
Create {{ type }}
|
||||||
|
{% endblocktrans %}
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,38 @@
|
||||||
|
{% extends container_template|default:"administration/base.html" %}
|
||||||
|
|
||||||
|
{% load i18n %}
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load static %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<section class="pf-c-page__main-section pf-m-light">
|
||||||
|
<div class="pf-c-content">
|
||||||
|
{% block above_form %}
|
||||||
|
{% endblock %}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<section class="pf-c-page__main-section">
|
||||||
|
<div class="pf-l-stack">
|
||||||
|
<div class="pf-l-stack__item">
|
||||||
|
<div class="pf-c-card">
|
||||||
|
<div class="pf-c-card__body">
|
||||||
|
<form id="main-form" action="" method="post" class="pf-c-form pf-m-horizontal" enctype="multipart/form-data">
|
||||||
|
{% include 'partials/form_horizontal.html' with form=form %}
|
||||||
|
{% block beneath_form %}
|
||||||
|
{% endblock %}
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
<footer class="pf-c-modal-box__footer">
|
||||||
|
<input class="pf-c-button pf-m-primary" type="submit" form="main-form" value="{% block action %}{% endblock %}" />
|
||||||
|
<a class="pf-c-button pf-m-secondary" href="{% back %}">{% trans "Cancel" %}</a>
|
||||||
|
</footer>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block scripts %}
|
||||||
|
{{ block.super }}
|
||||||
|
{{ form.media.js }}
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,20 @@
|
||||||
|
{% extends base_template|default:"generic/form.html" %}
|
||||||
|
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load i18n %}
|
||||||
|
|
||||||
|
{% block above_form %}
|
||||||
|
<h1>
|
||||||
|
{% trans form.title %}
|
||||||
|
</h1>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block beneath_form %}
|
||||||
|
<p>
|
||||||
|
{% trans form.body %}
|
||||||
|
</p>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block action %}
|
||||||
|
{% trans 'Confirm' %}
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,18 @@
|
||||||
|
{% extends base_template|default:"generic/form.html" %}
|
||||||
|
|
||||||
|
{% load authentik_utils %}
|
||||||
|
{% load i18n %}
|
||||||
|
|
||||||
|
{% block above_form %}
|
||||||
|
<h1>
|
||||||
|
{% blocktrans with type=form|form_verbose_name|title inst=form.instance %}
|
||||||
|
Update {{ inst }}
|
||||||
|
{% endblocktrans %}
|
||||||
|
</h1>
|
||||||
|
{% endblock %}
|
||||||
|
|
||||||
|
{% block action %}
|
||||||
|
{% blocktrans with type=form|form_verbose_name %}
|
||||||
|
Update {{ type }}
|
||||||
|
{% endblocktrans %}
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,62 @@
|
||||||
|
"""authentik admin templatetags"""
|
||||||
|
from django import template
|
||||||
|
from django.db.models import Model
|
||||||
|
from django.utils.html import mark_safe
|
||||||
|
from structlog import get_logger
|
||||||
|
|
||||||
|
register = template.Library()
|
||||||
|
LOGGER = get_logger()
|
||||||
|
|
||||||
|
|
||||||
|
@register.simple_tag()
|
||||||
|
def get_links(model_instance):
|
||||||
|
"""Find all link_ methods on an object instance, run them and return as dict"""
|
||||||
|
prefix = "link_"
|
||||||
|
links = {}
|
||||||
|
|
||||||
|
if not isinstance(model_instance, Model):
|
||||||
|
LOGGER.warning("Model is not instance of Model", model_instance=model_instance)
|
||||||
|
return links
|
||||||
|
|
||||||
|
try:
|
||||||
|
for name in dir(model_instance):
|
||||||
|
if not name.startswith(prefix):
|
||||||
|
continue
|
||||||
|
value = getattr(model_instance, name)
|
||||||
|
if not callable(value):
|
||||||
|
continue
|
||||||
|
human_name = name.replace(prefix, "").replace("_", " ").capitalize()
|
||||||
|
link = value()
|
||||||
|
if link:
|
||||||
|
links[human_name] = link
|
||||||
|
except NotImplementedError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return links
|
||||||
|
|
||||||
|
|
||||||
|
@register.simple_tag(takes_context=True)
|
||||||
|
def get_htmls(context, model_instance):
|
||||||
|
"""Find all html_ methods on an object instance, run them and return as dict"""
|
||||||
|
prefix = "html_"
|
||||||
|
htmls = []
|
||||||
|
|
||||||
|
if not isinstance(model_instance, Model):
|
||||||
|
LOGGER.warning("Model is not instance of Model", model_instance=model_instance)
|
||||||
|
return htmls
|
||||||
|
|
||||||
|
try:
|
||||||
|
for name in dir(model_instance):
|
||||||
|
if not name.startswith(prefix):
|
||||||
|
continue
|
||||||
|
value = getattr(model_instance, name)
|
||||||
|
if not callable(value):
|
||||||
|
continue
|
||||||
|
if name.startswith(prefix):
|
||||||
|
html = value(context.get("request"))
|
||||||
|
if html:
|
||||||
|
htmls.append(mark_safe(html))
|
||||||
|
except NotImplementedError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return htmls
|
|
@ -0,0 +1,66 @@
|
||||||
|
"""admin tests"""
|
||||||
|
from importlib import import_module
|
||||||
|
from typing import Callable
|
||||||
|
|
||||||
|
from django.forms import ModelForm
|
||||||
|
from django.shortcuts import reverse
|
||||||
|
from django.test import Client, TestCase
|
||||||
|
from django.urls.exceptions import NoReverseMatch
|
||||||
|
|
||||||
|
from authentik.admin.urls import urlpatterns
|
||||||
|
from authentik.core.models import Group, User
|
||||||
|
from authentik.lib.utils.reflection import get_apps
|
||||||
|
|
||||||
|
|
||||||
|
class TestAdmin(TestCase):
|
||||||
|
"""Generic admin tests"""
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.user = User.objects.create_user(username="test")
|
||||||
|
self.user.ak_groups.add(Group.objects.filter(is_superuser=True).first())
|
||||||
|
self.user.save()
|
||||||
|
self.client = Client()
|
||||||
|
self.client.force_login(self.user)
|
||||||
|
|
||||||
|
|
||||||
|
def generic_view_tester(view_name: str) -> Callable:
|
||||||
|
"""This is used instead of subTest for better visibility"""
|
||||||
|
|
||||||
|
def tester(self: TestAdmin):
|
||||||
|
try:
|
||||||
|
full_url = reverse(f"authentik_admin:{view_name}")
|
||||||
|
response = self.client.get(full_url)
|
||||||
|
self.assertTrue(response.status_code < 500)
|
||||||
|
except NoReverseMatch:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return tester
|
||||||
|
|
||||||
|
|
||||||
|
for url in urlpatterns:
|
||||||
|
method_name = url.name.replace("-", "_")
|
||||||
|
setattr(TestAdmin, f"test_view_{method_name}", generic_view_tester(url.name))
|
||||||
|
|
||||||
|
|
||||||
|
def generic_form_tester(form: ModelForm) -> Callable:
|
||||||
|
"""Test a form"""
|
||||||
|
|
||||||
|
def tester(self: TestAdmin):
|
||||||
|
form_inst = form()
|
||||||
|
self.assertFalse(form_inst.is_valid())
|
||||||
|
|
||||||
|
return tester
|
||||||
|
|
||||||
|
|
||||||
|
# Load the forms module from every app, so we have all forms loaded
|
||||||
|
for app in get_apps():
|
||||||
|
module = app.__module__.replace(".apps", ".forms")
|
||||||
|
try:
|
||||||
|
import_module(module)
|
||||||
|
except ImportError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
for form_class in ModelForm.__subclasses__():
|
||||||
|
setattr(
|
||||||
|
TestAdmin, f"test_form_{form_class.__name__}", generic_form_tester(form_class)
|
||||||
|
)
|
|
@ -0,0 +1,353 @@
|
||||||
|
"""authentik URL Configuration"""
|
||||||
|
from django.urls import path
|
||||||
|
|
||||||
|
from authentik.admin.views import (
|
||||||
|
applications,
|
||||||
|
certificate_key_pair,
|
||||||
|
flows,
|
||||||
|
groups,
|
||||||
|
outposts,
|
||||||
|
outposts_service_connections,
|
||||||
|
overview,
|
||||||
|
policies,
|
||||||
|
policies_bindings,
|
||||||
|
property_mappings,
|
||||||
|
providers,
|
||||||
|
sources,
|
||||||
|
stages,
|
||||||
|
stages_bindings,
|
||||||
|
stages_invitations,
|
||||||
|
stages_prompts,
|
||||||
|
tasks,
|
||||||
|
tokens,
|
||||||
|
users,
|
||||||
|
)
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
path(
|
||||||
|
"overview/cache/flow/",
|
||||||
|
overview.FlowCacheClearView.as_view(),
|
||||||
|
name="overview-clear-flow-cache",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"overview/cache/policy/",
|
||||||
|
overview.PolicyCacheClearView.as_view(),
|
||||||
|
name="overview-clear-policy-cache",
|
||||||
|
),
|
||||||
|
path("overview/", overview.AdministrationOverviewView.as_view(), name="overview"),
|
||||||
|
# Applications
|
||||||
|
path(
|
||||||
|
"applications/", applications.ApplicationListView.as_view(), name="applications"
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"applications/create/",
|
||||||
|
applications.ApplicationCreateView.as_view(),
|
||||||
|
name="application-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"applications/<uuid:pk>/update/",
|
||||||
|
applications.ApplicationUpdateView.as_view(),
|
||||||
|
name="application-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"applications/<uuid:pk>/delete/",
|
||||||
|
applications.ApplicationDeleteView.as_view(),
|
||||||
|
name="application-delete",
|
||||||
|
),
|
||||||
|
# Tokens
|
||||||
|
path("tokens/", tokens.TokenListView.as_view(), name="tokens"),
|
||||||
|
path(
|
||||||
|
"tokens/<uuid:pk>/delete/",
|
||||||
|
tokens.TokenDeleteView.as_view(),
|
||||||
|
name="token-delete",
|
||||||
|
),
|
||||||
|
# Sources
|
||||||
|
path("sources/", sources.SourceListView.as_view(), name="sources"),
|
||||||
|
path("sources/create/", sources.SourceCreateView.as_view(), name="source-create"),
|
||||||
|
path(
|
||||||
|
"sources/<uuid:pk>/update/",
|
||||||
|
sources.SourceUpdateView.as_view(),
|
||||||
|
name="source-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"sources/<uuid:pk>/delete/",
|
||||||
|
sources.SourceDeleteView.as_view(),
|
||||||
|
name="source-delete",
|
||||||
|
),
|
||||||
|
# Policies
|
||||||
|
path("policies/", policies.PolicyListView.as_view(), name="policies"),
|
||||||
|
path("policies/create/", policies.PolicyCreateView.as_view(), name="policy-create"),
|
||||||
|
path(
|
||||||
|
"policies/<uuid:pk>/update/",
|
||||||
|
policies.PolicyUpdateView.as_view(),
|
||||||
|
name="policy-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"policies/<uuid:pk>/delete/",
|
||||||
|
policies.PolicyDeleteView.as_view(),
|
||||||
|
name="policy-delete",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"policies/<uuid:pk>/test/",
|
||||||
|
policies.PolicyTestView.as_view(),
|
||||||
|
name="policy-test",
|
||||||
|
),
|
||||||
|
# Policy bindings
|
||||||
|
path(
|
||||||
|
"policies/bindings/",
|
||||||
|
policies_bindings.PolicyBindingListView.as_view(),
|
||||||
|
name="policies-bindings",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"policies/bindings/create/",
|
||||||
|
policies_bindings.PolicyBindingCreateView.as_view(),
|
||||||
|
name="policy-binding-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"policies/bindings/<uuid:pk>/update/",
|
||||||
|
policies_bindings.PolicyBindingUpdateView.as_view(),
|
||||||
|
name="policy-binding-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"policies/bindings/<uuid:pk>/delete/",
|
||||||
|
policies_bindings.PolicyBindingDeleteView.as_view(),
|
||||||
|
name="policy-binding-delete",
|
||||||
|
),
|
||||||
|
# Providers
|
||||||
|
path("providers/", providers.ProviderListView.as_view(), name="providers"),
|
||||||
|
path(
|
||||||
|
"providers/create/",
|
||||||
|
providers.ProviderCreateView.as_view(),
|
||||||
|
name="provider-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"providers/<int:pk>/update/",
|
||||||
|
providers.ProviderUpdateView.as_view(),
|
||||||
|
name="provider-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"providers/<int:pk>/delete/",
|
||||||
|
providers.ProviderDeleteView.as_view(),
|
||||||
|
name="provider-delete",
|
||||||
|
),
|
||||||
|
# Stages
|
||||||
|
path("stages/", stages.StageListView.as_view(), name="stages"),
|
||||||
|
path("stages/create/", stages.StageCreateView.as_view(), name="stage-create"),
|
||||||
|
path(
|
||||||
|
"stages/<uuid:pk>/update/",
|
||||||
|
stages.StageUpdateView.as_view(),
|
||||||
|
name="stage-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/<uuid:pk>/delete/",
|
||||||
|
stages.StageDeleteView.as_view(),
|
||||||
|
name="stage-delete",
|
||||||
|
),
|
||||||
|
# Stage bindings
|
||||||
|
path(
|
||||||
|
"stages/bindings/",
|
||||||
|
stages_bindings.StageBindingListView.as_view(),
|
||||||
|
name="stage-bindings",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/bindings/create/",
|
||||||
|
stages_bindings.StageBindingCreateView.as_view(),
|
||||||
|
name="stage-binding-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/bindings/<uuid:pk>/update/",
|
||||||
|
stages_bindings.StageBindingUpdateView.as_view(),
|
||||||
|
name="stage-binding-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/bindings/<uuid:pk>/delete/",
|
||||||
|
stages_bindings.StageBindingDeleteView.as_view(),
|
||||||
|
name="stage-binding-delete",
|
||||||
|
),
|
||||||
|
# Stage Prompts
|
||||||
|
path(
|
||||||
|
"stages/prompts/",
|
||||||
|
stages_prompts.PromptListView.as_view(),
|
||||||
|
name="stage-prompts",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/prompts/create/",
|
||||||
|
stages_prompts.PromptCreateView.as_view(),
|
||||||
|
name="stage-prompt-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/prompts/<uuid:pk>/update/",
|
||||||
|
stages_prompts.PromptUpdateView.as_view(),
|
||||||
|
name="stage-prompt-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/prompts/<uuid:pk>/delete/",
|
||||||
|
stages_prompts.PromptDeleteView.as_view(),
|
||||||
|
name="stage-prompt-delete",
|
||||||
|
),
|
||||||
|
# Stage Invitations
|
||||||
|
path(
|
||||||
|
"stages/invitations/",
|
||||||
|
stages_invitations.InvitationListView.as_view(),
|
||||||
|
name="stage-invitations",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/invitations/create/",
|
||||||
|
stages_invitations.InvitationCreateView.as_view(),
|
||||||
|
name="stage-invitation-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"stages/invitations/<uuid:pk>/delete/",
|
||||||
|
stages_invitations.InvitationDeleteView.as_view(),
|
||||||
|
name="stage-invitation-delete",
|
||||||
|
),
|
||||||
|
# Flows
|
||||||
|
path("flows/", flows.FlowListView.as_view(), name="flows"),
|
||||||
|
path(
|
||||||
|
"flows/create/",
|
||||||
|
flows.FlowCreateView.as_view(),
|
||||||
|
name="flow-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"flows/import/",
|
||||||
|
flows.FlowImportView.as_view(),
|
||||||
|
name="flow-import",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"flows/<uuid:pk>/update/",
|
||||||
|
flows.FlowUpdateView.as_view(),
|
||||||
|
name="flow-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"flows/<uuid:pk>/execute/",
|
||||||
|
flows.FlowDebugExecuteView.as_view(),
|
||||||
|
name="flow-execute",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"flows/<uuid:pk>/export/",
|
||||||
|
flows.FlowExportView.as_view(),
|
||||||
|
name="flow-export",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"flows/<uuid:pk>/delete/",
|
||||||
|
flows.FlowDeleteView.as_view(),
|
||||||
|
name="flow-delete",
|
||||||
|
),
|
||||||
|
# Property Mappings
|
||||||
|
path(
|
||||||
|
"property-mappings/",
|
||||||
|
property_mappings.PropertyMappingListView.as_view(),
|
||||||
|
name="property-mappings",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"property-mappings/create/",
|
||||||
|
property_mappings.PropertyMappingCreateView.as_view(),
|
||||||
|
name="property-mapping-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"property-mappings/<uuid:pk>/update/",
|
||||||
|
property_mappings.PropertyMappingUpdateView.as_view(),
|
||||||
|
name="property-mapping-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"property-mappings/<uuid:pk>/delete/",
|
||||||
|
property_mappings.PropertyMappingDeleteView.as_view(),
|
||||||
|
name="property-mapping-delete",
|
||||||
|
),
|
||||||
|
# Users
|
||||||
|
path("users/", users.UserListView.as_view(), name="users"),
|
||||||
|
path("users/create/", users.UserCreateView.as_view(), name="user-create"),
|
||||||
|
path("users/<int:pk>/update/", users.UserUpdateView.as_view(), name="user-update"),
|
||||||
|
path("users/<int:pk>/delete/", users.UserDeleteView.as_view(), name="user-delete"),
|
||||||
|
path(
|
||||||
|
"users/<int:pk>/disable/", users.UserDisableView.as_view(), name="user-disable"
|
||||||
|
),
|
||||||
|
path("users/<int:pk>/enable/", users.UserEnableView.as_view(), name="user-enable"),
|
||||||
|
path(
|
||||||
|
"users/<int:pk>/reset/",
|
||||||
|
users.UserPasswordResetView.as_view(),
|
||||||
|
name="user-password-reset",
|
||||||
|
),
|
||||||
|
# Groups
|
||||||
|
path("groups/", groups.GroupListView.as_view(), name="groups"),
|
||||||
|
path("groups/create/", groups.GroupCreateView.as_view(), name="group-create"),
|
||||||
|
path(
|
||||||
|
"groups/<uuid:pk>/update/",
|
||||||
|
groups.GroupUpdateView.as_view(),
|
||||||
|
name="group-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"groups/<uuid:pk>/delete/",
|
||||||
|
groups.GroupDeleteView.as_view(),
|
||||||
|
name="group-delete",
|
||||||
|
),
|
||||||
|
# Certificate-Key Pairs
|
||||||
|
path(
|
||||||
|
"crypto/certificates/",
|
||||||
|
certificate_key_pair.CertificateKeyPairListView.as_view(),
|
||||||
|
name="certificate_key_pair",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"crypto/certificates/create/",
|
||||||
|
certificate_key_pair.CertificateKeyPairCreateView.as_view(),
|
||||||
|
name="certificatekeypair-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"crypto/certificates/<uuid:pk>/update/",
|
||||||
|
certificate_key_pair.CertificateKeyPairUpdateView.as_view(),
|
||||||
|
name="certificatekeypair-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"crypto/certificates/<uuid:pk>/delete/",
|
||||||
|
certificate_key_pair.CertificateKeyPairDeleteView.as_view(),
|
||||||
|
name="certificatekeypair-delete",
|
||||||
|
),
|
||||||
|
# Outposts
|
||||||
|
path(
|
||||||
|
"outposts/",
|
||||||
|
outposts.OutpostListView.as_view(),
|
||||||
|
name="outposts",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"outposts/create/",
|
||||||
|
outposts.OutpostCreateView.as_view(),
|
||||||
|
name="outpost-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"outposts/<uuid:pk>/update/",
|
||||||
|
outposts.OutpostUpdateView.as_view(),
|
||||||
|
name="outpost-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"outposts/<uuid:pk>/delete/",
|
||||||
|
outposts.OutpostDeleteView.as_view(),
|
||||||
|
name="outpost-delete",
|
||||||
|
),
|
||||||
|
# Outpost Service Connections
|
||||||
|
path(
|
||||||
|
"outposts/service_connections/",
|
||||||
|
outposts_service_connections.OutpostServiceConnectionListView.as_view(),
|
||||||
|
name="outpost-service-connections",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"outposts/service_connections/create/",
|
||||||
|
outposts_service_connections.OutpostServiceConnectionCreateView.as_view(),
|
||||||
|
name="outpost-service-connection-create",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"outposts/service_connections/<uuid:pk>/update/",
|
||||||
|
outposts_service_connections.OutpostServiceConnectionUpdateView.as_view(),
|
||||||
|
name="outpost-service-connection-update",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"outposts/service_connections/<uuid:pk>/delete/",
|
||||||
|
outposts_service_connections.OutpostServiceConnectionDeleteView.as_view(),
|
||||||
|
name="outpost-service-connection-delete",
|
||||||
|
),
|
||||||
|
# Tasks
|
||||||
|
path(
|
||||||
|
"tasks/",
|
||||||
|
tasks.TaskListView.as_view(),
|
||||||
|
name="tasks",
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,93 @@
|
||||||
|
"""authentik Application administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.core.forms.applications import ApplicationForm
|
||||||
|
from authentik.core.models import Application
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
|
||||||
|
|
||||||
|
class ApplicationListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all applications"""
|
||||||
|
|
||||||
|
model = Application
|
||||||
|
permission_required = "authentik_core.view_application"
|
||||||
|
ordering = "name"
|
||||||
|
template_name = "administration/application/list.html"
|
||||||
|
|
||||||
|
search_fields = [
|
||||||
|
"name",
|
||||||
|
"slug",
|
||||||
|
"meta_launch_url",
|
||||||
|
"meta_icon_url",
|
||||||
|
"meta_description",
|
||||||
|
"meta_publisher",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ApplicationCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new Application"""
|
||||||
|
|
||||||
|
model = Application
|
||||||
|
form_class = ApplicationForm
|
||||||
|
permission_required = "authentik_core.add_application"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:applications")
|
||||||
|
success_message = _("Successfully created Application")
|
||||||
|
|
||||||
|
|
||||||
|
class ApplicationUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update application"""
|
||||||
|
|
||||||
|
model = Application
|
||||||
|
form_class = ApplicationForm
|
||||||
|
permission_required = "authentik_core.change_application"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:applications")
|
||||||
|
success_message = _("Successfully updated Application")
|
||||||
|
|
||||||
|
|
||||||
|
class ApplicationDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete application"""
|
||||||
|
|
||||||
|
model = Application
|
||||||
|
permission_required = "authentik_core.delete_application"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:applications")
|
||||||
|
success_message = _("Successfully deleted Application")
|
|
@ -0,0 +1,86 @@
|
||||||
|
"""authentik CertificateKeyPair administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.crypto.forms import CertificateKeyPairForm
|
||||||
|
from authentik.crypto.models import CertificateKeyPair
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
|
||||||
|
|
||||||
|
class CertificateKeyPairListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all keypairs"""
|
||||||
|
|
||||||
|
model = CertificateKeyPair
|
||||||
|
permission_required = "authentik_crypto.view_certificatekeypair"
|
||||||
|
ordering = "name"
|
||||||
|
template_name = "administration/certificatekeypair/list.html"
|
||||||
|
|
||||||
|
search_fields = ["name"]
|
||||||
|
|
||||||
|
|
||||||
|
class CertificateKeyPairCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new CertificateKeyPair"""
|
||||||
|
|
||||||
|
model = CertificateKeyPair
|
||||||
|
form_class = CertificateKeyPairForm
|
||||||
|
permission_required = "authentik_crypto.add_certificatekeypair"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:certificate_key_pair")
|
||||||
|
success_message = _("Successfully created CertificateKeyPair")
|
||||||
|
|
||||||
|
|
||||||
|
class CertificateKeyPairUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update certificatekeypair"""
|
||||||
|
|
||||||
|
model = CertificateKeyPair
|
||||||
|
form_class = CertificateKeyPairForm
|
||||||
|
permission_required = "authentik_crypto.change_certificatekeypair"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:certificate_key_pair")
|
||||||
|
success_message = _("Successfully updated Certificate-Key Pair")
|
||||||
|
|
||||||
|
|
||||||
|
class CertificateKeyPairDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete certificatekeypair"""
|
||||||
|
|
||||||
|
model = CertificateKeyPair
|
||||||
|
permission_required = "authentik_crypto.delete_certificatekeypair"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:certificate_key_pair")
|
||||||
|
success_message = _("Successfully deleted Certificate-Key Pair")
|
|
@ -0,0 +1,151 @@
|
||||||
|
"""authentik Flow administration"""
|
||||||
|
from django.contrib import messages
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.http import HttpRequest, HttpResponse, JsonResponse
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import DetailView, FormView, ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.flows.forms import FlowForm, FlowImportForm
|
||||||
|
from authentik.flows.models import Flow
|
||||||
|
from authentik.flows.planner import PLAN_CONTEXT_PENDING_USER
|
||||||
|
from authentik.flows.transfer.common import DataclassEncoder
|
||||||
|
from authentik.flows.transfer.exporter import FlowExporter
|
||||||
|
from authentik.flows.transfer.importer import FlowImporter
|
||||||
|
from authentik.flows.views import SESSION_KEY_PLAN, FlowPlanner
|
||||||
|
from authentik.lib.utils.urls import redirect_with_qs
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
|
||||||
|
|
||||||
|
class FlowListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all flows"""
|
||||||
|
|
||||||
|
model = Flow
|
||||||
|
permission_required = "authentik_flows.view_flow"
|
||||||
|
ordering = "name"
|
||||||
|
template_name = "administration/flow/list.html"
|
||||||
|
search_fields = ["name", "slug", "designation", "title"]
|
||||||
|
|
||||||
|
|
||||||
|
class FlowCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new Flow"""
|
||||||
|
|
||||||
|
model = Flow
|
||||||
|
form_class = FlowForm
|
||||||
|
permission_required = "authentik_flows.add_flow"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:flows")
|
||||||
|
success_message = _("Successfully created Flow")
|
||||||
|
|
||||||
|
|
||||||
|
class FlowUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update flow"""
|
||||||
|
|
||||||
|
model = Flow
|
||||||
|
form_class = FlowForm
|
||||||
|
permission_required = "authentik_flows.change_flow"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:flows")
|
||||||
|
success_message = _("Successfully updated Flow")
|
||||||
|
|
||||||
|
|
||||||
|
class FlowDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete flow"""
|
||||||
|
|
||||||
|
model = Flow
|
||||||
|
permission_required = "authentik_flows.delete_flow"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:flows")
|
||||||
|
success_message = _("Successfully deleted Flow")
|
||||||
|
|
||||||
|
|
||||||
|
class FlowDebugExecuteView(LoginRequiredMixin, PermissionRequiredMixin, DetailView):
|
||||||
|
"""Debug exectue flow, setting the current user as pending user"""
|
||||||
|
|
||||||
|
model = Flow
|
||||||
|
permission_required = "authentik_flows.view_flow"
|
||||||
|
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def get(self, request: HttpRequest, pk: str) -> HttpResponse:
|
||||||
|
"""Debug exectue flow, setting the current user as pending user"""
|
||||||
|
flow: Flow = self.get_object()
|
||||||
|
planner = FlowPlanner(flow)
|
||||||
|
planner.use_cache = False
|
||||||
|
plan = planner.plan(self.request, {PLAN_CONTEXT_PENDING_USER: request.user})
|
||||||
|
self.request.session[SESSION_KEY_PLAN] = plan
|
||||||
|
return redirect_with_qs(
|
||||||
|
"authentik_flows:flow-executor-shell",
|
||||||
|
self.request.GET,
|
||||||
|
flow_slug=flow.slug,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class FlowImportView(LoginRequiredMixin, FormView):
|
||||||
|
"""Import flow from JSON Export; only allowed for superusers
|
||||||
|
as these flows can contain python code"""
|
||||||
|
|
||||||
|
form_class = FlowImportForm
|
||||||
|
template_name = "administration/flow/import.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:flows")
|
||||||
|
|
||||||
|
def dispatch(self, request, *args, **kwargs):
|
||||||
|
if not request.user.is_superuser:
|
||||||
|
return self.handle_no_permission()
|
||||||
|
return super().dispatch(request, *args, **kwargs)
|
||||||
|
|
||||||
|
def form_valid(self, form: FlowImportForm) -> HttpResponse:
|
||||||
|
importer = FlowImporter(form.cleaned_data["flow"].read().decode())
|
||||||
|
successful = importer.apply()
|
||||||
|
if not successful:
|
||||||
|
messages.error(self.request, _("Failed to import flow."))
|
||||||
|
else:
|
||||||
|
messages.success(self.request, _("Successfully imported flow."))
|
||||||
|
return super().form_valid(form)
|
||||||
|
|
||||||
|
|
||||||
|
class FlowExportView(LoginRequiredMixin, PermissionRequiredMixin, DetailView):
|
||||||
|
"""Export Flow"""
|
||||||
|
|
||||||
|
model = Flow
|
||||||
|
permission_required = "authentik_flows.export_flow"
|
||||||
|
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def get(self, request: HttpRequest, pk: str) -> HttpResponse:
|
||||||
|
"""Debug exectue flow, setting the current user as pending user"""
|
||||||
|
flow: Flow = self.get_object()
|
||||||
|
exporter = FlowExporter(flow)
|
||||||
|
response = JsonResponse(exporter.export(), encoder=DataclassEncoder, safe=False)
|
||||||
|
response["Content-Disposition"] = f'attachment; filename="{flow.slug}.akflow"'
|
||||||
|
return response
|
|
@ -0,0 +1,83 @@
|
||||||
|
"""authentik Group administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.core.forms.groups import GroupForm
|
||||||
|
from authentik.core.models import Group
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
|
||||||
|
|
||||||
|
class GroupListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all groups"""
|
||||||
|
|
||||||
|
model = Group
|
||||||
|
permission_required = "authentik_core.view_group"
|
||||||
|
ordering = "name"
|
||||||
|
template_name = "administration/group/list.html"
|
||||||
|
search_fields = ["name", "attributes"]
|
||||||
|
|
||||||
|
|
||||||
|
class GroupCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new Group"""
|
||||||
|
|
||||||
|
model = Group
|
||||||
|
form_class = GroupForm
|
||||||
|
permission_required = "authentik_core.add_group"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:groups")
|
||||||
|
success_message = _("Successfully created Group")
|
||||||
|
|
||||||
|
|
||||||
|
class GroupUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update group"""
|
||||||
|
|
||||||
|
model = Group
|
||||||
|
form_class = GroupForm
|
||||||
|
permission_required = "authentik_core.change_group"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:groups")
|
||||||
|
success_message = _("Successfully updated Group")
|
||||||
|
|
||||||
|
|
||||||
|
class GroupDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete group"""
|
||||||
|
|
||||||
|
model = Group
|
||||||
|
permission_required = "authentik_flows.delete_group"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:groups")
|
||||||
|
success_message = _("Successfully deleted Group")
|
|
@ -0,0 +1,93 @@
|
||||||
|
"""authentik Outpost administration"""
|
||||||
|
from dataclasses import asdict
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
from authentik.outposts.forms import OutpostForm
|
||||||
|
from authentik.outposts.models import Outpost, OutpostConfig
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all outposts"""
|
||||||
|
|
||||||
|
model = Outpost
|
||||||
|
permission_required = "authentik_outposts.view_outpost"
|
||||||
|
ordering = "name"
|
||||||
|
template_name = "administration/outpost/list.html"
|
||||||
|
search_fields = ["name", "_config"]
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new Outpost"""
|
||||||
|
|
||||||
|
model = Outpost
|
||||||
|
form_class = OutpostForm
|
||||||
|
permission_required = "authentik_outposts.add_outpost"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:outposts")
|
||||||
|
success_message = _("Successfully created Outpost")
|
||||||
|
|
||||||
|
def get_initial(self) -> Dict[str, Any]:
|
||||||
|
return {
|
||||||
|
"_config": asdict(
|
||||||
|
OutpostConfig(authentik_host=self.request.build_absolute_uri("/"))
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update outpost"""
|
||||||
|
|
||||||
|
model = Outpost
|
||||||
|
form_class = OutpostForm
|
||||||
|
permission_required = "authentik_outposts.change_outpost"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:outposts")
|
||||||
|
success_message = _("Successfully updated Outpost")
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete outpost"""
|
||||||
|
|
||||||
|
model = Outpost
|
||||||
|
permission_required = "authentik_outposts.delete_outpost"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:outposts")
|
||||||
|
success_message = _("Successfully deleted Outpost")
|
|
@ -0,0 +1,83 @@
|
||||||
|
"""authentik OutpostServiceConnection administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
InheritanceCreateView,
|
||||||
|
InheritanceListView,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.outposts.models import OutpostServiceConnection
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostServiceConnectionListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
InheritanceListView,
|
||||||
|
):
|
||||||
|
"""Show list of all outpost-service-connections"""
|
||||||
|
|
||||||
|
model = OutpostServiceConnection
|
||||||
|
permission_required = "authentik_outposts.add_outpostserviceconnection"
|
||||||
|
template_name = "administration/outpost_service_connection/list.html"
|
||||||
|
ordering = "pk"
|
||||||
|
search_fields = ["pk", "name"]
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostServiceConnectionCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
InheritanceCreateView,
|
||||||
|
):
|
||||||
|
"""Create new OutpostServiceConnection"""
|
||||||
|
|
||||||
|
model = OutpostServiceConnection
|
||||||
|
permission_required = "authentik_outposts.add_outpostserviceconnection"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:outpost-service-connections")
|
||||||
|
success_message = _("Successfully created OutpostServiceConnection")
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostServiceConnectionUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
):
|
||||||
|
"""Update outpostserviceconnection"""
|
||||||
|
|
||||||
|
model = OutpostServiceConnection
|
||||||
|
permission_required = "authentik_outposts.change_outpostserviceconnection"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:outpost-service-connections")
|
||||||
|
success_message = _("Successfully updated OutpostServiceConnection")
|
||||||
|
|
||||||
|
|
||||||
|
class OutpostServiceConnectionDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete outpostserviceconnection"""
|
||||||
|
|
||||||
|
model = OutpostServiceConnection
|
||||||
|
permission_required = "authentik_outposts.delete_outpostserviceconnection"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:outpost-service-connections")
|
||||||
|
success_message = _("Successfully deleted OutpostServiceConnection")
|
|
@ -0,0 +1,85 @@
|
||||||
|
"""authentik administration overview"""
|
||||||
|
from typing import Union
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.core.cache import cache
|
||||||
|
from django.http.request import HttpRequest
|
||||||
|
from django.http.response import HttpResponse
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import FormView, TemplateView
|
||||||
|
from packaging.version import LegacyVersion, Version, parse
|
||||||
|
from structlog import get_logger
|
||||||
|
|
||||||
|
from authentik import __version__
|
||||||
|
from authentik.admin.forms.overview import FlowCacheClearForm, PolicyCacheClearForm
|
||||||
|
from authentik.admin.mixins import AdminRequiredMixin
|
||||||
|
from authentik.admin.tasks import VERSION_CACHE_KEY, update_latest_version
|
||||||
|
from authentik.core.models import Provider, User
|
||||||
|
from authentik.policies.models import Policy
|
||||||
|
|
||||||
|
LOGGER = get_logger()
|
||||||
|
|
||||||
|
|
||||||
|
class AdministrationOverviewView(AdminRequiredMixin, TemplateView):
|
||||||
|
"""Overview View"""
|
||||||
|
|
||||||
|
template_name = "administration/overview.html"
|
||||||
|
|
||||||
|
def get_latest_version(self) -> Union[LegacyVersion, Version]:
|
||||||
|
"""Get latest version from cache"""
|
||||||
|
version_in_cache = cache.get(VERSION_CACHE_KEY)
|
||||||
|
if not version_in_cache:
|
||||||
|
if not settings.DEBUG:
|
||||||
|
update_latest_version.delay()
|
||||||
|
return parse(__version__)
|
||||||
|
return parse(version_in_cache)
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
kwargs["policy_count"] = len(Policy.objects.all())
|
||||||
|
kwargs["user_count"] = len(User.objects.all()) - 1 # Remove anonymous user
|
||||||
|
kwargs["provider_count"] = len(Provider.objects.all())
|
||||||
|
kwargs["version"] = parse(__version__)
|
||||||
|
kwargs["version_latest"] = self.get_latest_version()
|
||||||
|
kwargs["providers_without_application"] = Provider.objects.filter(
|
||||||
|
application=None
|
||||||
|
)
|
||||||
|
kwargs["policies_without_binding"] = len(
|
||||||
|
Policy.objects.filter(bindings__isnull=True, promptstage__isnull=True)
|
||||||
|
)
|
||||||
|
kwargs["cached_policies"] = len(cache.keys("policy_*"))
|
||||||
|
kwargs["cached_flows"] = len(cache.keys("flow_*"))
|
||||||
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyCacheClearView(AdminRequiredMixin, SuccessMessageMixin, FormView):
|
||||||
|
"""View to clear Policy cache"""
|
||||||
|
|
||||||
|
form_class = PolicyCacheClearForm
|
||||||
|
|
||||||
|
template_name = "generic/form_non_model.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:overview")
|
||||||
|
success_message = _("Successfully cleared Policy cache")
|
||||||
|
|
||||||
|
def post(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
|
||||||
|
keys = cache.keys("policy_*")
|
||||||
|
cache.delete_many(keys)
|
||||||
|
LOGGER.debug("Cleared Policy cache", keys=len(keys))
|
||||||
|
return super().post(request, *args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
class FlowCacheClearView(AdminRequiredMixin, SuccessMessageMixin, FormView):
|
||||||
|
"""View to clear Flow cache"""
|
||||||
|
|
||||||
|
form_class = FlowCacheClearForm
|
||||||
|
|
||||||
|
template_name = "generic/form_non_model.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:overview")
|
||||||
|
success_message = _("Successfully cleared Flow cache")
|
||||||
|
|
||||||
|
def post(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
|
||||||
|
keys = cache.keys("flow_*")
|
||||||
|
cache.delete_many(keys)
|
||||||
|
LOGGER.debug("Cleared flow cache", keys=len(keys))
|
||||||
|
return super().post(request, *args, **kwargs)
|
|
@ -0,0 +1,129 @@
|
||||||
|
"""authentik Policy administration"""
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
|
from django.contrib import messages
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.db.models import QuerySet
|
||||||
|
from django.http import HttpResponse
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import FormView
|
||||||
|
from django.views.generic.detail import DetailView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.forms.policies import PolicyTestForm
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
InheritanceCreateView,
|
||||||
|
InheritanceListView,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.policies.models import Policy, PolicyBinding
|
||||||
|
from authentik.policies.process import PolicyProcess, PolicyRequest
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
InheritanceListView,
|
||||||
|
):
|
||||||
|
"""Show list of all policies"""
|
||||||
|
|
||||||
|
model = Policy
|
||||||
|
permission_required = "authentik_policies.view_policy"
|
||||||
|
ordering = "name"
|
||||||
|
template_name = "administration/policy/list.html"
|
||||||
|
search_fields = ["name"]
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
InheritanceCreateView,
|
||||||
|
):
|
||||||
|
"""Create new Policy"""
|
||||||
|
|
||||||
|
model = Policy
|
||||||
|
permission_required = "authentik_policies.add_policy"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:policies")
|
||||||
|
success_message = _("Successfully created Policy")
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
):
|
||||||
|
"""Update policy"""
|
||||||
|
|
||||||
|
model = Policy
|
||||||
|
permission_required = "authentik_policies.change_policy"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:policies")
|
||||||
|
success_message = _("Successfully updated Policy")
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete policy"""
|
||||||
|
|
||||||
|
model = Policy
|
||||||
|
permission_required = "authentik_policies.delete_policy"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:policies")
|
||||||
|
success_message = _("Successfully deleted Policy")
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyTestView(LoginRequiredMixin, DetailView, PermissionRequiredMixin, FormView):
|
||||||
|
"""View to test policy(s)"""
|
||||||
|
|
||||||
|
model = Policy
|
||||||
|
form_class = PolicyTestForm
|
||||||
|
permission_required = "authentik_policies.view_policy"
|
||||||
|
template_name = "administration/policy/test.html"
|
||||||
|
object = None
|
||||||
|
|
||||||
|
def get_object(self, queryset=None) -> QuerySet:
|
||||||
|
return (
|
||||||
|
Policy.objects.filter(pk=self.kwargs.get("pk")).select_subclasses().first()
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
|
||||||
|
kwargs["policy"] = self.get_object()
|
||||||
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
def post(self, *args, **kwargs) -> HttpResponse:
|
||||||
|
self.object = self.get_object()
|
||||||
|
return super().post(*args, **kwargs)
|
||||||
|
|
||||||
|
def form_valid(self, form: PolicyTestForm) -> HttpResponse:
|
||||||
|
policy = self.get_object()
|
||||||
|
user = form.cleaned_data.get("user")
|
||||||
|
|
||||||
|
p_request = PolicyRequest(user)
|
||||||
|
p_request.http_request = self.request
|
||||||
|
p_request.context = form.cleaned_data
|
||||||
|
|
||||||
|
proc = PolicyProcess(PolicyBinding(policy=policy), p_request, None)
|
||||||
|
result = proc.execute()
|
||||||
|
if result.passing:
|
||||||
|
messages.success(self.request, _("User successfully passed policy."))
|
||||||
|
else:
|
||||||
|
messages.error(self.request, _("User didn't pass policy."))
|
||||||
|
return self.render_to_response(self.get_context_data(form=form, result=result))
|
|
@ -0,0 +1,99 @@
|
||||||
|
"""authentik PolicyBinding administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.db.models import QuerySet
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
from guardian.shortcuts import get_objects_for_user
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
from authentik.policies.forms import PolicyBindingForm
|
||||||
|
from authentik.policies.models import PolicyBinding
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyBindingListView(
|
||||||
|
LoginRequiredMixin, PermissionListMixin, UserPaginateListMixin, ListView
|
||||||
|
):
|
||||||
|
"""Show list of all policies"""
|
||||||
|
|
||||||
|
model = PolicyBinding
|
||||||
|
permission_required = "authentik_policies.view_policybinding"
|
||||||
|
ordering = ["order", "target"]
|
||||||
|
template_name = "administration/policy_binding/list.html"
|
||||||
|
|
||||||
|
def get_queryset(self) -> QuerySet:
|
||||||
|
# Since `select_subclasses` does not work with a foreign key, we have to do two queries here
|
||||||
|
# First, get all pbm objects that have bindings attached
|
||||||
|
objects = (
|
||||||
|
get_objects_for_user(
|
||||||
|
self.request.user, "authentik_policies.view_policybindingmodel"
|
||||||
|
)
|
||||||
|
.filter(policies__isnull=False)
|
||||||
|
.select_subclasses()
|
||||||
|
.select_related()
|
||||||
|
.order_by("pk")
|
||||||
|
)
|
||||||
|
for pbm in objects:
|
||||||
|
pbm.bindings = get_objects_for_user(
|
||||||
|
self.request.user, self.permission_required
|
||||||
|
).filter(target__pk=pbm.pbm_uuid)
|
||||||
|
return objects
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyBindingCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new PolicyBinding"""
|
||||||
|
|
||||||
|
model = PolicyBinding
|
||||||
|
permission_required = "authentik_policies.add_policybinding"
|
||||||
|
form_class = PolicyBindingForm
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:policies-bindings")
|
||||||
|
success_message = _("Successfully created PolicyBinding")
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyBindingUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update policybinding"""
|
||||||
|
|
||||||
|
model = PolicyBinding
|
||||||
|
permission_required = "authentik_policies.change_policybinding"
|
||||||
|
form_class = PolicyBindingForm
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:policies-bindings")
|
||||||
|
success_message = _("Successfully updated PolicyBinding")
|
||||||
|
|
||||||
|
|
||||||
|
class PolicyBindingDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete policybinding"""
|
||||||
|
|
||||||
|
model = PolicyBinding
|
||||||
|
permission_required = "authentik_policies.delete_policybinding"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:policies-bindings")
|
||||||
|
success_message = _("Successfully deleted PolicyBinding")
|
|
@ -0,0 +1,83 @@
|
||||||
|
"""authentik PropertyMapping administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
InheritanceCreateView,
|
||||||
|
InheritanceListView,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.core.models import PropertyMapping
|
||||||
|
|
||||||
|
|
||||||
|
class PropertyMappingListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
InheritanceListView,
|
||||||
|
):
|
||||||
|
"""Show list of all property_mappings"""
|
||||||
|
|
||||||
|
model = PropertyMapping
|
||||||
|
permission_required = "authentik_core.view_propertymapping"
|
||||||
|
template_name = "administration/property_mapping/list.html"
|
||||||
|
ordering = "name"
|
||||||
|
search_fields = ["name", "expression"]
|
||||||
|
|
||||||
|
|
||||||
|
class PropertyMappingCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
InheritanceCreateView,
|
||||||
|
):
|
||||||
|
"""Create new PropertyMapping"""
|
||||||
|
|
||||||
|
model = PropertyMapping
|
||||||
|
permission_required = "authentik_core.add_propertymapping"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:property-mappings")
|
||||||
|
success_message = _("Successfully created Property Mapping")
|
||||||
|
|
||||||
|
|
||||||
|
class PropertyMappingUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
):
|
||||||
|
"""Update property_mapping"""
|
||||||
|
|
||||||
|
model = PropertyMapping
|
||||||
|
permission_required = "authentik_core.change_propertymapping"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:property-mappings")
|
||||||
|
success_message = _("Successfully updated Property Mapping")
|
||||||
|
|
||||||
|
|
||||||
|
class PropertyMappingDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete property_mapping"""
|
||||||
|
|
||||||
|
model = PropertyMapping
|
||||||
|
permission_required = "authentik_core.delete_propertymapping"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:property-mappings")
|
||||||
|
success_message = _("Successfully deleted Property Mapping")
|
|
@ -0,0 +1,83 @@
|
||||||
|
"""authentik Provider administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
InheritanceCreateView,
|
||||||
|
InheritanceListView,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.core.models import Provider
|
||||||
|
|
||||||
|
|
||||||
|
class ProviderListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
InheritanceListView,
|
||||||
|
):
|
||||||
|
"""Show list of all providers"""
|
||||||
|
|
||||||
|
model = Provider
|
||||||
|
permission_required = "authentik_core.add_provider"
|
||||||
|
template_name = "administration/provider/list.html"
|
||||||
|
ordering = "pk"
|
||||||
|
search_fields = ["pk", "name"]
|
||||||
|
|
||||||
|
|
||||||
|
class ProviderCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
InheritanceCreateView,
|
||||||
|
):
|
||||||
|
"""Create new Provider"""
|
||||||
|
|
||||||
|
model = Provider
|
||||||
|
permission_required = "authentik_core.add_provider"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:providers")
|
||||||
|
success_message = _("Successfully created Provider")
|
||||||
|
|
||||||
|
|
||||||
|
class ProviderUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
):
|
||||||
|
"""Update provider"""
|
||||||
|
|
||||||
|
model = Provider
|
||||||
|
permission_required = "authentik_core.change_provider"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:providers")
|
||||||
|
success_message = _("Successfully updated Provider")
|
||||||
|
|
||||||
|
|
||||||
|
class ProviderDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete provider"""
|
||||||
|
|
||||||
|
model = Provider
|
||||||
|
permission_required = "authentik_core.delete_provider"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:providers")
|
||||||
|
success_message = _("Successfully deleted Provider")
|
|
@ -0,0 +1,81 @@
|
||||||
|
"""authentik Source administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
InheritanceCreateView,
|
||||||
|
InheritanceListView,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.core.models import Source
|
||||||
|
|
||||||
|
|
||||||
|
class SourceListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
InheritanceListView,
|
||||||
|
):
|
||||||
|
"""Show list of all sources"""
|
||||||
|
|
||||||
|
model = Source
|
||||||
|
permission_required = "authentik_core.view_source"
|
||||||
|
ordering = "name"
|
||||||
|
template_name = "administration/source/list.html"
|
||||||
|
search_fields = ["name", "slug"]
|
||||||
|
|
||||||
|
|
||||||
|
class SourceCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
InheritanceCreateView,
|
||||||
|
):
|
||||||
|
"""Create new Source"""
|
||||||
|
|
||||||
|
model = Source
|
||||||
|
permission_required = "authentik_core.add_source"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:sources")
|
||||||
|
success_message = _("Successfully created Source")
|
||||||
|
|
||||||
|
|
||||||
|
class SourceUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
):
|
||||||
|
"""Update source"""
|
||||||
|
|
||||||
|
model = Source
|
||||||
|
permission_required = "authentik_core.change_source"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:sources")
|
||||||
|
success_message = _("Successfully updated Source")
|
||||||
|
|
||||||
|
|
||||||
|
class SourceDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete source"""
|
||||||
|
|
||||||
|
model = Source
|
||||||
|
permission_required = "authentik_core.delete_source"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:sources")
|
||||||
|
success_message = _("Successfully deleted Source")
|
|
@ -0,0 +1,79 @@
|
||||||
|
"""authentik Stage administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
InheritanceCreateView,
|
||||||
|
InheritanceListView,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.flows.models import Stage
|
||||||
|
|
||||||
|
|
||||||
|
class StageListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
InheritanceListView,
|
||||||
|
):
|
||||||
|
"""Show list of all stages"""
|
||||||
|
|
||||||
|
model = Stage
|
||||||
|
template_name = "administration/stage/list.html"
|
||||||
|
permission_required = "authentik_flows.view_stage"
|
||||||
|
ordering = "name"
|
||||||
|
search_fields = ["name"]
|
||||||
|
|
||||||
|
|
||||||
|
class StageCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
InheritanceCreateView,
|
||||||
|
):
|
||||||
|
"""Create new Stage"""
|
||||||
|
|
||||||
|
model = Stage
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
permission_required = "authentik_flows.add_stage"
|
||||||
|
|
||||||
|
success_url = reverse_lazy("authentik_admin:stages")
|
||||||
|
success_message = _("Successfully created Stage")
|
||||||
|
|
||||||
|
|
||||||
|
class StageUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
InheritanceUpdateView,
|
||||||
|
):
|
||||||
|
"""Update stage"""
|
||||||
|
|
||||||
|
model = Stage
|
||||||
|
permission_required = "authentik_flows.update_application"
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stages")
|
||||||
|
success_message = _("Successfully updated Stage")
|
||||||
|
|
||||||
|
|
||||||
|
class StageDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete stage"""
|
||||||
|
|
||||||
|
model = Stage
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
permission_required = "authentik_flows.delete_stage"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stages")
|
||||||
|
success_message = _("Successfully deleted Stage")
|
|
@ -0,0 +1,79 @@
|
||||||
|
"""authentik StageBinding administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.flows.forms import FlowStageBindingForm
|
||||||
|
from authentik.flows.models import FlowStageBinding
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
|
||||||
|
|
||||||
|
class StageBindingListView(
|
||||||
|
LoginRequiredMixin, PermissionListMixin, UserPaginateListMixin, ListView
|
||||||
|
):
|
||||||
|
"""Show list of all flows"""
|
||||||
|
|
||||||
|
model = FlowStageBinding
|
||||||
|
permission_required = "authentik_flows.view_flowstagebinding"
|
||||||
|
ordering = ["target", "order"]
|
||||||
|
template_name = "administration/stage_binding/list.html"
|
||||||
|
|
||||||
|
|
||||||
|
class StageBindingCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new StageBinding"""
|
||||||
|
|
||||||
|
model = FlowStageBinding
|
||||||
|
permission_required = "authentik_flows.add_flowstagebinding"
|
||||||
|
form_class = FlowStageBindingForm
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-bindings")
|
||||||
|
success_message = _("Successfully created StageBinding")
|
||||||
|
|
||||||
|
|
||||||
|
class StageBindingUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update FlowStageBinding"""
|
||||||
|
|
||||||
|
model = FlowStageBinding
|
||||||
|
permission_required = "authentik_flows.change_flowstagebinding"
|
||||||
|
form_class = FlowStageBindingForm
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-bindings")
|
||||||
|
success_message = _("Successfully updated StageBinding")
|
||||||
|
|
||||||
|
|
||||||
|
class StageBindingDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete FlowStageBinding"""
|
||||||
|
|
||||||
|
model = FlowStageBinding
|
||||||
|
permission_required = "authentik_flows.delete_flowstagebinding"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-bindings")
|
||||||
|
success_message = _("Successfully deleted FlowStageBinding")
|
|
@ -0,0 +1,76 @@
|
||||||
|
"""authentik Invitation administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.http import HttpResponseRedirect
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
from authentik.stages.invitation.forms import InvitationForm
|
||||||
|
from authentik.stages.invitation.models import Invitation
|
||||||
|
from authentik.stages.invitation.signals import invitation_created
|
||||||
|
|
||||||
|
|
||||||
|
class InvitationListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all invitations"""
|
||||||
|
|
||||||
|
model = Invitation
|
||||||
|
permission_required = "authentik_stages_invitation.view_invitation"
|
||||||
|
template_name = "administration/stage_invitation/list.html"
|
||||||
|
ordering = "-expires"
|
||||||
|
search_fields = ["created_by__username", "expires", "fixed_data"]
|
||||||
|
|
||||||
|
|
||||||
|
class InvitationCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new Invitation"""
|
||||||
|
|
||||||
|
model = Invitation
|
||||||
|
form_class = InvitationForm
|
||||||
|
permission_required = "authentik_stages_invitation.add_invitation"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-invitations")
|
||||||
|
success_message = _("Successfully created Invitation")
|
||||||
|
|
||||||
|
def form_valid(self, form):
|
||||||
|
obj = form.save(commit=False)
|
||||||
|
obj.created_by = self.request.user
|
||||||
|
obj.save()
|
||||||
|
invitation_created.send(sender=self, request=self.request, invitation=obj)
|
||||||
|
return HttpResponseRedirect(self.success_url)
|
||||||
|
|
||||||
|
|
||||||
|
class InvitationDeleteView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Delete invitation"""
|
||||||
|
|
||||||
|
model = Invitation
|
||||||
|
permission_required = "authentik_stages_invitation.delete_invitation"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-invitations")
|
||||||
|
success_message = _("Successfully deleted Invitation")
|
|
@ -0,0 +1,88 @@
|
||||||
|
"""authentik Prompt administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView, UpdateView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
from authentik.stages.prompt.forms import PromptAdminForm
|
||||||
|
from authentik.stages.prompt.models import Prompt
|
||||||
|
|
||||||
|
|
||||||
|
class PromptListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all prompts"""
|
||||||
|
|
||||||
|
model = Prompt
|
||||||
|
permission_required = "authentik_stages_prompt.view_prompt"
|
||||||
|
ordering = "order"
|
||||||
|
template_name = "administration/stage_prompt/list.html"
|
||||||
|
search_fields = [
|
||||||
|
"field_key",
|
||||||
|
"label",
|
||||||
|
"type",
|
||||||
|
"placeholder",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class PromptCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create new Prompt"""
|
||||||
|
|
||||||
|
model = Prompt
|
||||||
|
form_class = PromptAdminForm
|
||||||
|
permission_required = "authentik_stages_prompt.add_prompt"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-prompts")
|
||||||
|
success_message = _("Successfully created Prompt")
|
||||||
|
|
||||||
|
|
||||||
|
class PromptUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update prompt"""
|
||||||
|
|
||||||
|
model = Prompt
|
||||||
|
form_class = PromptAdminForm
|
||||||
|
permission_required = "authentik_stages_prompt.change_prompt"
|
||||||
|
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-prompts")
|
||||||
|
success_message = _("Successfully updated Prompt")
|
||||||
|
|
||||||
|
|
||||||
|
class PromptDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete prompt"""
|
||||||
|
|
||||||
|
model = Prompt
|
||||||
|
permission_required = "authentik_stages_prompt.delete_prompt"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:stage-prompts")
|
||||||
|
success_message = _("Successfully deleted Prompt")
|
|
@ -0,0 +1,23 @@
|
||||||
|
"""authentik Tasks List"""
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
|
from django.views.generic.base import TemplateView
|
||||||
|
|
||||||
|
from authentik.admin.mixins import AdminRequiredMixin
|
||||||
|
from authentik.lib.tasks import TaskInfo, TaskResultStatus
|
||||||
|
|
||||||
|
|
||||||
|
class TaskListView(AdminRequiredMixin, TemplateView):
|
||||||
|
"""Show list of all background tasks"""
|
||||||
|
|
||||||
|
template_name = "administration/task/list.html"
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
|
||||||
|
kwargs = super().get_context_data(**kwargs)
|
||||||
|
kwargs["object_list"] = sorted(
|
||||||
|
TaskInfo.all().values(), key=lambda x: x.task_name
|
||||||
|
)
|
||||||
|
kwargs["task_successful"] = TaskResultStatus.SUCCESSFUL
|
||||||
|
kwargs["task_warning"] = TaskResultStatus.WARNING
|
||||||
|
kwargs["task_error"] = TaskResultStatus.ERROR
|
||||||
|
return kwargs
|
|
@ -0,0 +1,45 @@
|
||||||
|
"""authentik Token administration"""
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.urls import reverse_lazy
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import ListView
|
||||||
|
from guardian.mixins import PermissionListMixin, PermissionRequiredMixin
|
||||||
|
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.core.models import Token
|
||||||
|
|
||||||
|
|
||||||
|
class TokenListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all tokens"""
|
||||||
|
|
||||||
|
model = Token
|
||||||
|
permission_required = "authentik_core.view_token"
|
||||||
|
ordering = "expires"
|
||||||
|
template_name = "administration/token/list.html"
|
||||||
|
search_fields = [
|
||||||
|
"identifier",
|
||||||
|
"intent",
|
||||||
|
"user__username",
|
||||||
|
"description",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class TokenDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete token"""
|
||||||
|
|
||||||
|
model = Token
|
||||||
|
permission_required = "authentik_core.delete_token"
|
||||||
|
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:tokens")
|
||||||
|
success_message = _("Successfully deleted Token")
|
|
@ -0,0 +1,168 @@
|
||||||
|
"""authentik User administration"""
|
||||||
|
from django.contrib import messages
|
||||||
|
from django.contrib.auth.mixins import LoginRequiredMixin
|
||||||
|
from django.contrib.auth.mixins import (
|
||||||
|
PermissionRequiredMixin as DjangoPermissionRequiredMixin,
|
||||||
|
)
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.http import HttpRequest, HttpResponse
|
||||||
|
from django.http.response import HttpResponseRedirect
|
||||||
|
from django.shortcuts import redirect
|
||||||
|
from django.urls import reverse, reverse_lazy
|
||||||
|
from django.utils.http import urlencode
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.generic import DetailView, ListView, UpdateView
|
||||||
|
from guardian.mixins import (
|
||||||
|
PermissionListMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
get_anonymous_user,
|
||||||
|
)
|
||||||
|
|
||||||
|
from authentik.admin.forms.users import UserForm
|
||||||
|
from authentik.admin.views.utils import (
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
DeleteMessageView,
|
||||||
|
SearchListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
)
|
||||||
|
from authentik.core.models import Token, User
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
|
||||||
|
|
||||||
|
class UserListView(
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionListMixin,
|
||||||
|
UserPaginateListMixin,
|
||||||
|
SearchListMixin,
|
||||||
|
ListView,
|
||||||
|
):
|
||||||
|
"""Show list of all users"""
|
||||||
|
|
||||||
|
model = User
|
||||||
|
permission_required = "authentik_core.view_user"
|
||||||
|
ordering = "username"
|
||||||
|
template_name = "administration/user/list.html"
|
||||||
|
search_fields = ["username", "name", "attributes"]
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
return super().get_queryset().exclude(pk=get_anonymous_user().pk)
|
||||||
|
|
||||||
|
|
||||||
|
class UserCreateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
DjangoPermissionRequiredMixin,
|
||||||
|
CreateAssignPermView,
|
||||||
|
):
|
||||||
|
"""Create user"""
|
||||||
|
|
||||||
|
model = User
|
||||||
|
form_class = UserForm
|
||||||
|
permission_required = "authentik_core.add_user"
|
||||||
|
|
||||||
|
template_name = "generic/create.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:users")
|
||||||
|
success_message = _("Successfully created User")
|
||||||
|
|
||||||
|
|
||||||
|
class UserUpdateView(
|
||||||
|
SuccessMessageMixin,
|
||||||
|
BackSuccessUrlMixin,
|
||||||
|
LoginRequiredMixin,
|
||||||
|
PermissionRequiredMixin,
|
||||||
|
UpdateView,
|
||||||
|
):
|
||||||
|
"""Update user"""
|
||||||
|
|
||||||
|
model = User
|
||||||
|
form_class = UserForm
|
||||||
|
permission_required = "authentik_core.change_user"
|
||||||
|
|
||||||
|
# By default the object's name is user which is used by other checks
|
||||||
|
context_object_name = "object"
|
||||||
|
template_name = "generic/update.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:users")
|
||||||
|
success_message = _("Successfully updated User")
|
||||||
|
|
||||||
|
|
||||||
|
class UserDeleteView(LoginRequiredMixin, PermissionRequiredMixin, DeleteMessageView):
|
||||||
|
"""Delete user"""
|
||||||
|
|
||||||
|
model = User
|
||||||
|
permission_required = "authentik_core.delete_user"
|
||||||
|
|
||||||
|
# By default the object's name is user which is used by other checks
|
||||||
|
context_object_name = "object"
|
||||||
|
template_name = "generic/delete.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:users")
|
||||||
|
success_message = _("Successfully deleted User")
|
||||||
|
|
||||||
|
|
||||||
|
class UserDisableView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, BackSuccessUrlMixin, DeleteMessageView
|
||||||
|
):
|
||||||
|
"""Disable user"""
|
||||||
|
|
||||||
|
object: User
|
||||||
|
|
||||||
|
model = User
|
||||||
|
permission_required = "authentik_core.update_user"
|
||||||
|
|
||||||
|
# By default the object's name is user which is used by other checks
|
||||||
|
context_object_name = "object"
|
||||||
|
template_name = "administration/user/disable.html"
|
||||||
|
success_url = reverse_lazy("authentik_admin:users")
|
||||||
|
success_message = _("Successfully disabled User")
|
||||||
|
|
||||||
|
def delete(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
|
||||||
|
self.object: User = self.get_object()
|
||||||
|
success_url = self.get_success_url()
|
||||||
|
self.object.is_active = False
|
||||||
|
self.object.save()
|
||||||
|
return HttpResponseRedirect(success_url)
|
||||||
|
|
||||||
|
|
||||||
|
class UserEnableView(
|
||||||
|
LoginRequiredMixin, PermissionRequiredMixin, BackSuccessUrlMixin, DetailView
|
||||||
|
):
|
||||||
|
"""Enable user"""
|
||||||
|
|
||||||
|
object: User
|
||||||
|
|
||||||
|
model = User
|
||||||
|
permission_required = "authentik_core.update_user"
|
||||||
|
|
||||||
|
# By default the object's name is user which is used by other checks
|
||||||
|
context_object_name = "object"
|
||||||
|
success_url = reverse_lazy("authentik_admin:users")
|
||||||
|
success_message = _("Successfully enabled User")
|
||||||
|
|
||||||
|
def get(self, request: HttpRequest, *args, **kwargs):
|
||||||
|
self.object: User = self.get_object()
|
||||||
|
success_url = self.get_success_url()
|
||||||
|
self.object.is_active = True
|
||||||
|
self.object.save()
|
||||||
|
return HttpResponseRedirect(success_url)
|
||||||
|
|
||||||
|
|
||||||
|
class UserPasswordResetView(LoginRequiredMixin, PermissionRequiredMixin, DetailView):
|
||||||
|
"""Get Password reset link for user"""
|
||||||
|
|
||||||
|
model = User
|
||||||
|
permission_required = "authentik_core.reset_user_password"
|
||||||
|
|
||||||
|
def get(self, request: HttpRequest, *args, **kwargs) -> HttpResponse:
|
||||||
|
"""Create token for user and return link"""
|
||||||
|
super().get(request, *args, **kwargs)
|
||||||
|
token, __ = Token.objects.get_or_create(
|
||||||
|
identifier="password-reset-temp", user=self.object
|
||||||
|
)
|
||||||
|
querystring = urlencode({"token": token.key})
|
||||||
|
link = request.build_absolute_uri(
|
||||||
|
reverse("authentik_flows:default-recovery") + f"?{querystring}"
|
||||||
|
)
|
||||||
|
messages.success(
|
||||||
|
request, _("Password reset link: <pre>%(link)s</pre>" % {"link": link})
|
||||||
|
)
|
||||||
|
return redirect("authentik_admin:users")
|
|
@ -0,0 +1,124 @@
|
||||||
|
"""authentik admin util views"""
|
||||||
|
from typing import Any, Dict, List, Optional
|
||||||
|
from urllib.parse import urlparse
|
||||||
|
|
||||||
|
from django.contrib import messages
|
||||||
|
from django.contrib.messages.views import SuccessMessageMixin
|
||||||
|
from django.contrib.postgres.search import SearchQuery, SearchVector
|
||||||
|
from django.db.models import QuerySet
|
||||||
|
from django.http import Http404
|
||||||
|
from django.http.request import HttpRequest
|
||||||
|
from django.views.generic import DeleteView, ListView, UpdateView
|
||||||
|
from django.views.generic.list import MultipleObjectMixin
|
||||||
|
|
||||||
|
from authentik.lib.utils.reflection import all_subclasses
|
||||||
|
from authentik.lib.views import CreateAssignPermView
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteMessageView(SuccessMessageMixin, DeleteView):
|
||||||
|
"""DeleteView which shows `self.success_message` on successful deletion"""
|
||||||
|
|
||||||
|
def delete(self, request, *args, **kwargs):
|
||||||
|
messages.success(self.request, self.success_message)
|
||||||
|
return super().delete(request, *args, **kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
class InheritanceListView(ListView):
|
||||||
|
"""ListView for objects using InheritanceManager"""
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs):
|
||||||
|
kwargs["types"] = {x.__name__: x for x in all_subclasses(self.model)}
|
||||||
|
return super().get_context_data(**kwargs)
|
||||||
|
|
||||||
|
def get_queryset(self):
|
||||||
|
return super().get_queryset().select_subclasses()
|
||||||
|
|
||||||
|
|
||||||
|
class SearchListMixin(MultipleObjectMixin):
|
||||||
|
"""Accept search query using `search` querystring parameter. Requires self.search_fields,
|
||||||
|
a list of all fields to search. Can contain special lookups like __icontains"""
|
||||||
|
|
||||||
|
search_fields: List[str]
|
||||||
|
|
||||||
|
def get_queryset(self) -> QuerySet:
|
||||||
|
queryset = super().get_queryset()
|
||||||
|
if "search" in self.request.GET:
|
||||||
|
raw_query = self.request.GET["search"]
|
||||||
|
if raw_query == "":
|
||||||
|
# Empty query, don't search at all
|
||||||
|
return queryset
|
||||||
|
search = SearchQuery(raw_query, search_type="websearch")
|
||||||
|
return queryset.annotate(search=SearchVector(*self.search_fields)).filter(
|
||||||
|
search=search
|
||||||
|
)
|
||||||
|
return queryset
|
||||||
|
|
||||||
|
|
||||||
|
class InheritanceCreateView(CreateAssignPermView):
|
||||||
|
"""CreateView for objects using InheritanceManager"""
|
||||||
|
|
||||||
|
def get_form_class(self):
|
||||||
|
provider_type = self.request.GET.get("type")
|
||||||
|
try:
|
||||||
|
model = next(
|
||||||
|
x for x in all_subclasses(self.model) if x.__name__ == provider_type
|
||||||
|
)
|
||||||
|
except StopIteration as exc:
|
||||||
|
raise Http404 from exc
|
||||||
|
return model().form
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
|
||||||
|
kwargs = super().get_context_data(**kwargs)
|
||||||
|
form_cls = self.get_form_class()
|
||||||
|
if hasattr(form_cls, "template_name"):
|
||||||
|
kwargs["base_template"] = form_cls.template_name
|
||||||
|
return kwargs
|
||||||
|
|
||||||
|
|
||||||
|
class InheritanceUpdateView(UpdateView):
|
||||||
|
"""UpdateView for objects using InheritanceManager"""
|
||||||
|
|
||||||
|
def get_context_data(self, **kwargs: Any) -> Dict[str, Any]:
|
||||||
|
kwargs = super().get_context_data(**kwargs)
|
||||||
|
form_cls = self.get_form_class()
|
||||||
|
if hasattr(form_cls, "template_name"):
|
||||||
|
kwargs["base_template"] = form_cls.template_name
|
||||||
|
return kwargs
|
||||||
|
|
||||||
|
def get_form_class(self):
|
||||||
|
return self.get_object().form
|
||||||
|
|
||||||
|
def get_object(self, queryset=None):
|
||||||
|
return (
|
||||||
|
self.model.objects.filter(pk=self.kwargs.get("pk"))
|
||||||
|
.select_subclasses()
|
||||||
|
.first()
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class BackSuccessUrlMixin:
|
||||||
|
"""Checks if a relative URL has been given as ?back param, and redirect to it. Otherwise
|
||||||
|
default to self.success_url."""
|
||||||
|
|
||||||
|
request: HttpRequest
|
||||||
|
|
||||||
|
success_url: Optional[str]
|
||||||
|
|
||||||
|
def get_success_url(self) -> str:
|
||||||
|
"""get_success_url from FormMixin"""
|
||||||
|
back_param = self.request.GET.get("back")
|
||||||
|
if back_param:
|
||||||
|
if not bool(urlparse(back_param).netloc):
|
||||||
|
return back_param
|
||||||
|
return str(self.success_url)
|
||||||
|
|
||||||
|
|
||||||
|
class UserPaginateListMixin:
|
||||||
|
"""Get paginate_by value from user's attributes, defaulting to 15"""
|
||||||
|
|
||||||
|
request: HttpRequest
|
||||||
|
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def get_paginate_by(self, queryset: QuerySet) -> int:
|
||||||
|
"""get_paginate_by Function of ListView"""
|
||||||
|
return self.request.user.attributes.get("paginate_by", 15)
|
|
@ -0,0 +1,12 @@
|
||||||
|
"""authentik API AppConfig"""
|
||||||
|
|
||||||
|
from django.apps import AppConfig
|
||||||
|
|
||||||
|
|
||||||
|
class AuthentikAPIConfig(AppConfig):
|
||||||
|
"""authentik API Config"""
|
||||||
|
|
||||||
|
name = "authentik.api"
|
||||||
|
label = "authentik_api"
|
||||||
|
mountpoint = "api/"
|
||||||
|
verbose_name = "authentik API"
|
|
@ -0,0 +1,57 @@
|
||||||
|
"""API Authentication"""
|
||||||
|
from base64 import b64decode
|
||||||
|
from typing import Any, Optional, Tuple, Union
|
||||||
|
|
||||||
|
from rest_framework.authentication import BaseAuthentication, get_authorization_header
|
||||||
|
from rest_framework.request import Request
|
||||||
|
from structlog import get_logger
|
||||||
|
|
||||||
|
from authentik.core.models import Token, TokenIntents, User
|
||||||
|
|
||||||
|
LOGGER = get_logger()
|
||||||
|
|
||||||
|
|
||||||
|
def token_from_header(raw_header: bytes) -> Optional[Token]:
|
||||||
|
"""raw_header in the Format of `Basic dGVzdDp0ZXN0`"""
|
||||||
|
auth_credentials = raw_header.decode()
|
||||||
|
# Accept headers with Type format and without
|
||||||
|
if " " in auth_credentials:
|
||||||
|
auth_type, auth_credentials = auth_credentials.split()
|
||||||
|
if auth_type.lower() != "basic":
|
||||||
|
LOGGER.debug(
|
||||||
|
"Unsupported authentication type, denying", type=auth_type.lower()
|
||||||
|
)
|
||||||
|
return None
|
||||||
|
try:
|
||||||
|
auth_credentials = b64decode(auth_credentials.encode()).decode()
|
||||||
|
except UnicodeDecodeError:
|
||||||
|
return None
|
||||||
|
# Accept credentials with username and without
|
||||||
|
if ":" in auth_credentials:
|
||||||
|
_, password = auth_credentials.split(":")
|
||||||
|
else:
|
||||||
|
password = auth_credentials
|
||||||
|
if password == "":
|
||||||
|
return None
|
||||||
|
tokens = Token.filter_not_expired(key=password, intent=TokenIntents.INTENT_API)
|
||||||
|
if not tokens.exists():
|
||||||
|
LOGGER.debug("Token not found")
|
||||||
|
return None
|
||||||
|
return tokens.first()
|
||||||
|
|
||||||
|
|
||||||
|
class AuthentikTokenAuthentication(BaseAuthentication):
|
||||||
|
"""Token-based authentication using HTTP Basic authentication"""
|
||||||
|
|
||||||
|
def authenticate(self, request: Request) -> Union[Tuple[User, Any], None]:
|
||||||
|
"""Token-based authentication using HTTP Basic authentication"""
|
||||||
|
auth = get_authorization_header(request)
|
||||||
|
|
||||||
|
token = token_from_header(auth)
|
||||||
|
if not token:
|
||||||
|
return None
|
||||||
|
|
||||||
|
return (token.user, None)
|
||||||
|
|
||||||
|
def authenticate_header(self, request: Request) -> str:
|
||||||
|
return 'Basic realm="authentik"'
|
|
@ -0,0 +1,7 @@
|
||||||
|
{% extends "rest_framework/base.html" %}
|
||||||
|
|
||||||
|
{% block branding %}
|
||||||
|
<span class='navbar-brand'>
|
||||||
|
authentik
|
||||||
|
</span>
|
||||||
|
{% endblock %}
|
|
@ -0,0 +1,8 @@
|
||||||
|
"""authentik api urls"""
|
||||||
|
from django.urls import include, path
|
||||||
|
|
||||||
|
from authentik.api.v2.urls import urlpatterns as v2_urls
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
path("v2beta/", include(v2_urls)),
|
||||||
|
]
|
|
@ -0,0 +1,46 @@
|
||||||
|
"""core Configs API"""
|
||||||
|
from drf_yasg2.utils import swagger_auto_schema
|
||||||
|
from rest_framework.permissions import AllowAny
|
||||||
|
from rest_framework.request import Request
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework.serializers import ReadOnlyField, Serializer
|
||||||
|
from rest_framework.viewsets import ViewSet
|
||||||
|
|
||||||
|
from authentik.lib.config import CONFIG
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigSerializer(Serializer):
|
||||||
|
"""Serialize authentik Config into DRF Object"""
|
||||||
|
|
||||||
|
branding_logo = ReadOnlyField()
|
||||||
|
branding_title = ReadOnlyField()
|
||||||
|
|
||||||
|
error_reporting_enabled = ReadOnlyField()
|
||||||
|
error_reporting_environment = ReadOnlyField()
|
||||||
|
error_reporting_send_pii = ReadOnlyField()
|
||||||
|
|
||||||
|
def create(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def update(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
|
class ConfigsViewSet(ViewSet):
|
||||||
|
"""Read-only view set that returns the current session's Configs"""
|
||||||
|
|
||||||
|
permission_classes = [AllowAny]
|
||||||
|
|
||||||
|
@swagger_auto_schema(responses={200: ConfigSerializer(many=True)})
|
||||||
|
def list(self, request: Request) -> Response:
|
||||||
|
"""Retrive public configuration options"""
|
||||||
|
config = ConfigSerializer(
|
||||||
|
{
|
||||||
|
"branding_logo": CONFIG.y("authentik.branding.logo"),
|
||||||
|
"branding_title": CONFIG.y("authentik.branding.title"),
|
||||||
|
"error_reporting_enabled": CONFIG.y("error_reporting.enabled"),
|
||||||
|
"error_reporting_environment": CONFIG.y("error_reporting.environment"),
|
||||||
|
"error_reporting_send_pii": CONFIG.y("error_reporting.send_pii"),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
return Response(config.data)
|
|
@ -0,0 +1,160 @@
|
||||||
|
"""api v2 urls"""
|
||||||
|
from django.urls import path, re_path
|
||||||
|
from drf_yasg2 import openapi
|
||||||
|
from drf_yasg2.views import get_schema_view
|
||||||
|
from rest_framework import routers
|
||||||
|
from rest_framework.permissions import AllowAny
|
||||||
|
|
||||||
|
from authentik.admin.api.overview import AdministrationOverviewViewSet
|
||||||
|
from authentik.admin.api.overview_metrics import AdministrationMetricsViewSet
|
||||||
|
from authentik.admin.api.tasks import TaskViewSet
|
||||||
|
from authentik.api.v2.config import ConfigsViewSet
|
||||||
|
from authentik.api.v2.messages import MessagesViewSet
|
||||||
|
from authentik.audit.api import EventViewSet
|
||||||
|
from authentik.core.api.applications import ApplicationViewSet
|
||||||
|
from authentik.core.api.groups import GroupViewSet
|
||||||
|
from authentik.core.api.propertymappings import PropertyMappingViewSet
|
||||||
|
from authentik.core.api.providers import ProviderViewSet
|
||||||
|
from authentik.core.api.sources import SourceViewSet
|
||||||
|
from authentik.core.api.tokens import TokenViewSet
|
||||||
|
from authentik.core.api.users import UserViewSet
|
||||||
|
from authentik.crypto.api import CertificateKeyPairViewSet
|
||||||
|
from authentik.flows.api import FlowStageBindingViewSet, FlowViewSet, StageViewSet
|
||||||
|
from authentik.outposts.api import (
|
||||||
|
DockerServiceConnectionViewSet,
|
||||||
|
KubernetesServiceConnectionViewSet,
|
||||||
|
OutpostViewSet,
|
||||||
|
)
|
||||||
|
from authentik.policies.api import PolicyBindingViewSet, PolicyViewSet
|
||||||
|
from authentik.policies.dummy.api import DummyPolicyViewSet
|
||||||
|
from authentik.policies.expiry.api import PasswordExpiryPolicyViewSet
|
||||||
|
from authentik.policies.expression.api import ExpressionPolicyViewSet
|
||||||
|
from authentik.policies.group_membership.api import GroupMembershipPolicyViewSet
|
||||||
|
from authentik.policies.hibp.api import HaveIBeenPwendPolicyViewSet
|
||||||
|
from authentik.policies.password.api import PasswordPolicyViewSet
|
||||||
|
from authentik.policies.reputation.api import ReputationPolicyViewSet
|
||||||
|
from authentik.providers.oauth2.api import OAuth2ProviderViewSet, ScopeMappingViewSet
|
||||||
|
from authentik.providers.proxy.api import (
|
||||||
|
ProxyOutpostConfigViewSet,
|
||||||
|
ProxyProviderViewSet,
|
||||||
|
)
|
||||||
|
from authentik.providers.saml.api import SAMLPropertyMappingViewSet, SAMLProviderViewSet
|
||||||
|
from authentik.sources.ldap.api import LDAPPropertyMappingViewSet, LDAPSourceViewSet
|
||||||
|
from authentik.sources.oauth.api import OAuthSourceViewSet
|
||||||
|
from authentik.sources.saml.api import SAMLSourceViewSet
|
||||||
|
from authentik.stages.captcha.api import CaptchaStageViewSet
|
||||||
|
from authentik.stages.consent.api import ConsentStageViewSet
|
||||||
|
from authentik.stages.dummy.api import DummyStageViewSet
|
||||||
|
from authentik.stages.email.api import EmailStageViewSet
|
||||||
|
from authentik.stages.identification.api import IdentificationStageViewSet
|
||||||
|
from authentik.stages.invitation.api import InvitationStageViewSet, InvitationViewSet
|
||||||
|
from authentik.stages.otp_static.api import OTPStaticStageViewSet
|
||||||
|
from authentik.stages.otp_time.api import OTPTimeStageViewSet
|
||||||
|
from authentik.stages.otp_validate.api import OTPValidateStageViewSet
|
||||||
|
from authentik.stages.password.api import PasswordStageViewSet
|
||||||
|
from authentik.stages.prompt.api import PromptStageViewSet, PromptViewSet
|
||||||
|
from authentik.stages.user_delete.api import UserDeleteStageViewSet
|
||||||
|
from authentik.stages.user_login.api import UserLoginStageViewSet
|
||||||
|
from authentik.stages.user_logout.api import UserLogoutStageViewSet
|
||||||
|
from authentik.stages.user_write.api import UserWriteStageViewSet
|
||||||
|
|
||||||
|
router = routers.DefaultRouter()
|
||||||
|
|
||||||
|
router.register("root/messages", MessagesViewSet, basename="messages")
|
||||||
|
router.register("root/config", ConfigsViewSet, basename="configs")
|
||||||
|
|
||||||
|
router.register(
|
||||||
|
"admin/overview", AdministrationOverviewViewSet, basename="admin_overview"
|
||||||
|
)
|
||||||
|
router.register("admin/metrics", AdministrationMetricsViewSet, basename="admin_metrics")
|
||||||
|
router.register("admin/system_tasks", TaskViewSet, basename="admin_system_tasks")
|
||||||
|
|
||||||
|
router.register("core/applications", ApplicationViewSet)
|
||||||
|
router.register("core/groups", GroupViewSet)
|
||||||
|
router.register("core/users", UserViewSet)
|
||||||
|
router.register("core/tokens", TokenViewSet)
|
||||||
|
|
||||||
|
router.register("outposts/outposts", OutpostViewSet)
|
||||||
|
router.register("outposts/service_connections/docker", DockerServiceConnectionViewSet)
|
||||||
|
router.register(
|
||||||
|
"outposts/service_connections/kubernetes", KubernetesServiceConnectionViewSet
|
||||||
|
)
|
||||||
|
router.register("outposts/proxy", ProxyOutpostConfigViewSet)
|
||||||
|
|
||||||
|
router.register("flows/instances", FlowViewSet)
|
||||||
|
router.register("flows/bindings", FlowStageBindingViewSet)
|
||||||
|
|
||||||
|
router.register("crypto/certificatekeypairs", CertificateKeyPairViewSet)
|
||||||
|
|
||||||
|
router.register("audit/events", EventViewSet)
|
||||||
|
|
||||||
|
router.register("sources/all", SourceViewSet)
|
||||||
|
router.register("sources/ldap", LDAPSourceViewSet)
|
||||||
|
router.register("sources/saml", SAMLSourceViewSet)
|
||||||
|
router.register("sources/oauth", OAuthSourceViewSet)
|
||||||
|
|
||||||
|
router.register("policies/all", PolicyViewSet)
|
||||||
|
router.register("policies/bindings", PolicyBindingViewSet)
|
||||||
|
router.register("policies/expression", ExpressionPolicyViewSet)
|
||||||
|
router.register("policies/group_membership", GroupMembershipPolicyViewSet)
|
||||||
|
router.register("policies/haveibeenpwned", HaveIBeenPwendPolicyViewSet)
|
||||||
|
router.register("policies/password_expiry", PasswordExpiryPolicyViewSet)
|
||||||
|
router.register("policies/password", PasswordPolicyViewSet)
|
||||||
|
router.register("policies/reputation", ReputationPolicyViewSet)
|
||||||
|
|
||||||
|
router.register("providers/all", ProviderViewSet)
|
||||||
|
router.register("providers/proxy", ProxyProviderViewSet)
|
||||||
|
router.register("providers/oauth2", OAuth2ProviderViewSet)
|
||||||
|
router.register("providers/saml", SAMLProviderViewSet)
|
||||||
|
|
||||||
|
router.register("propertymappings/all", PropertyMappingViewSet)
|
||||||
|
router.register("propertymappings/ldap", LDAPPropertyMappingViewSet)
|
||||||
|
router.register("propertymappings/saml", SAMLPropertyMappingViewSet)
|
||||||
|
router.register("propertymappings/scope", ScopeMappingViewSet)
|
||||||
|
|
||||||
|
router.register("stages/all", StageViewSet)
|
||||||
|
router.register("stages/captcha", CaptchaStageViewSet)
|
||||||
|
router.register("stages/consent", ConsentStageViewSet)
|
||||||
|
router.register("stages/email", EmailStageViewSet)
|
||||||
|
router.register("stages/identification", IdentificationStageViewSet)
|
||||||
|
router.register("stages/invitation", InvitationStageViewSet)
|
||||||
|
router.register("stages/invitation/invitations", InvitationViewSet)
|
||||||
|
router.register("stages/otp_static", OTPStaticStageViewSet)
|
||||||
|
router.register("stages/otp_time", OTPTimeStageViewSet)
|
||||||
|
router.register("stages/otp_validate", OTPValidateStageViewSet)
|
||||||
|
router.register("stages/password", PasswordStageViewSet)
|
||||||
|
router.register("stages/prompt/prompts", PromptViewSet)
|
||||||
|
router.register("stages/prompt/stages", PromptStageViewSet)
|
||||||
|
router.register("stages/user_delete", UserDeleteStageViewSet)
|
||||||
|
router.register("stages/user_login", UserLoginStageViewSet)
|
||||||
|
router.register("stages/user_logout", UserLogoutStageViewSet)
|
||||||
|
router.register("stages/user_write", UserWriteStageViewSet)
|
||||||
|
|
||||||
|
router.register("stages/dummy", DummyStageViewSet)
|
||||||
|
router.register("policies/dummy", DummyPolicyViewSet)
|
||||||
|
|
||||||
|
info = openapi.Info(
|
||||||
|
title="authentik API",
|
||||||
|
default_version="v2",
|
||||||
|
contact=openapi.Contact(email="hello@beryju.org"),
|
||||||
|
license=openapi.License(name="MIT License"),
|
||||||
|
)
|
||||||
|
SchemaView = get_schema_view(
|
||||||
|
info,
|
||||||
|
public=True,
|
||||||
|
permission_classes=(AllowAny,),
|
||||||
|
)
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
re_path(
|
||||||
|
r"^swagger(?P<format>\.json|\.yaml)$",
|
||||||
|
SchemaView.without_ui(cache_timeout=0),
|
||||||
|
name="schema-json",
|
||||||
|
),
|
||||||
|
path(
|
||||||
|
"swagger/",
|
||||||
|
SchemaView.with_ui("swagger", cache_timeout=0),
|
||||||
|
name="schema-swagger-ui",
|
||||||
|
),
|
||||||
|
path("redoc/", SchemaView.with_ui("redoc", cache_timeout=0), name="schema-redoc"),
|
||||||
|
] + router.urls
|
|
@ -0,0 +1,70 @@
|
||||||
|
"""Audit API Views"""
|
||||||
|
from django.db.models.aggregates import Count
|
||||||
|
from django.db.models.fields.json import KeyTextTransform
|
||||||
|
from drf_yasg2.utils import swagger_auto_schema
|
||||||
|
from rest_framework.decorators import action
|
||||||
|
from rest_framework.fields import DictField, IntegerField
|
||||||
|
from rest_framework.request import Request
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework.serializers import ModelSerializer, Serializer
|
||||||
|
from rest_framework.viewsets import ReadOnlyModelViewSet
|
||||||
|
|
||||||
|
from authentik.audit.models import Event, EventAction
|
||||||
|
|
||||||
|
|
||||||
|
class EventSerializer(ModelSerializer):
|
||||||
|
"""Event Serializer"""
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
|
||||||
|
model = Event
|
||||||
|
fields = [
|
||||||
|
"pk",
|
||||||
|
"user",
|
||||||
|
"action",
|
||||||
|
"app",
|
||||||
|
"context",
|
||||||
|
"client_ip",
|
||||||
|
"created",
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class EventTopPerUserSerialier(Serializer):
|
||||||
|
"""Response object of Event's top_per_user"""
|
||||||
|
|
||||||
|
application = DictField()
|
||||||
|
counted_events = IntegerField()
|
||||||
|
unique_users = IntegerField()
|
||||||
|
|
||||||
|
def create(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
def update(self, request: Request) -> Response:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
|
||||||
|
class EventViewSet(ReadOnlyModelViewSet):
|
||||||
|
"""Event Read-Only Viewset"""
|
||||||
|
|
||||||
|
queryset = Event.objects.all()
|
||||||
|
serializer_class = EventSerializer
|
||||||
|
|
||||||
|
@swagger_auto_schema(
|
||||||
|
method="GET", responses={200: EventTopPerUserSerialier(many=True)}
|
||||||
|
)
|
||||||
|
@action(detail=False, methods=["GET"])
|
||||||
|
def top_per_user(self, request: Request):
|
||||||
|
"""Get the top_n events grouped by user count"""
|
||||||
|
filtered_action = request.query_params.get("filter_action", EventAction.LOGIN)
|
||||||
|
top_n = request.query_params.get("top_n", 15)
|
||||||
|
return Response(
|
||||||
|
Event.objects.filter(action=filtered_action)
|
||||||
|
.exclude(context__authorized_application=None)
|
||||||
|
.annotate(application=KeyTextTransform("authorized_application", "context"))
|
||||||
|
.annotate(user_pk=KeyTextTransform("pk", "user"))
|
||||||
|
.values("application")
|
||||||
|
.annotate(counted_events=Count("application"))
|
||||||
|
.annotate(unique_users=Count("user_pk", distinct=True))
|
||||||
|
.values("unique_users", "application", "counted_events")
|
||||||
|
.order_by("-counted_events")[:top_n]
|
||||||
|
)
|
|
@ -0,0 +1,16 @@
|
||||||
|
"""authentik audit app"""
|
||||||
|
from importlib import import_module
|
||||||
|
|
||||||
|
from django.apps import AppConfig
|
||||||
|
|
||||||
|
|
||||||
|
class AuthentikAuditConfig(AppConfig):
|
||||||
|
"""authentik audit app"""
|
||||||
|
|
||||||
|
name = "authentik.audit"
|
||||||
|
label = "authentik_audit"
|
||||||
|
verbose_name = "authentik Audit"
|
||||||
|
mountpoint = "audit/"
|
||||||
|
|
||||||
|
def ready(self):
|
||||||
|
import_module("authentik.audit.signals")
|
|
@ -0,0 +1,85 @@
|
||||||
|
"""Audit middleware"""
|
||||||
|
from functools import partial
|
||||||
|
from typing import Callable
|
||||||
|
|
||||||
|
from django.contrib.auth.models import User
|
||||||
|
from django.db.models import Model
|
||||||
|
from django.db.models.signals import post_save, pre_delete
|
||||||
|
from django.http import HttpRequest, HttpResponse
|
||||||
|
|
||||||
|
from authentik.audit.models import Event, EventAction, model_to_dict
|
||||||
|
from authentik.audit.signals import EventNewThread
|
||||||
|
from authentik.core.middleware import LOCAL
|
||||||
|
|
||||||
|
|
||||||
|
class AuditMiddleware:
|
||||||
|
"""Register handlers for duration of request-response that log creation/update/deletion
|
||||||
|
of models"""
|
||||||
|
|
||||||
|
get_response: Callable[[HttpRequest], HttpResponse]
|
||||||
|
|
||||||
|
def __init__(self, get_response: Callable[[HttpRequest], HttpResponse]):
|
||||||
|
self.get_response = get_response
|
||||||
|
|
||||||
|
def __call__(self, request: HttpRequest) -> HttpResponse:
|
||||||
|
# Connect signal for automatic logging
|
||||||
|
if hasattr(request, "user") and getattr(
|
||||||
|
request.user, "is_authenticated", False
|
||||||
|
):
|
||||||
|
post_save_handler = partial(
|
||||||
|
self.post_save_handler, user=request.user, request=request
|
||||||
|
)
|
||||||
|
pre_delete_handler = partial(
|
||||||
|
self.pre_delete_handler, user=request.user, request=request
|
||||||
|
)
|
||||||
|
post_save.connect(
|
||||||
|
post_save_handler,
|
||||||
|
dispatch_uid=LOCAL.authentik["request_id"],
|
||||||
|
weak=False,
|
||||||
|
)
|
||||||
|
pre_delete.connect(
|
||||||
|
pre_delete_handler,
|
||||||
|
dispatch_uid=LOCAL.authentik["request_id"],
|
||||||
|
weak=False,
|
||||||
|
)
|
||||||
|
|
||||||
|
response = self.get_response(request)
|
||||||
|
|
||||||
|
post_save.disconnect(dispatch_uid=LOCAL.authentik["request_id"])
|
||||||
|
pre_delete.disconnect(dispatch_uid=LOCAL.authentik["request_id"])
|
||||||
|
|
||||||
|
return response
|
||||||
|
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def process_exception(self, request: HttpRequest, exception: Exception):
|
||||||
|
"""Unregister handlers in case of exception"""
|
||||||
|
post_save.disconnect(dispatch_uid=LOCAL.authentik["request_id"])
|
||||||
|
pre_delete.disconnect(dispatch_uid=LOCAL.authentik["request_id"])
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def post_save_handler(
|
||||||
|
user: User, request: HttpRequest, sender, instance: Model, created: bool, **_
|
||||||
|
):
|
||||||
|
"""Signal handler for all object's post_save"""
|
||||||
|
if isinstance(instance, Event):
|
||||||
|
return
|
||||||
|
|
||||||
|
action = EventAction.MODEL_CREATED if created else EventAction.MODEL_UPDATED
|
||||||
|
EventNewThread(action, request, user=user, model=model_to_dict(instance)).run()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
# pylint: disable=unused-argument
|
||||||
|
def pre_delete_handler(
|
||||||
|
user: User, request: HttpRequest, sender, instance: Model, **_
|
||||||
|
):
|
||||||
|
"""Signal handler for all object's pre_delete"""
|
||||||
|
if isinstance(instance, Event):
|
||||||
|
return
|
||||||
|
|
||||||
|
EventNewThread(
|
||||||
|
EventAction.MODEL_DELETED,
|
||||||
|
request,
|
||||||
|
user=user,
|
||||||
|
model=model_to_dict(instance),
|
||||||
|
).run()
|
|
@ -0,0 +1,33 @@
|
||||||
|
# Generated by Django 3.1.1 on 2020-09-18 21:16
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("authentik_audit", "0001_initial"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="event",
|
||||||
|
name="action",
|
||||||
|
field=models.TextField(
|
||||||
|
choices=[
|
||||||
|
("LOGIN", "login"),
|
||||||
|
("LOGIN_FAILED", "login_failed"),
|
||||||
|
("LOGOUT", "logout"),
|
||||||
|
("AUTHORIZE_APPLICATION", "authorize_application"),
|
||||||
|
("SUSPICIOUS_REQUEST", "suspicious_request"),
|
||||||
|
("SIGN_UP", "sign_up"),
|
||||||
|
("PASSWORD_RESET", "password_reset"),
|
||||||
|
("INVITE_CREATED", "invitation_created"),
|
||||||
|
("INVITE_USED", "invitation_used"),
|
||||||
|
("IMPERSONATION_STARTED", "impersonation_started"),
|
||||||
|
("IMPERSONATION_ENDED", "impersonation_ended"),
|
||||||
|
("CUSTOM", "custom"),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,64 @@
|
||||||
|
# Generated by Django 3.1.1 on 2020-09-17 11:55
|
||||||
|
from django.apps.registry import Apps
|
||||||
|
from django.db import migrations, models
|
||||||
|
from django.db.backends.base.schema import BaseDatabaseSchemaEditor
|
||||||
|
|
||||||
|
import authentik.audit.models
|
||||||
|
|
||||||
|
|
||||||
|
def convert_user_to_json(apps: Apps, schema_editor: BaseDatabaseSchemaEditor):
|
||||||
|
Event = apps.get_model("authentik_audit", "Event")
|
||||||
|
|
||||||
|
db_alias = schema_editor.connection.alias
|
||||||
|
for event in Event.objects.all():
|
||||||
|
event.delete()
|
||||||
|
# Because event objects cannot be updated, we have to re-create them
|
||||||
|
event.pk = None
|
||||||
|
event.user_json = (
|
||||||
|
authentik.audit.models.get_user(event.user) if event.user else {}
|
||||||
|
)
|
||||||
|
event._state.adding = True
|
||||||
|
event.save()
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("authentik_audit", "0002_auto_20200918_2116"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="event",
|
||||||
|
name="action",
|
||||||
|
field=models.TextField(
|
||||||
|
choices=[
|
||||||
|
("LOGIN", "login"),
|
||||||
|
("LOGIN_FAILED", "login_failed"),
|
||||||
|
("LOGOUT", "logout"),
|
||||||
|
("AUTHORIZE_APPLICATION", "authorize_application"),
|
||||||
|
("SUSPICIOUS_REQUEST", "suspicious_request"),
|
||||||
|
("SIGN_UP", "sign_up"),
|
||||||
|
("PASSWORD_RESET", "password_reset"),
|
||||||
|
("INVITE_CREATED", "invitation_created"),
|
||||||
|
("INVITE_USED", "invitation_used"),
|
||||||
|
("IMPERSONATION_STARTED", "impersonation_started"),
|
||||||
|
("IMPERSONATION_ENDED", "impersonation_ended"),
|
||||||
|
("CUSTOM", "custom"),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name="event",
|
||||||
|
name="user_json",
|
||||||
|
field=models.JSONField(default=dict),
|
||||||
|
),
|
||||||
|
migrations.RunPython(convert_user_to_json),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="event",
|
||||||
|
name="user",
|
||||||
|
),
|
||||||
|
migrations.RenameField(
|
||||||
|
model_name="event", old_name="user_json", new_name="user"
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,37 @@
|
||||||
|
# Generated by Django 3.1.1 on 2020-09-21 18:29
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("authentik_audit", "0003_auto_20200917_1155"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="event",
|
||||||
|
name="action",
|
||||||
|
field=models.TextField(
|
||||||
|
choices=[
|
||||||
|
("login", "Login"),
|
||||||
|
("login_failed", "Login Failed"),
|
||||||
|
("logout", "Logout"),
|
||||||
|
("sign_up", "Sign Up"),
|
||||||
|
("authorize_application", "Authorize Application"),
|
||||||
|
("suspicious_request", "Suspicious Request"),
|
||||||
|
("password_set", "Password Set"),
|
||||||
|
("invitation_created", "Invite Created"),
|
||||||
|
("invitation_used", "Invite Used"),
|
||||||
|
("source_linked", "Source Linked"),
|
||||||
|
("impersonation_started", "Impersonation Started"),
|
||||||
|
("impersonation_ended", "Impersonation Ended"),
|
||||||
|
("model_created", "Model Created"),
|
||||||
|
("model_updated", "Model Updated"),
|
||||||
|
("model_deleted", "Model Deleted"),
|
||||||
|
("custom_", "Custom Prefix"),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,37 @@
|
||||||
|
# Generated by Django 3.1.2 on 2020-10-05 21:39
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("authentik_audit", "0004_auto_20200921_1829"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="event",
|
||||||
|
name="action",
|
||||||
|
field=models.TextField(
|
||||||
|
choices=[
|
||||||
|
("login", "Login"),
|
||||||
|
("login_failed", "Login Failed"),
|
||||||
|
("logout", "Logout"),
|
||||||
|
("user_write", "User Write"),
|
||||||
|
("suspicious_request", "Suspicious Request"),
|
||||||
|
("password_set", "Password Set"),
|
||||||
|
("invitation_created", "Invite Created"),
|
||||||
|
("invitation_used", "Invite Used"),
|
||||||
|
("authorize_application", "Authorize Application"),
|
||||||
|
("source_linked", "Source Linked"),
|
||||||
|
("impersonation_started", "Impersonation Started"),
|
||||||
|
("impersonation_ended", "Impersonation Ended"),
|
||||||
|
("model_created", "Model Created"),
|
||||||
|
("model_updated", "Model Updated"),
|
||||||
|
("model_deleted", "Model Deleted"),
|
||||||
|
("custom_", "Custom Prefix"),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,42 @@
|
||||||
|
# Generated by Django 3.1.2 on 2020-10-17 20:24
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
("authentik_audit", "0005_auto_20201005_2139"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name="event",
|
||||||
|
name="date",
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="event",
|
||||||
|
name="action",
|
||||||
|
field=models.TextField(
|
||||||
|
choices=[
|
||||||
|
("login", "Login"),
|
||||||
|
("login_failed", "Login Failed"),
|
||||||
|
("logout", "Logout"),
|
||||||
|
("user_write", "User Write"),
|
||||||
|
("suspicious_request", "Suspicious Request"),
|
||||||
|
("password_set", "Password Set"),
|
||||||
|
("token_view", "Token View"),
|
||||||
|
("invitation_created", "Invite Created"),
|
||||||
|
("invitation_used", "Invite Used"),
|
||||||
|
("authorize_application", "Authorize Application"),
|
||||||
|
("source_linked", "Source Linked"),
|
||||||
|
("impersonation_started", "Impersonation Started"),
|
||||||
|
("impersonation_ended", "Impersonation Ended"),
|
||||||
|
("model_created", "Model Created"),
|
||||||
|
("model_updated", "Model Updated"),
|
||||||
|
("model_deleted", "Model Deleted"),
|
||||||
|
("custom_", "Custom Prefix"),
|
||||||
|
]
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
|
@ -0,0 +1,199 @@
|
||||||
|
"""authentik audit models"""
|
||||||
|
from inspect import getmodule, stack
|
||||||
|
from typing import Any, Dict, Optional, Union
|
||||||
|
from uuid import UUID, uuid4
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
from django.contrib.auth.models import AnonymousUser
|
||||||
|
from django.core.exceptions import ValidationError
|
||||||
|
from django.db import models
|
||||||
|
from django.db.models.base import Model
|
||||||
|
from django.http import HttpRequest
|
||||||
|
from django.utils.translation import gettext as _
|
||||||
|
from django.views.debug import SafeExceptionReporterFilter
|
||||||
|
from guardian.utils import get_anonymous_user
|
||||||
|
from structlog import get_logger
|
||||||
|
|
||||||
|
from authentik.core.middleware import (
|
||||||
|
SESSION_IMPERSONATE_ORIGINAL_USER,
|
||||||
|
SESSION_IMPERSONATE_USER,
|
||||||
|
)
|
||||||
|
from authentik.core.models import User
|
||||||
|
from authentik.lib.utils.http import get_client_ip
|
||||||
|
|
||||||
|
LOGGER = get_logger("authentik.audit")
|
||||||
|
|
||||||
|
|
||||||
|
def cleanse_dict(source: Dict[Any, Any]) -> Dict[Any, Any]:
|
||||||
|
"""Cleanse a dictionary, recursively"""
|
||||||
|
final_dict = {}
|
||||||
|
for key, value in source.items():
|
||||||
|
try:
|
||||||
|
if SafeExceptionReporterFilter.hidden_settings.search(key):
|
||||||
|
final_dict[key] = SafeExceptionReporterFilter.cleansed_substitute
|
||||||
|
else:
|
||||||
|
final_dict[key] = value
|
||||||
|
except TypeError:
|
||||||
|
final_dict[key] = value
|
||||||
|
if isinstance(value, dict):
|
||||||
|
final_dict[key] = cleanse_dict(value)
|
||||||
|
return final_dict
|
||||||
|
|
||||||
|
|
||||||
|
def model_to_dict(model: Model) -> Dict[str, Any]:
|
||||||
|
"""Convert model to dict"""
|
||||||
|
name = str(model)
|
||||||
|
if hasattr(model, "name"):
|
||||||
|
name = model.name
|
||||||
|
return {
|
||||||
|
"app": model._meta.app_label,
|
||||||
|
"model_name": model._meta.model_name,
|
||||||
|
"pk": model.pk,
|
||||||
|
"name": name,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def get_user(user: User, original_user: Optional[User] = None) -> Dict[str, Any]:
|
||||||
|
"""Convert user object to dictionary, optionally including the original user"""
|
||||||
|
if isinstance(user, AnonymousUser):
|
||||||
|
user = get_anonymous_user()
|
||||||
|
user_data = {
|
||||||
|
"username": user.username,
|
||||||
|
"pk": user.pk,
|
||||||
|
"email": user.email,
|
||||||
|
}
|
||||||
|
if original_user:
|
||||||
|
original_data = get_user(original_user)
|
||||||
|
original_data["on_behalf_of"] = user_data
|
||||||
|
return original_data
|
||||||
|
return user_data
|
||||||
|
|
||||||
|
|
||||||
|
def sanitize_dict(source: Dict[Any, Any]) -> Dict[Any, Any]:
|
||||||
|
"""clean source of all Models that would interfere with the JSONField.
|
||||||
|
Models are replaced with a dictionary of {
|
||||||
|
app: str,
|
||||||
|
name: str,
|
||||||
|
pk: Any
|
||||||
|
}"""
|
||||||
|
final_dict = {}
|
||||||
|
for key, value in source.items():
|
||||||
|
if isinstance(value, dict):
|
||||||
|
final_dict[key] = sanitize_dict(value)
|
||||||
|
elif isinstance(value, models.Model):
|
||||||
|
final_dict[key] = sanitize_dict(model_to_dict(value))
|
||||||
|
elif isinstance(value, UUID):
|
||||||
|
final_dict[key] = value.hex
|
||||||
|
else:
|
||||||
|
final_dict[key] = value
|
||||||
|
return final_dict
|
||||||
|
|
||||||
|
|
||||||
|
class EventAction(models.TextChoices):
|
||||||
|
"""All possible actions to save into the audit log"""
|
||||||
|
|
||||||
|
LOGIN = "login"
|
||||||
|
LOGIN_FAILED = "login_failed"
|
||||||
|
LOGOUT = "logout"
|
||||||
|
|
||||||
|
USER_WRITE = "user_write"
|
||||||
|
SUSPICIOUS_REQUEST = "suspicious_request"
|
||||||
|
PASSWORD_SET = "password_set" # noqa # nosec
|
||||||
|
|
||||||
|
TOKEN_VIEW = "token_view"
|
||||||
|
|
||||||
|
INVITE_CREATED = "invitation_created"
|
||||||
|
INVITE_USED = "invitation_used"
|
||||||
|
|
||||||
|
AUTHORIZE_APPLICATION = "authorize_application"
|
||||||
|
SOURCE_LINKED = "source_linked"
|
||||||
|
|
||||||
|
IMPERSONATION_STARTED = "impersonation_started"
|
||||||
|
IMPERSONATION_ENDED = "impersonation_ended"
|
||||||
|
|
||||||
|
MODEL_CREATED = "model_created"
|
||||||
|
MODEL_UPDATED = "model_updated"
|
||||||
|
MODEL_DELETED = "model_deleted"
|
||||||
|
|
||||||
|
CUSTOM_PREFIX = "custom_"
|
||||||
|
|
||||||
|
|
||||||
|
class Event(models.Model):
|
||||||
|
"""An individual audit log event"""
|
||||||
|
|
||||||
|
event_uuid = models.UUIDField(primary_key=True, editable=False, default=uuid4)
|
||||||
|
user = models.JSONField(default=dict)
|
||||||
|
action = models.TextField(choices=EventAction.choices)
|
||||||
|
app = models.TextField()
|
||||||
|
context = models.JSONField(default=dict, blank=True)
|
||||||
|
client_ip = models.GenericIPAddressField(null=True)
|
||||||
|
created = models.DateTimeField(auto_now_add=True)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _get_app_from_request(request: HttpRequest) -> str:
|
||||||
|
if not isinstance(request, HttpRequest):
|
||||||
|
return ""
|
||||||
|
return request.resolver_match.app_name
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def new(
|
||||||
|
action: Union[str, EventAction],
|
||||||
|
app: Optional[str] = None,
|
||||||
|
_inspect_offset: int = 1,
|
||||||
|
**kwargs,
|
||||||
|
) -> "Event":
|
||||||
|
"""Create new Event instance from arguments. Instance is NOT saved."""
|
||||||
|
if not isinstance(action, EventAction):
|
||||||
|
action = EventAction.CUSTOM_PREFIX + action
|
||||||
|
if not app:
|
||||||
|
app = getmodule(stack()[_inspect_offset][0]).__name__
|
||||||
|
cleaned_kwargs = cleanse_dict(sanitize_dict(kwargs))
|
||||||
|
event = Event(action=action, app=app, context=cleaned_kwargs)
|
||||||
|
return event
|
||||||
|
|
||||||
|
def from_http(
|
||||||
|
self, request: HttpRequest, user: Optional[settings.AUTH_USER_MODEL] = None
|
||||||
|
) -> "Event":
|
||||||
|
"""Add data from a Django-HttpRequest, allowing the creation of
|
||||||
|
Events independently from requests.
|
||||||
|
`user` arguments optionally overrides user from requests."""
|
||||||
|
if hasattr(request, "user"):
|
||||||
|
self.user = get_user(
|
||||||
|
request.user,
|
||||||
|
request.session.get(SESSION_IMPERSONATE_ORIGINAL_USER, None),
|
||||||
|
)
|
||||||
|
if user:
|
||||||
|
self.user = get_user(user)
|
||||||
|
# Check if we're currently impersonating, and add that user
|
||||||
|
if hasattr(request, "session"):
|
||||||
|
if SESSION_IMPERSONATE_ORIGINAL_USER in request.session:
|
||||||
|
self.user = get_user(request.session[SESSION_IMPERSONATE_ORIGINAL_USER])
|
||||||
|
self.user["on_behalf_of"] = get_user(
|
||||||
|
request.session[SESSION_IMPERSONATE_USER]
|
||||||
|
)
|
||||||
|
# User 255.255.255.255 as fallback if IP cannot be determined
|
||||||
|
self.client_ip = get_client_ip(request) or "255.255.255.255"
|
||||||
|
# If there's no app set, we get it from the requests too
|
||||||
|
if not self.app:
|
||||||
|
self.app = Event._get_app_from_request(request)
|
||||||
|
self.save()
|
||||||
|
return self
|
||||||
|
|
||||||
|
def save(self, *args, **kwargs):
|
||||||
|
if not self._state.adding:
|
||||||
|
raise ValidationError(
|
||||||
|
"you may not edit an existing %s" % self._meta.model_name
|
||||||
|
)
|
||||||
|
LOGGER.debug(
|
||||||
|
"Created Audit event",
|
||||||
|
action=self.action,
|
||||||
|
context=self.context,
|
||||||
|
client_ip=self.client_ip,
|
||||||
|
user=self.user,
|
||||||
|
)
|
||||||
|
return super().save(*args, **kwargs)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
|
||||||
|
verbose_name = _("Audit Event")
|
||||||
|
verbose_name_plural = _("Audit Events")
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue