Compare commits
	
		
			76 Commits
		
	
	
		
			root/more-
			...
			website/ed
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| c8087adc42 | |||
| 3e40fab883 | |||
| 9959221c21 | |||
| e20d0891a0 | |||
| be8ced9fbe | |||
| b6973f6b46 | |||
| eca7022afd | |||
| d9b67e946d | |||
| 35c8ca1120 | |||
| dba17ee6de | |||
| 151529a929 | |||
| cc6ed4439e | |||
| 9b6bc47dc2 | |||
| aae6f05f38 | |||
| de2ecfd28f | |||
| e70c5a1a05 | |||
| 720f1757b8 | |||
| fec46dfca5 | |||
| db059d95a3 | |||
| 833317c5fb | |||
| 44e4a5a99b | |||
| be9b44a085 | |||
| 186e1bf6aa | |||
| 75b605fb9f | |||
| 7b208d9944 | |||
| c49185dc67 | |||
| 10bfc4e6e8 | |||
| fcab99027b | |||
| 312f364e0c | |||
| 2488eb9872 | |||
| 5b132c8b1e | |||
| 3ff20ca9f4 | |||
| 61eb9fafd4 | |||
| 5752497b4d | |||
| 09803fee11 | |||
| 3fae9e5102 | |||
| fffc8c7c0c | |||
| 3d532d4feb | |||
| e1d565d40e | |||
| ee37e9235b | |||
| 8248163958 | |||
| 9acebec1f6 | |||
| 2a96900dc7 | |||
| ca42506fa0 | |||
| 34de6bfd3a | |||
| 2d94b16411 | |||
| 98503f6009 | |||
| ac4ba5d9e2 | |||
| f19ed14bf8 | |||
| 085debf170 | |||
| cacdf64408 | |||
| 23665d173f | |||
| 272fdc516b | |||
| b08dcc2289 | |||
| c84be1d961 | |||
| 875fc5c735 | |||
| 66cefcc918 | |||
| 5d4c38032f | |||
| 7123b2c57b | |||
| fc00bdee63 | |||
| a056703da0 | |||
| 3f9502072d | |||
| 2d254d6a7e | |||
| a7e3dca917 | |||
| 5d8408287f | |||
| 30beca9118 | |||
| 8946b81dbd | |||
| db96e1a901 | |||
| 8b4e0361c4 | |||
| 22cb5b7379 | |||
| 2d0117d096 | |||
| 035bda4eac | |||
| 50906214e5 | |||
| e505f274b6 | |||
| fe52f44dca | |||
| 3146e5a50f | 
@ -1,16 +1,16 @@
 | 
				
			|||||||
[bumpversion]
 | 
					[bumpversion]
 | 
				
			||||||
current_version = 2024.12.2
 | 
					current_version = 2024.10.0
 | 
				
			||||||
tag = True
 | 
					tag = True
 | 
				
			||||||
commit = True
 | 
					commit = True
 | 
				
			||||||
parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)(?:-(?P<rc_t>[a-zA-Z-]+)(?P<rc_n>[1-9]\\d*))?
 | 
					parse = (?P<major>\d+)\.(?P<minor>\d+)\.(?P<patch>\d+)(?:-(?P<rc_t>[a-zA-Z-]+)(?P<rc_n>[1-9]\\d*))?
 | 
				
			||||||
serialize =
 | 
					serialize = 
 | 
				
			||||||
	{major}.{minor}.{patch}-{rc_t}{rc_n}
 | 
						{major}.{minor}.{patch}-{rc_t}{rc_n}
 | 
				
			||||||
	{major}.{minor}.{patch}
 | 
						{major}.{minor}.{patch}
 | 
				
			||||||
message = release: {new_version}
 | 
					message = release: {new_version}
 | 
				
			||||||
tag_name = version/{new_version}
 | 
					tag_name = version/{new_version}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
[bumpversion:part:rc_t]
 | 
					[bumpversion:part:rc_t]
 | 
				
			||||||
values =
 | 
					values = 
 | 
				
			||||||
	rc
 | 
						rc
 | 
				
			||||||
	final
 | 
						final
 | 
				
			||||||
optional_value = final
 | 
					optional_value = final
 | 
				
			||||||
@ -30,5 +30,3 @@ optional_value = final
 | 
				
			|||||||
[bumpversion:file:internal/constants/constants.go]
 | 
					[bumpversion:file:internal/constants/constants.go]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
[bumpversion:file:web/src/common/constants.ts]
 | 
					[bumpversion:file:web/src/common/constants.ts]
 | 
				
			||||||
 | 
					 | 
				
			||||||
[bumpversion:file:website/docs/install-config/install/aws/template.yaml]
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -11,9 +11,9 @@ inputs:
 | 
				
			|||||||
    description: "Docker image arch"
 | 
					    description: "Docker image arch"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
outputs:
 | 
					outputs:
 | 
				
			||||||
  shouldPush:
 | 
					  shouldBuild:
 | 
				
			||||||
    description: "Whether to push the image or not"
 | 
					    description: "Whether to build image or not"
 | 
				
			||||||
    value: ${{ steps.ev.outputs.shouldPush }}
 | 
					    value: ${{ steps.ev.outputs.shouldBuild }}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  sha:
 | 
					  sha:
 | 
				
			||||||
    description: "sha"
 | 
					    description: "sha"
 | 
				
			||||||
 | 
				
			|||||||
@ -7,14 +7,7 @@ from time import time
 | 
				
			|||||||
parser = configparser.ConfigParser()
 | 
					parser = configparser.ConfigParser()
 | 
				
			||||||
parser.read(".bumpversion.cfg")
 | 
					parser.read(".bumpversion.cfg")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Decide if we should push the image or not
 | 
					should_build = str(len(os.environ.get("DOCKER_USERNAME", "")) > 0).lower()
 | 
				
			||||||
should_push = True
 | 
					 | 
				
			||||||
if len(os.environ.get("DOCKER_USERNAME", "")) < 1:
 | 
					 | 
				
			||||||
    # Don't push if we don't have DOCKER_USERNAME, i.e. no secrets are available
 | 
					 | 
				
			||||||
    should_push = False
 | 
					 | 
				
			||||||
if os.environ.get("GITHUB_REPOSITORY").lower() == "goauthentik/authentik-internal":
 | 
					 | 
				
			||||||
    # Don't push on the internal repo
 | 
					 | 
				
			||||||
    should_push = False
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
branch_name = os.environ["GITHUB_REF"]
 | 
					branch_name = os.environ["GITHUB_REF"]
 | 
				
			||||||
if os.environ.get("GITHUB_HEAD_REF", "") != "":
 | 
					if os.environ.get("GITHUB_HEAD_REF", "") != "":
 | 
				
			||||||
@ -71,7 +64,7 @@ def get_attest_image_names(image_with_tags: list[str]):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
with open(os.environ["GITHUB_OUTPUT"], "a+", encoding="utf-8") as _output:
 | 
					with open(os.environ["GITHUB_OUTPUT"], "a+", encoding="utf-8") as _output:
 | 
				
			||||||
    print(f"shouldPush={str(should_push).lower()}", file=_output)
 | 
					    print(f"shouldBuild={should_build}", file=_output)
 | 
				
			||||||
    print(f"sha={sha}", file=_output)
 | 
					    print(f"sha={sha}", file=_output)
 | 
				
			||||||
    print(f"version={version}", file=_output)
 | 
					    print(f"version={version}", file=_output)
 | 
				
			||||||
    print(f"prerelease={prerelease}", file=_output)
 | 
					    print(f"prerelease={prerelease}", file=_output)
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										2
									
								
								.github/actions/setup/action.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/actions/setup/action.yml
									
									
									
									
										vendored
									
									
								
							@ -35,7 +35,7 @@ runs:
 | 
				
			|||||||
      run: |
 | 
					      run: |
 | 
				
			||||||
        export PSQL_TAG=${{ inputs.postgresql_version }}
 | 
					        export PSQL_TAG=${{ inputs.postgresql_version }}
 | 
				
			||||||
        docker compose -f .github/actions/setup/docker-compose.yml up -d
 | 
					        docker compose -f .github/actions/setup/docker-compose.yml up -d
 | 
				
			||||||
        poetry install --sync
 | 
					        poetry install
 | 
				
			||||||
        cd web && npm ci
 | 
					        cd web && npm ci
 | 
				
			||||||
    - name: Generate config
 | 
					    - name: Generate config
 | 
				
			||||||
      shell: poetry run python {0}
 | 
					      shell: poetry run python {0}
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										1
									
								
								.github/workflows/api-py-publish.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/api-py-publish.yml
									
									
									
									
										vendored
									
									
								
							@ -7,7 +7,6 @@ on:
 | 
				
			|||||||
  workflow_dispatch:
 | 
					  workflow_dispatch:
 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  build:
 | 
					  build:
 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    permissions:
 | 
					    permissions:
 | 
				
			||||||
      id-token: write
 | 
					      id-token: write
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										1
									
								
								.github/workflows/api-ts-publish.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/api-ts-publish.yml
									
									
									
									
										vendored
									
									
								
							@ -7,7 +7,6 @@ on:
 | 
				
			|||||||
  workflow_dispatch:
 | 
					  workflow_dispatch:
 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  build:
 | 
					  build:
 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - id: generate_token
 | 
					      - id: generate_token
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										46
									
								
								.github/workflows/ci-aws-cfn.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										46
									
								
								.github/workflows/ci-aws-cfn.yml
									
									
									
									
										vendored
									
									
								
							@ -1,46 +0,0 @@
 | 
				
			|||||||
name: authentik-ci-aws-cfn
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
on:
 | 
					 | 
				
			||||||
  push:
 | 
					 | 
				
			||||||
    branches:
 | 
					 | 
				
			||||||
      - main
 | 
					 | 
				
			||||||
      - next
 | 
					 | 
				
			||||||
      - version-*
 | 
					 | 
				
			||||||
  pull_request:
 | 
					 | 
				
			||||||
    branches:
 | 
					 | 
				
			||||||
      - main
 | 
					 | 
				
			||||||
      - version-*
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
env:
 | 
					 | 
				
			||||||
  POSTGRES_DB: authentik
 | 
					 | 
				
			||||||
  POSTGRES_USER: authentik
 | 
					 | 
				
			||||||
  POSTGRES_PASSWORD: "EK-5jnKfjrGRm<77"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
jobs:
 | 
					 | 
				
			||||||
  check-changes-applied:
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - uses: actions/checkout@v4
 | 
					 | 
				
			||||||
      - name: Setup authentik env
 | 
					 | 
				
			||||||
        uses: ./.github/actions/setup
 | 
					 | 
				
			||||||
      - uses: actions/setup-node@v4
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          node-version-file: website/package.json
 | 
					 | 
				
			||||||
          cache: "npm"
 | 
					 | 
				
			||||||
          cache-dependency-path: website/package-lock.json
 | 
					 | 
				
			||||||
      - working-directory: website/
 | 
					 | 
				
			||||||
        run: |
 | 
					 | 
				
			||||||
          npm ci
 | 
					 | 
				
			||||||
      - name: Check changes have been applied
 | 
					 | 
				
			||||||
        run: |
 | 
					 | 
				
			||||||
          poetry run make aws-cfn
 | 
					 | 
				
			||||||
          git diff --exit-code
 | 
					 | 
				
			||||||
  ci-aws-cfn-mark:
 | 
					 | 
				
			||||||
    if: always()
 | 
					 | 
				
			||||||
    needs:
 | 
					 | 
				
			||||||
      - check-changes-applied
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - uses: re-actors/alls-green@release/v1
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          jobs: ${{ toJSON(needs) }}
 | 
					 | 
				
			||||||
							
								
								
									
										27
									
								
								.github/workflows/ci-main.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										27
									
								
								.github/workflows/ci-main.yml
									
									
									
									
										vendored
									
									
								
							@ -116,7 +116,7 @@ jobs:
 | 
				
			|||||||
          poetry run make test
 | 
					          poetry run make test
 | 
				
			||||||
          poetry run coverage xml
 | 
					          poetry run coverage xml
 | 
				
			||||||
      - if: ${{ always() }}
 | 
					      - if: ${{ always() }}
 | 
				
			||||||
        uses: codecov/codecov-action@v5
 | 
					        uses: codecov/codecov-action@v4
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          flags: unit
 | 
					          flags: unit
 | 
				
			||||||
          token: ${{ secrets.CODECOV_TOKEN }}
 | 
					          token: ${{ secrets.CODECOV_TOKEN }}
 | 
				
			||||||
@ -134,13 +134,13 @@ jobs:
 | 
				
			|||||||
      - name: Setup authentik env
 | 
					      - name: Setup authentik env
 | 
				
			||||||
        uses: ./.github/actions/setup
 | 
					        uses: ./.github/actions/setup
 | 
				
			||||||
      - name: Create k8s Kind Cluster
 | 
					      - name: Create k8s Kind Cluster
 | 
				
			||||||
        uses: helm/kind-action@v1.12.0
 | 
					        uses: helm/kind-action@v1.10.0
 | 
				
			||||||
      - name: run integration
 | 
					      - name: run integration
 | 
				
			||||||
        run: |
 | 
					        run: |
 | 
				
			||||||
          poetry run coverage run manage.py test tests/integration
 | 
					          poetry run coverage run manage.py test tests/integration
 | 
				
			||||||
          poetry run coverage xml
 | 
					          poetry run coverage xml
 | 
				
			||||||
      - if: ${{ always() }}
 | 
					      - if: ${{ always() }}
 | 
				
			||||||
        uses: codecov/codecov-action@v5
 | 
					        uses: codecov/codecov-action@v4
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          flags: integration
 | 
					          flags: integration
 | 
				
			||||||
          token: ${{ secrets.CODECOV_TOKEN }}
 | 
					          token: ${{ secrets.CODECOV_TOKEN }}
 | 
				
			||||||
@ -198,7 +198,7 @@ jobs:
 | 
				
			|||||||
          poetry run coverage run manage.py test ${{ matrix.job.glob }}
 | 
					          poetry run coverage run manage.py test ${{ matrix.job.glob }}
 | 
				
			||||||
          poetry run coverage xml
 | 
					          poetry run coverage xml
 | 
				
			||||||
      - if: ${{ always() }}
 | 
					      - if: ${{ always() }}
 | 
				
			||||||
        uses: codecov/codecov-action@v5
 | 
					        uses: codecov/codecov-action@v4
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          flags: e2e
 | 
					          flags: e2e
 | 
				
			||||||
          token: ${{ secrets.CODECOV_TOKEN }}
 | 
					          token: ${{ secrets.CODECOV_TOKEN }}
 | 
				
			||||||
@ -209,7 +209,6 @@ jobs:
 | 
				
			|||||||
          file: unittest.xml
 | 
					          file: unittest.xml
 | 
				
			||||||
          token: ${{ secrets.CODECOV_TOKEN }}
 | 
					          token: ${{ secrets.CODECOV_TOKEN }}
 | 
				
			||||||
  ci-core-mark:
 | 
					  ci-core-mark:
 | 
				
			||||||
    if: always()
 | 
					 | 
				
			||||||
    needs:
 | 
					    needs:
 | 
				
			||||||
      - lint
 | 
					      - lint
 | 
				
			||||||
      - test-migrations
 | 
					      - test-migrations
 | 
				
			||||||
@ -219,9 +218,7 @@ jobs:
 | 
				
			|||||||
      - test-e2e
 | 
					      - test-e2e
 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - uses: re-actors/alls-green@release/v1
 | 
					      - run: echo mark
 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          jobs: ${{ toJSON(needs) }}
 | 
					 | 
				
			||||||
  build:
 | 
					  build:
 | 
				
			||||||
    strategy:
 | 
					    strategy:
 | 
				
			||||||
      fail-fast: false
 | 
					      fail-fast: false
 | 
				
			||||||
@ -243,7 +240,7 @@ jobs:
 | 
				
			|||||||
        with:
 | 
					        with:
 | 
				
			||||||
          ref: ${{ github.event.pull_request.head.sha }}
 | 
					          ref: ${{ github.event.pull_request.head.sha }}
 | 
				
			||||||
      - name: Set up QEMU
 | 
					      - name: Set up QEMU
 | 
				
			||||||
        uses: docker/setup-qemu-action@v3.3.0
 | 
					        uses: docker/setup-qemu-action@v3.2.0
 | 
				
			||||||
      - name: Set up Docker Buildx
 | 
					      - name: Set up Docker Buildx
 | 
				
			||||||
        uses: docker/setup-buildx-action@v3
 | 
					        uses: docker/setup-buildx-action@v3
 | 
				
			||||||
      - name: prepare variables
 | 
					      - name: prepare variables
 | 
				
			||||||
@ -255,7 +252,7 @@ jobs:
 | 
				
			|||||||
          image-name: ghcr.io/goauthentik/dev-server
 | 
					          image-name: ghcr.io/goauthentik/dev-server
 | 
				
			||||||
          image-arch: ${{ matrix.arch }}
 | 
					          image-arch: ${{ matrix.arch }}
 | 
				
			||||||
      - name: Login to Container Registry
 | 
					      - name: Login to Container Registry
 | 
				
			||||||
        if: ${{ steps.ev.outputs.shouldPush == 'true' }}
 | 
					        if: ${{ steps.ev.outputs.shouldBuild == 'true' }}
 | 
				
			||||||
        uses: docker/login-action@v3
 | 
					        uses: docker/login-action@v3
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          registry: ghcr.io
 | 
					          registry: ghcr.io
 | 
				
			||||||
@ -272,15 +269,15 @@ jobs:
 | 
				
			|||||||
            GEOIPUPDATE_ACCOUNT_ID=${{ secrets.GEOIPUPDATE_ACCOUNT_ID }}
 | 
					            GEOIPUPDATE_ACCOUNT_ID=${{ secrets.GEOIPUPDATE_ACCOUNT_ID }}
 | 
				
			||||||
            GEOIPUPDATE_LICENSE_KEY=${{ secrets.GEOIPUPDATE_LICENSE_KEY }}
 | 
					            GEOIPUPDATE_LICENSE_KEY=${{ secrets.GEOIPUPDATE_LICENSE_KEY }}
 | 
				
			||||||
          tags: ${{ steps.ev.outputs.imageTags }}
 | 
					          tags: ${{ steps.ev.outputs.imageTags }}
 | 
				
			||||||
          push: ${{ steps.ev.outputs.shouldPush == 'true' }}
 | 
					          push: ${{ steps.ev.outputs.shouldBuild == 'true' }}
 | 
				
			||||||
          build-args: |
 | 
					          build-args: |
 | 
				
			||||||
            GIT_BUILD_HASH=${{ steps.ev.outputs.sha }}
 | 
					            GIT_BUILD_HASH=${{ steps.ev.outputs.sha }}
 | 
				
			||||||
          cache-from: type=registry,ref=ghcr.io/goauthentik/dev-server:buildcache
 | 
					          cache-from: type=registry,ref=ghcr.io/goauthentik/dev-server:buildcache
 | 
				
			||||||
          cache-to: ${{ steps.ev.outputs.shouldPush == 'true' && 'type=registry,ref=ghcr.io/goauthentik/dev-server:buildcache,mode=max' || '' }}
 | 
					          cache-to: ${{ steps.ev.outputs.shouldBuild == 'true' && 'type=registry,ref=ghcr.io/goauthentik/dev-server:buildcache,mode=max' || '' }}
 | 
				
			||||||
          platforms: linux/${{ matrix.arch }}
 | 
					          platforms: linux/${{ matrix.arch }}
 | 
				
			||||||
      - uses: actions/attest-build-provenance@v2
 | 
					      - uses: actions/attest-build-provenance@v1
 | 
				
			||||||
        id: attest
 | 
					        id: attest
 | 
				
			||||||
        if: ${{ steps.ev.outputs.shouldPush == 'true' }}
 | 
					        if: ${{ steps.ev.outputs.shouldBuild == 'true' }}
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
					          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
				
			||||||
          subject-digest: ${{ steps.push.outputs.digest }}
 | 
					          subject-digest: ${{ steps.push.outputs.digest }}
 | 
				
			||||||
@ -306,7 +303,7 @@ jobs:
 | 
				
			|||||||
        with:
 | 
					        with:
 | 
				
			||||||
          image-name: ghcr.io/goauthentik/dev-server
 | 
					          image-name: ghcr.io/goauthentik/dev-server
 | 
				
			||||||
      - name: Comment on PR
 | 
					      - name: Comment on PR
 | 
				
			||||||
        if: ${{ steps.ev.outputs.shouldPush == 'true' }}
 | 
					        if: ${{ steps.ev.outputs.shouldBuild == 'true' }}
 | 
				
			||||||
        uses: ./.github/actions/comment-pr-instructions
 | 
					        uses: ./.github/actions/comment-pr-instructions
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          tag: ${{ steps.ev.outputs.imageMainTag }}
 | 
					          tag: ${{ steps.ev.outputs.imageMainTag }}
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										17
									
								
								.github/workflows/ci-outpost.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										17
									
								
								.github/workflows/ci-outpost.yml
									
									
									
									
										vendored
									
									
								
							@ -49,15 +49,12 @@ jobs:
 | 
				
			|||||||
        run: |
 | 
					        run: |
 | 
				
			||||||
          go test -timeout 0 -v -race -coverprofile=coverage.out -covermode=atomic -cover ./...
 | 
					          go test -timeout 0 -v -race -coverprofile=coverage.out -covermode=atomic -cover ./...
 | 
				
			||||||
  ci-outpost-mark:
 | 
					  ci-outpost-mark:
 | 
				
			||||||
    if: always()
 | 
					 | 
				
			||||||
    needs:
 | 
					    needs:
 | 
				
			||||||
      - lint-golint
 | 
					      - lint-golint
 | 
				
			||||||
      - test-unittest
 | 
					      - test-unittest
 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - uses: re-actors/alls-green@release/v1
 | 
					      - run: echo mark
 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          jobs: ${{ toJSON(needs) }}
 | 
					 | 
				
			||||||
  build-container:
 | 
					  build-container:
 | 
				
			||||||
    timeout-minutes: 120
 | 
					    timeout-minutes: 120
 | 
				
			||||||
    needs:
 | 
					    needs:
 | 
				
			||||||
@ -82,7 +79,7 @@ jobs:
 | 
				
			|||||||
        with:
 | 
					        with:
 | 
				
			||||||
          ref: ${{ github.event.pull_request.head.sha }}
 | 
					          ref: ${{ github.event.pull_request.head.sha }}
 | 
				
			||||||
      - name: Set up QEMU
 | 
					      - name: Set up QEMU
 | 
				
			||||||
        uses: docker/setup-qemu-action@v3.3.0
 | 
					        uses: docker/setup-qemu-action@v3.2.0
 | 
				
			||||||
      - name: Set up Docker Buildx
 | 
					      - name: Set up Docker Buildx
 | 
				
			||||||
        uses: docker/setup-buildx-action@v3
 | 
					        uses: docker/setup-buildx-action@v3
 | 
				
			||||||
      - name: prepare variables
 | 
					      - name: prepare variables
 | 
				
			||||||
@ -93,7 +90,7 @@ jobs:
 | 
				
			|||||||
        with:
 | 
					        with:
 | 
				
			||||||
          image-name: ghcr.io/goauthentik/dev-${{ matrix.type }}
 | 
					          image-name: ghcr.io/goauthentik/dev-${{ matrix.type }}
 | 
				
			||||||
      - name: Login to Container Registry
 | 
					      - name: Login to Container Registry
 | 
				
			||||||
        if: ${{ steps.ev.outputs.shouldPush == 'true' }}
 | 
					        if: ${{ steps.ev.outputs.shouldBuild == 'true' }}
 | 
				
			||||||
        uses: docker/login-action@v3
 | 
					        uses: docker/login-action@v3
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          registry: ghcr.io
 | 
					          registry: ghcr.io
 | 
				
			||||||
@ -107,16 +104,16 @@ jobs:
 | 
				
			|||||||
        with:
 | 
					        with:
 | 
				
			||||||
          tags: ${{ steps.ev.outputs.imageTags }}
 | 
					          tags: ${{ steps.ev.outputs.imageTags }}
 | 
				
			||||||
          file: ${{ matrix.type }}.Dockerfile
 | 
					          file: ${{ matrix.type }}.Dockerfile
 | 
				
			||||||
          push: ${{ steps.ev.outputs.shouldPush == 'true' }}
 | 
					          push: ${{ steps.ev.outputs.shouldBuild == 'true' }}
 | 
				
			||||||
          build-args: |
 | 
					          build-args: |
 | 
				
			||||||
            GIT_BUILD_HASH=${{ steps.ev.outputs.sha }}
 | 
					            GIT_BUILD_HASH=${{ steps.ev.outputs.sha }}
 | 
				
			||||||
          platforms: linux/amd64,linux/arm64
 | 
					          platforms: linux/amd64,linux/arm64
 | 
				
			||||||
          context: .
 | 
					          context: .
 | 
				
			||||||
          cache-from: type=registry,ref=ghcr.io/goauthentik/dev-${{ matrix.type }}:buildcache
 | 
					          cache-from: type=registry,ref=ghcr.io/goauthentik/dev-${{ matrix.type }}:buildcache
 | 
				
			||||||
          cache-to: ${{ steps.ev.outputs.shouldPush == 'true' && format('type=registry,ref=ghcr.io/goauthentik/dev-{0}:buildcache,mode=max', matrix.type) || '' }}
 | 
					          cache-to: ${{ steps.ev.outputs.shouldBuild == 'true' && format('type=registry,ref=ghcr.io/goauthentik/dev-{0}:buildcache,mode=max', matrix.type) || '' }}
 | 
				
			||||||
      - uses: actions/attest-build-provenance@v2
 | 
					      - uses: actions/attest-build-provenance@v1
 | 
				
			||||||
        id: attest
 | 
					        id: attest
 | 
				
			||||||
        if: ${{ steps.ev.outputs.shouldPush == 'true' }}
 | 
					        if: ${{ steps.ev.outputs.shouldBuild == 'true' }}
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
					          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
				
			||||||
          subject-digest: ${{ steps.push.outputs.digest }}
 | 
					          subject-digest: ${{ steps.push.outputs.digest }}
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										5
									
								
								.github/workflows/ci-web.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/ci-web.yml
									
									
									
									
										vendored
									
									
								
							@ -61,15 +61,12 @@ jobs:
 | 
				
			|||||||
        working-directory: web/
 | 
					        working-directory: web/
 | 
				
			||||||
        run: npm run build
 | 
					        run: npm run build
 | 
				
			||||||
  ci-web-mark:
 | 
					  ci-web-mark:
 | 
				
			||||||
    if: always()
 | 
					 | 
				
			||||||
    needs:
 | 
					    needs:
 | 
				
			||||||
      - build
 | 
					      - build
 | 
				
			||||||
      - lint
 | 
					      - lint
 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - uses: re-actors/alls-green@release/v1
 | 
					      - run: echo mark
 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          jobs: ${{ toJSON(needs) }}
 | 
					 | 
				
			||||||
  test:
 | 
					  test:
 | 
				
			||||||
    needs:
 | 
					    needs:
 | 
				
			||||||
      - ci-web-mark
 | 
					      - ci-web-mark
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										5
									
								
								.github/workflows/ci-website.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/ci-website.yml
									
									
									
									
										vendored
									
									
								
							@ -62,13 +62,10 @@ jobs:
 | 
				
			|||||||
        working-directory: website/
 | 
					        working-directory: website/
 | 
				
			||||||
        run: npm run ${{ matrix.job }}
 | 
					        run: npm run ${{ matrix.job }}
 | 
				
			||||||
  ci-website-mark:
 | 
					  ci-website-mark:
 | 
				
			||||||
    if: always()
 | 
					 | 
				
			||||||
    needs:
 | 
					    needs:
 | 
				
			||||||
      - lint
 | 
					      - lint
 | 
				
			||||||
      - test
 | 
					      - test
 | 
				
			||||||
      - build
 | 
					      - build
 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - uses: re-actors/alls-green@release/v1
 | 
					      - run: echo mark
 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          jobs: ${{ toJSON(needs) }}
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -11,7 +11,6 @@ env:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  build:
 | 
					  build:
 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - id: generate_token
 | 
					      - id: generate_token
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										1
									
								
								.github/workflows/ghcr-retention.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/ghcr-retention.yml
									
									
									
									
										vendored
									
									
								
							@ -7,7 +7,6 @@ on:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  clean-ghcr:
 | 
					  clean-ghcr:
 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    name: Delete old unused container images
 | 
					    name: Delete old unused container images
 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										1
									
								
								.github/workflows/publish-source-docs.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/publish-source-docs.yml
									
									
									
									
										vendored
									
									
								
							@ -12,7 +12,6 @@ env:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  publish-source-docs:
 | 
					  publish-source-docs:
 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    timeout-minutes: 120
 | 
					    timeout-minutes: 120
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										1
									
								
								.github/workflows/release-next-branch.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/release-next-branch.yml
									
									
									
									
										vendored
									
									
								
							@ -11,7 +11,6 @@ permissions:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  update-next:
 | 
					  update-next:
 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    environment: internal-production
 | 
					    environment: internal-production
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										29
									
								
								.github/workflows/release-publish.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/workflows/release-publish.yml
									
									
									
									
										vendored
									
									
								
							@ -17,7 +17,7 @@ jobs:
 | 
				
			|||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - uses: actions/checkout@v4
 | 
					      - uses: actions/checkout@v4
 | 
				
			||||||
      - name: Set up QEMU
 | 
					      - name: Set up QEMU
 | 
				
			||||||
        uses: docker/setup-qemu-action@v3.3.0
 | 
					        uses: docker/setup-qemu-action@v3.2.0
 | 
				
			||||||
      - name: Set up Docker Buildx
 | 
					      - name: Set up Docker Buildx
 | 
				
			||||||
        uses: docker/setup-buildx-action@v3
 | 
					        uses: docker/setup-buildx-action@v3
 | 
				
			||||||
      - name: prepare variables
 | 
					      - name: prepare variables
 | 
				
			||||||
@ -55,7 +55,7 @@ jobs:
 | 
				
			|||||||
            VERSION=${{ github.ref }}
 | 
					            VERSION=${{ github.ref }}
 | 
				
			||||||
          tags: ${{ steps.ev.outputs.imageTags }}
 | 
					          tags: ${{ steps.ev.outputs.imageTags }}
 | 
				
			||||||
          platforms: linux/amd64,linux/arm64
 | 
					          platforms: linux/amd64,linux/arm64
 | 
				
			||||||
      - uses: actions/attest-build-provenance@v2
 | 
					      - uses: actions/attest-build-provenance@v1
 | 
				
			||||||
        id: attest
 | 
					        id: attest
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
					          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
				
			||||||
@ -83,7 +83,7 @@ jobs:
 | 
				
			|||||||
        with:
 | 
					        with:
 | 
				
			||||||
          go-version-file: "go.mod"
 | 
					          go-version-file: "go.mod"
 | 
				
			||||||
      - name: Set up QEMU
 | 
					      - name: Set up QEMU
 | 
				
			||||||
        uses: docker/setup-qemu-action@v3.3.0
 | 
					        uses: docker/setup-qemu-action@v3.2.0
 | 
				
			||||||
      - name: Set up Docker Buildx
 | 
					      - name: Set up Docker Buildx
 | 
				
			||||||
        uses: docker/setup-buildx-action@v3
 | 
					        uses: docker/setup-buildx-action@v3
 | 
				
			||||||
      - name: prepare variables
 | 
					      - name: prepare variables
 | 
				
			||||||
@ -119,7 +119,7 @@ jobs:
 | 
				
			|||||||
          file: ${{ matrix.type }}.Dockerfile
 | 
					          file: ${{ matrix.type }}.Dockerfile
 | 
				
			||||||
          platforms: linux/amd64,linux/arm64
 | 
					          platforms: linux/amd64,linux/arm64
 | 
				
			||||||
          context: .
 | 
					          context: .
 | 
				
			||||||
      - uses: actions/attest-build-provenance@v2
 | 
					      - uses: actions/attest-build-provenance@v1
 | 
				
			||||||
        id: attest
 | 
					        id: attest
 | 
				
			||||||
        with:
 | 
					        with:
 | 
				
			||||||
          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
					          subject-name: ${{ steps.ev.outputs.attestImageNames }}
 | 
				
			||||||
@ -169,27 +169,6 @@ jobs:
 | 
				
			|||||||
          file: ./authentik-outpost-${{ matrix.type }}_${{ matrix.goos }}_${{ matrix.goarch }}
 | 
					          file: ./authentik-outpost-${{ matrix.type }}_${{ matrix.goos }}_${{ matrix.goarch }}
 | 
				
			||||||
          asset_name: authentik-outpost-${{ matrix.type }}_${{ matrix.goos }}_${{ matrix.goarch }}
 | 
					          asset_name: authentik-outpost-${{ matrix.type }}_${{ matrix.goos }}_${{ matrix.goarch }}
 | 
				
			||||||
          tag: ${{ github.ref }}
 | 
					          tag: ${{ github.ref }}
 | 
				
			||||||
  upload-aws-cfn-template:
 | 
					 | 
				
			||||||
    permissions:
 | 
					 | 
				
			||||||
      # Needed for AWS login
 | 
					 | 
				
			||||||
      id-token: write
 | 
					 | 
				
			||||||
      contents: read
 | 
					 | 
				
			||||||
    needs:
 | 
					 | 
				
			||||||
      - build-server
 | 
					 | 
				
			||||||
      - build-outpost
 | 
					 | 
				
			||||||
    env:
 | 
					 | 
				
			||||||
      AWS_REGION: eu-central-1
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - uses: actions/checkout@v4
 | 
					 | 
				
			||||||
      - uses: aws-actions/configure-aws-credentials@v4
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          role-to-assume: "arn:aws:iam::016170277896:role/github_goauthentik_authentik"
 | 
					 | 
				
			||||||
          aws-region: ${{ env.AWS_REGION }}
 | 
					 | 
				
			||||||
      - name: Upload template
 | 
					 | 
				
			||||||
        run: |
 | 
					 | 
				
			||||||
          aws s3 cp --acl=public-read website/docs/install-config/install/aws/template.yaml s3://authentik-cloudformation-templates/authentik.ecs.${{ github.ref }}.yaml
 | 
					 | 
				
			||||||
          aws s3 cp --acl=public-read website/docs/install-config/install/aws/template.yaml s3://authentik-cloudformation-templates/authentik.ecs.latest.yaml
 | 
					 | 
				
			||||||
  test-release:
 | 
					  test-release:
 | 
				
			||||||
    needs:
 | 
					    needs:
 | 
				
			||||||
      - build-server
 | 
					      - build-server
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										21
									
								
								.github/workflows/repo-mirror.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										21
									
								
								.github/workflows/repo-mirror.yml
									
									
									
									
										vendored
									
									
								
							@ -1,21 +0,0 @@
 | 
				
			|||||||
name: "authentik-repo-mirror"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
on: [push, delete]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
jobs:
 | 
					 | 
				
			||||||
  to_internal:
 | 
					 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					 | 
				
			||||||
    steps:
 | 
					 | 
				
			||||||
      - uses: actions/checkout@v4
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          fetch-depth: 0
 | 
					 | 
				
			||||||
      - if: ${{ env.MIRROR_KEY != '' }}
 | 
					 | 
				
			||||||
        uses: pixta-dev/repository-mirroring-action@v1
 | 
					 | 
				
			||||||
        with:
 | 
					 | 
				
			||||||
          target_repo_url:
 | 
					 | 
				
			||||||
            git@github.com:goauthentik/authentik-internal.git
 | 
					 | 
				
			||||||
          ssh_private_key:
 | 
					 | 
				
			||||||
            ${{ secrets.GH_MIRROR_KEY }}
 | 
					 | 
				
			||||||
        env:
 | 
					 | 
				
			||||||
          MIRROR_KEY: ${{ secrets.GH_MIRROR_KEY }}
 | 
					 | 
				
			||||||
							
								
								
									
										1
									
								
								.github/workflows/repo-stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/workflows/repo-stale.yml
									
									
									
									
										vendored
									
									
								
							@ -11,7 +11,6 @@ permissions:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
jobs:
 | 
					jobs:
 | 
				
			||||||
  stale:
 | 
					  stale:
 | 
				
			||||||
    if: ${{ github.repository != 'goauthentik/authentik-internal' }}
 | 
					 | 
				
			||||||
    runs-on: ubuntu-latest
 | 
					    runs-on: ubuntu-latest
 | 
				
			||||||
    steps:
 | 
					    steps:
 | 
				
			||||||
      - id: generate_token
 | 
					      - id: generate_token
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							@ -33,8 +33,7 @@
 | 
				
			|||||||
        "!If sequence",
 | 
					        "!If sequence",
 | 
				
			||||||
        "!Index scalar",
 | 
					        "!Index scalar",
 | 
				
			||||||
        "!KeyOf scalar",
 | 
					        "!KeyOf scalar",
 | 
				
			||||||
        "!Value scalar",
 | 
					        "!Value scalar"
 | 
				
			||||||
        "!AtIndex scalar"
 | 
					 | 
				
			||||||
    ],
 | 
					    ],
 | 
				
			||||||
    "typescript.preferences.importModuleSpecifier": "non-relative",
 | 
					    "typescript.preferences.importModuleSpecifier": "non-relative",
 | 
				
			||||||
    "typescript.preferences.importModuleSpecifierEnding": "index",
 | 
					    "typescript.preferences.importModuleSpecifierEnding": "index",
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										10
									
								
								CODEOWNERS
									
									
									
									
									
								
							
							
						
						
									
										10
									
								
								CODEOWNERS
									
									
									
									
									
								
							@ -19,18 +19,10 @@ Dockerfile                      @goauthentik/infrastructure
 | 
				
			|||||||
*Dockerfile                     @goauthentik/infrastructure
 | 
					*Dockerfile                     @goauthentik/infrastructure
 | 
				
			||||||
.dockerignore                   @goauthentik/infrastructure
 | 
					.dockerignore                   @goauthentik/infrastructure
 | 
				
			||||||
docker-compose.yml              @goauthentik/infrastructure
 | 
					docker-compose.yml              @goauthentik/infrastructure
 | 
				
			||||||
Makefile                        @goauthentik/infrastructure
 | 
					 | 
				
			||||||
.editorconfig                   @goauthentik/infrastructure
 | 
					 | 
				
			||||||
CODEOWNERS                      @goauthentik/infrastructure
 | 
					 | 
				
			||||||
# Web
 | 
					# Web
 | 
				
			||||||
web/                            @goauthentik/frontend
 | 
					web/                            @goauthentik/frontend
 | 
				
			||||||
tests/wdio/                     @goauthentik/frontend
 | 
					tests/wdio/                     @goauthentik/frontend
 | 
				
			||||||
# Locale
 | 
					 | 
				
			||||||
locale/                         @goauthentik/backend @goauthentik/frontend
 | 
					 | 
				
			||||||
web/xliff/                      @goauthentik/backend @goauthentik/frontend
 | 
					 | 
				
			||||||
# Docs & Website
 | 
					# Docs & Website
 | 
				
			||||||
website/                        @goauthentik/docs
 | 
					website/                        @goauthentik/docs
 | 
				
			||||||
CODE_OF_CONDUCT.md              @goauthentik/docs
 | 
					 | 
				
			||||||
# Security
 | 
					# Security
 | 
				
			||||||
SECURITY.md                     @goauthentik/security @goauthentik/docs
 | 
					website/docs/security/          @goauthentik/security
 | 
				
			||||||
website/docs/security/          @goauthentik/security @goauthentik/docs
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -1 +1 @@
 | 
				
			|||||||
website/docs/developer-docs/index.md
 | 
					website/developer-docs/index.md
 | 
				
			||||||
@ -80,7 +80,7 @@ RUN --mount=type=cache,sharing=locked,target=/go/pkg/mod \
 | 
				
			|||||||
    go build -o /go/authentik ./cmd/server
 | 
					    go build -o /go/authentik ./cmd/server
 | 
				
			||||||
 | 
					
 | 
				
			||||||
# Stage 4: MaxMind GeoIP
 | 
					# Stage 4: MaxMind GeoIP
 | 
				
			||||||
FROM --platform=${BUILDPLATFORM} ghcr.io/maxmind/geoipupdate:v7.1.0 AS geoip
 | 
					FROM --platform=${BUILDPLATFORM} ghcr.io/maxmind/geoipupdate:v7.0.1 AS geoip
 | 
				
			||||||
 | 
					
 | 
				
			||||||
ENV GEOIPUPDATE_EDITION_IDS="GeoLite2-City GeoLite2-ASN"
 | 
					ENV GEOIPUPDATE_EDITION_IDS="GeoLite2-City GeoLite2-ASN"
 | 
				
			||||||
ENV GEOIPUPDATE_VERBOSE="1"
 | 
					ENV GEOIPUPDATE_VERBOSE="1"
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										5
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										5
									
								
								Makefile
									
									
									
									
									
								
							@ -5,7 +5,7 @@ PWD = $(shell pwd)
 | 
				
			|||||||
UID = $(shell id -u)
 | 
					UID = $(shell id -u)
 | 
				
			||||||
GID = $(shell id -g)
 | 
					GID = $(shell id -g)
 | 
				
			||||||
NPM_VERSION = $(shell python -m scripts.npm_version)
 | 
					NPM_VERSION = $(shell python -m scripts.npm_version)
 | 
				
			||||||
PY_SOURCES = authentik tests scripts lifecycle .github website/docs/install-config/install/aws
 | 
					PY_SOURCES = authentik tests scripts lifecycle .github
 | 
				
			||||||
DOCKER_IMAGE ?= "authentik:test"
 | 
					DOCKER_IMAGE ?= "authentik:test"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
GEN_API_TS = "gen-ts-api"
 | 
					GEN_API_TS = "gen-ts-api"
 | 
				
			||||||
@ -252,9 +252,6 @@ website-build:
 | 
				
			|||||||
website-watch:  ## Build and watch the documentation website, updating automatically
 | 
					website-watch:  ## Build and watch the documentation website, updating automatically
 | 
				
			||||||
	cd website && npm run watch
 | 
						cd website && npm run watch
 | 
				
			||||||
 | 
					
 | 
				
			||||||
aws-cfn:
 | 
					 | 
				
			||||||
	cd website && npm run aws-cfn
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
#########################
 | 
					#########################
 | 
				
			||||||
## Docker
 | 
					## Docker
 | 
				
			||||||
#########################
 | 
					#########################
 | 
				
			||||||
 | 
				
			|||||||
@ -2,7 +2,7 @@ authentik takes security very seriously. We follow the rules of [responsible di
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
## Independent audits and pentests
 | 
					## Independent audits and pentests
 | 
				
			||||||
 | 
					
 | 
				
			||||||
We are committed to engaging in regular pentesting and security audits of authentik. Defining and adhering to a cadence of external testing ensures a stronger probability that our code base, our features, and our architecture is as secure and non-exploitable as possible. For more details about specfic audits and pentests, refer to "Audits and Certificates" in our [Security documentation](https://docs.goauthentik.io/docs/security).
 | 
					In May/June of 2023 [Cure53](https://cure53.de) conducted an audit and pentest. The [results](https://cure53.de/pentest-report_authentik.pdf) are published on the [Cure53 website](https://cure53.de/#publications-2023). For more details about authentik's response to the findings of the audit refer to [2023-06 Cure53 Code audit](https://goauthentik.io/docs/security/2023-06-cure53).
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## What authentik classifies as a CVE
 | 
					## What authentik classifies as a CVE
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -20,8 +20,8 @@ Even if the issue is not a CVE, we still greatly appreciate your help in hardeni
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
| Version   | Supported |
 | 
					| Version   | Supported |
 | 
				
			||||||
| --------- | --------- |
 | 
					| --------- | --------- |
 | 
				
			||||||
 | 
					| 2024.8.x  | ✅        |
 | 
				
			||||||
| 2024.10.x | ✅        |
 | 
					| 2024.10.x | ✅        |
 | 
				
			||||||
| 2024.12.x | ✅        |
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
## Reporting a Vulnerability
 | 
					## Reporting a Vulnerability
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -2,7 +2,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from os import environ
 | 
					from os import environ
 | 
				
			||||||
 | 
					
 | 
				
			||||||
__version__ = "2024.12.2"
 | 
					__version__ = "2024.10.0"
 | 
				
			||||||
ENV_GIT_HASH_KEY = "GIT_BUILD_HASH"
 | 
					ENV_GIT_HASH_KEY = "GIT_BUILD_HASH"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -16,5 +16,5 @@ def get_full_version() -> str:
 | 
				
			|||||||
    """Get full version, with build hash appended"""
 | 
					    """Get full version, with build hash appended"""
 | 
				
			||||||
    version = __version__
 | 
					    version = __version__
 | 
				
			||||||
    if (build_hash := get_build_hash()) != "":
 | 
					    if (build_hash := get_build_hash()) != "":
 | 
				
			||||||
        return f"{version}+{build_hash}"
 | 
					        version += "." + build_hash
 | 
				
			||||||
    return version
 | 
					    return version
 | 
				
			||||||
 | 
				
			|||||||
@ -7,9 +7,7 @@ from sys import version as python_version
 | 
				
			|||||||
from typing import TypedDict
 | 
					from typing import TypedDict
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from cryptography.hazmat.backends.openssl.backend import backend
 | 
					from cryptography.hazmat.backends.openssl.backend import backend
 | 
				
			||||||
from django.conf import settings
 | 
					 | 
				
			||||||
from django.utils.timezone import now
 | 
					from django.utils.timezone import now
 | 
				
			||||||
from django.views.debug import SafeExceptionReporterFilter
 | 
					 | 
				
			||||||
from drf_spectacular.utils import extend_schema
 | 
					from drf_spectacular.utils import extend_schema
 | 
				
			||||||
from rest_framework.fields import SerializerMethodField
 | 
					from rest_framework.fields import SerializerMethodField
 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
@ -54,16 +52,10 @@ class SystemInfoSerializer(PassiveSerializer):
 | 
				
			|||||||
    def get_http_headers(self, request: Request) -> dict[str, str]:
 | 
					    def get_http_headers(self, request: Request) -> dict[str, str]:
 | 
				
			||||||
        """Get HTTP Request headers"""
 | 
					        """Get HTTP Request headers"""
 | 
				
			||||||
        headers = {}
 | 
					        headers = {}
 | 
				
			||||||
        raw_session = request._request.COOKIES.get(settings.SESSION_COOKIE_NAME)
 | 
					 | 
				
			||||||
        for key, value in request.META.items():
 | 
					        for key, value in request.META.items():
 | 
				
			||||||
            if not isinstance(value, str):
 | 
					            if not isinstance(value, str):
 | 
				
			||||||
                continue
 | 
					                continue
 | 
				
			||||||
            actual_value = value
 | 
					            headers[key] = value
 | 
				
			||||||
            if raw_session in actual_value:
 | 
					 | 
				
			||||||
                actual_value = actual_value.replace(
 | 
					 | 
				
			||||||
                    raw_session, SafeExceptionReporterFilter.cleansed_substitute
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
            headers[key] = actual_value
 | 
					 | 
				
			||||||
        return headers
 | 
					        return headers
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def get_http_host(self, request: Request) -> str:
 | 
					    def get_http_host(self, request: Request) -> str:
 | 
				
			||||||
 | 
				
			|||||||
@ -1,16 +1,12 @@
 | 
				
			|||||||
"""authentik administration overview"""
 | 
					"""authentik administration overview"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from socket import gethostname
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.conf import settings
 | 
					from django.conf import settings
 | 
				
			||||||
from drf_spectacular.utils import extend_schema, inline_serializer
 | 
					from drf_spectacular.utils import extend_schema, inline_serializer
 | 
				
			||||||
from packaging.version import parse
 | 
					from rest_framework.fields import IntegerField
 | 
				
			||||||
from rest_framework.fields import BooleanField, CharField
 | 
					 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
from rest_framework.response import Response
 | 
					from rest_framework.response import Response
 | 
				
			||||||
from rest_framework.views import APIView
 | 
					from rest_framework.views import APIView
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik import get_full_version
 | 
					 | 
				
			||||||
from authentik.rbac.permissions import HasPermission
 | 
					from authentik.rbac.permissions import HasPermission
 | 
				
			||||||
from authentik.root.celery import CELERY_APP
 | 
					from authentik.root.celery import CELERY_APP
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -20,38 +16,11 @@ class WorkerView(APIView):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    permission_classes = [HasPermission("authentik_rbac.view_system_info")]
 | 
					    permission_classes = [HasPermission("authentik_rbac.view_system_info")]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @extend_schema(
 | 
					    @extend_schema(responses=inline_serializer("Workers", fields={"count": IntegerField()}))
 | 
				
			||||||
        responses=inline_serializer(
 | 
					 | 
				
			||||||
            "Worker",
 | 
					 | 
				
			||||||
            fields={
 | 
					 | 
				
			||||||
                "worker_id": CharField(),
 | 
					 | 
				
			||||||
                "version": CharField(),
 | 
					 | 
				
			||||||
                "version_matching": BooleanField(),
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
            many=True,
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    def get(self, request: Request) -> Response:
 | 
					    def get(self, request: Request) -> Response:
 | 
				
			||||||
        """Get currently connected worker count."""
 | 
					        """Get currently connected worker count."""
 | 
				
			||||||
        raw: list[dict[str, dict]] = CELERY_APP.control.ping(timeout=0.5)
 | 
					        count = len(CELERY_APP.control.ping(timeout=0.5))
 | 
				
			||||||
        our_version = parse(get_full_version())
 | 
					 | 
				
			||||||
        response = []
 | 
					 | 
				
			||||||
        for worker in raw:
 | 
					 | 
				
			||||||
            key = list(worker.keys())[0]
 | 
					 | 
				
			||||||
            version = worker[key].get("version")
 | 
					 | 
				
			||||||
            version_matching = False
 | 
					 | 
				
			||||||
            if version:
 | 
					 | 
				
			||||||
                version_matching = parse(version) == our_version
 | 
					 | 
				
			||||||
            response.append(
 | 
					 | 
				
			||||||
                {"worker_id": key, "version": version, "version_matching": version_matching}
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
        # In debug we run with `task_always_eager`, so tasks are ran on the main process
 | 
					        # In debug we run with `task_always_eager`, so tasks are ran on the main process
 | 
				
			||||||
        if settings.DEBUG:  # pragma: no cover
 | 
					        if settings.DEBUG:  # pragma: no cover
 | 
				
			||||||
            response.append(
 | 
					            count += 1
 | 
				
			||||||
                {
 | 
					        return Response({"count": count})
 | 
				
			||||||
                    "worker_id": f"authentik-debug@{gethostname()}",
 | 
					 | 
				
			||||||
                    "version": get_full_version(),
 | 
					 | 
				
			||||||
                    "version_matching": True,
 | 
					 | 
				
			||||||
                }
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
        return Response(response)
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -1,10 +1,11 @@
 | 
				
			|||||||
"""authentik admin app config"""
 | 
					"""authentik admin app config"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from prometheus_client import Info
 | 
					from prometheus_client import Gauge, Info
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik.blueprints.apps import ManagedAppConfig
 | 
					from authentik.blueprints.apps import ManagedAppConfig
 | 
				
			||||||
 | 
					
 | 
				
			||||||
PROM_INFO = Info("authentik_version", "Currently running authentik version")
 | 
					PROM_INFO = Info("authentik_version", "Currently running authentik version")
 | 
				
			||||||
 | 
					GAUGE_WORKERS = Gauge("authentik_admin_workers", "Currently connected workers")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class AuthentikAdminConfig(ManagedAppConfig):
 | 
					class AuthentikAdminConfig(ManagedAppConfig):
 | 
				
			||||||
 | 
				
			|||||||
@ -1,35 +1,14 @@
 | 
				
			|||||||
"""admin signals"""
 | 
					"""admin signals"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from django.dispatch import receiver
 | 
					from django.dispatch import receiver
 | 
				
			||||||
from packaging.version import parse
 | 
					 | 
				
			||||||
from prometheus_client import Gauge
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik import get_full_version
 | 
					from authentik.admin.apps import GAUGE_WORKERS
 | 
				
			||||||
from authentik.root.celery import CELERY_APP
 | 
					from authentik.root.celery import CELERY_APP
 | 
				
			||||||
from authentik.root.monitoring import monitoring_set
 | 
					from authentik.root.monitoring import monitoring_set
 | 
				
			||||||
 | 
					
 | 
				
			||||||
GAUGE_WORKERS = Gauge(
 | 
					 | 
				
			||||||
    "authentik_admin_workers",
 | 
					 | 
				
			||||||
    "Currently connected workers, their versions and if they are the same version as authentik",
 | 
					 | 
				
			||||||
    ["version", "version_matched"],
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
_version = parse(get_full_version())
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
@receiver(monitoring_set)
 | 
					@receiver(monitoring_set)
 | 
				
			||||||
def monitoring_set_workers(sender, **kwargs):
 | 
					def monitoring_set_workers(sender, **kwargs):
 | 
				
			||||||
    """Set worker gauge"""
 | 
					    """Set worker gauge"""
 | 
				
			||||||
    raw: list[dict[str, dict]] = CELERY_APP.control.ping(timeout=0.5)
 | 
					    count = len(CELERY_APP.control.ping(timeout=0.5))
 | 
				
			||||||
    worker_version_count = {}
 | 
					    GAUGE_WORKERS.set(count)
 | 
				
			||||||
    for worker in raw:
 | 
					 | 
				
			||||||
        key = list(worker.keys())[0]
 | 
					 | 
				
			||||||
        version = worker[key].get("version")
 | 
					 | 
				
			||||||
        version_matching = False
 | 
					 | 
				
			||||||
        if version:
 | 
					 | 
				
			||||||
            version_matching = parse(version) == _version
 | 
					 | 
				
			||||||
        worker_version_count.setdefault(version, {"count": 0, "matching": version_matching})
 | 
					 | 
				
			||||||
        worker_version_count[version]["count"] += 1
 | 
					 | 
				
			||||||
    for version, stats in worker_version_count.items():
 | 
					 | 
				
			||||||
        GAUGE_WORKERS.labels(version, stats["matching"]).set(stats["count"])
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -34,7 +34,7 @@ class TestAdminAPI(TestCase):
 | 
				
			|||||||
        response = self.client.get(reverse("authentik_api:admin_workers"))
 | 
					        response = self.client.get(reverse("authentik_api:admin_workers"))
 | 
				
			||||||
        self.assertEqual(response.status_code, 200)
 | 
					        self.assertEqual(response.status_code, 200)
 | 
				
			||||||
        body = loads(response.content)
 | 
					        body = loads(response.content)
 | 
				
			||||||
        self.assertEqual(len(body), 0)
 | 
					        self.assertEqual(body["count"], 0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def test_metrics(self):
 | 
					    def test_metrics(self):
 | 
				
			||||||
        """Test metrics API"""
 | 
					        """Test metrics API"""
 | 
				
			||||||
 | 
				
			|||||||
							
								
								
									
										67
									
								
								authentik/api/authorization.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								authentik/api/authorization.py
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,67 @@
 | 
				
			|||||||
 | 
					"""API Authorization"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from django.conf import settings
 | 
				
			||||||
 | 
					from django.db.models import Model
 | 
				
			||||||
 | 
					from django.db.models.query import QuerySet
 | 
				
			||||||
 | 
					from django_filters.rest_framework import DjangoFilterBackend
 | 
				
			||||||
 | 
					from rest_framework.authentication import get_authorization_header
 | 
				
			||||||
 | 
					from rest_framework.filters import BaseFilterBackend
 | 
				
			||||||
 | 
					from rest_framework.permissions import BasePermission
 | 
				
			||||||
 | 
					from rest_framework.request import Request
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authentication import validate_auth
 | 
				
			||||||
 | 
					from authentik.rbac.filters import ObjectFilter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class OwnerFilter(BaseFilterBackend):
 | 
				
			||||||
 | 
					    """Filter objects by their owner"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    owner_key = "user"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def filter_queryset(self, request: Request, queryset: QuerySet, view) -> QuerySet:
 | 
				
			||||||
 | 
					        if request.user.is_superuser:
 | 
				
			||||||
 | 
					            return queryset
 | 
				
			||||||
 | 
					        return queryset.filter(**{self.owner_key: request.user})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class SecretKeyFilter(DjangoFilterBackend):
 | 
				
			||||||
 | 
					    """Allow access to all objects when authenticated with secret key as token.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    Replaces both DjangoFilterBackend and ObjectFilter"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def filter_queryset(self, request: Request, queryset: QuerySet, view) -> QuerySet:
 | 
				
			||||||
 | 
					        auth_header = get_authorization_header(request)
 | 
				
			||||||
 | 
					        token = validate_auth(auth_header)
 | 
				
			||||||
 | 
					        if token and token == settings.SECRET_KEY:
 | 
				
			||||||
 | 
					            return queryset
 | 
				
			||||||
 | 
					        queryset = ObjectFilter().filter_queryset(request, queryset, view)
 | 
				
			||||||
 | 
					        return super().filter_queryset(request, queryset, view)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class OwnerPermissions(BasePermission):
 | 
				
			||||||
 | 
					    """Authorize requests by an object's owner matching the requesting user"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    owner_key = "user"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def has_permission(self, request: Request, view) -> bool:
 | 
				
			||||||
 | 
					        """If the user is authenticated, we allow all requests here. For listing, the
 | 
				
			||||||
 | 
					        object-level permissions are done by the filter backend"""
 | 
				
			||||||
 | 
					        return request.user.is_authenticated
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def has_object_permission(self, request: Request, view, obj: Model) -> bool:
 | 
				
			||||||
 | 
					        """Check if the object's owner matches the currently logged in user"""
 | 
				
			||||||
 | 
					        if not hasattr(obj, self.owner_key):
 | 
				
			||||||
 | 
					            return False
 | 
				
			||||||
 | 
					        owner = getattr(obj, self.owner_key)
 | 
				
			||||||
 | 
					        if owner != request.user:
 | 
				
			||||||
 | 
					            return False
 | 
				
			||||||
 | 
					        return True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class OwnerSuperuserPermissions(OwnerPermissions):
 | 
				
			||||||
 | 
					    """Similar to OwnerPermissions, except always allow access for superusers"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def has_object_permission(self, request: Request, view, obj: Model) -> bool:
 | 
				
			||||||
 | 
					        if request.user.is_superuser:
 | 
				
			||||||
 | 
					            return True
 | 
				
			||||||
 | 
					        return super().has_object_permission(request, view, obj)
 | 
				
			||||||
@ -7,7 +7,7 @@ API Browser - {{ brand.branding_title }}
 | 
				
			|||||||
{% endblock %}
 | 
					{% endblock %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{% block head %}
 | 
					{% block head %}
 | 
				
			||||||
<script src="{% versioned_script 'dist/standalone/api-browser/index-%v.js' %}" type="module"></script>
 | 
					{% versioned_script "dist/standalone/api-browser/index-%v.js" %}
 | 
				
			||||||
<meta name="theme-color" content="#151515" media="(prefers-color-scheme: light)">
 | 
					<meta name="theme-color" content="#151515" media="(prefers-color-scheme: light)">
 | 
				
			||||||
<meta name="theme-color" content="#151515" media="(prefers-color-scheme: dark)">
 | 
					<meta name="theme-color" content="#151515" media="(prefers-color-scheme: dark)">
 | 
				
			||||||
{% endblock %}
 | 
					{% endblock %}
 | 
				
			||||||
 | 
				
			|||||||
@ -1,68 +0,0 @@
 | 
				
			|||||||
"""Test and debug Blueprints"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import atexit
 | 
					 | 
				
			||||||
import readline
 | 
					 | 
				
			||||||
from pathlib import Path
 | 
					 | 
				
			||||||
from pprint import pformat
 | 
					 | 
				
			||||||
from sys import exit as sysexit
 | 
					 | 
				
			||||||
from textwrap import indent
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.core.management.base import BaseCommand, no_translations
 | 
					 | 
				
			||||||
from structlog.stdlib import get_logger
 | 
					 | 
				
			||||||
from yaml import load
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from authentik.blueprints.v1.common import BlueprintLoader, EntryInvalidError
 | 
					 | 
				
			||||||
from authentik.core.management.commands.shell import get_banner_text
 | 
					 | 
				
			||||||
from authentik.lib.utils.errors import exception_to_string
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
LOGGER = get_logger()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Command(BaseCommand):
 | 
					 | 
				
			||||||
    """Test and debug Blueprints"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    lines = []
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def __init__(self, *args, **kwargs) -> None:
 | 
					 | 
				
			||||||
        super().__init__(*args, **kwargs)
 | 
					 | 
				
			||||||
        histfolder = Path("~").expanduser() / Path(".local/share/authentik")
 | 
					 | 
				
			||||||
        histfolder.mkdir(parents=True, exist_ok=True)
 | 
					 | 
				
			||||||
        histfile = histfolder / Path("blueprint_shell_history")
 | 
					 | 
				
			||||||
        readline.parse_and_bind("tab: complete")
 | 
					 | 
				
			||||||
        readline.parse_and_bind("set editing-mode vi")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        try:
 | 
					 | 
				
			||||||
            readline.read_history_file(str(histfile))
 | 
					 | 
				
			||||||
        except FileNotFoundError:
 | 
					 | 
				
			||||||
            pass
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        atexit.register(readline.write_history_file, str(histfile))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    @no_translations
 | 
					 | 
				
			||||||
    def handle(self, *args, **options):
 | 
					 | 
				
			||||||
        """Interactively debug blueprint files"""
 | 
					 | 
				
			||||||
        self.stdout.write(get_banner_text("Blueprint shell"))
 | 
					 | 
				
			||||||
        self.stdout.write("Type '.eval' to evaluate previously entered statement(s).")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        def do_eval():
 | 
					 | 
				
			||||||
            yaml_input = "\n".join([line for line in self.lines if line])
 | 
					 | 
				
			||||||
            data = load(yaml_input, BlueprintLoader)
 | 
					 | 
				
			||||||
            self.stdout.write(pformat(data))
 | 
					 | 
				
			||||||
            self.lines = []
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        while True:
 | 
					 | 
				
			||||||
            try:
 | 
					 | 
				
			||||||
                line = input("> ")
 | 
					 | 
				
			||||||
                if line == ".eval":
 | 
					 | 
				
			||||||
                    do_eval()
 | 
					 | 
				
			||||||
                else:
 | 
					 | 
				
			||||||
                    self.lines.append(line)
 | 
					 | 
				
			||||||
            except EntryInvalidError as exc:
 | 
					 | 
				
			||||||
                self.stdout.write("Failed to evaluate expression:")
 | 
					 | 
				
			||||||
                self.stdout.write(indent(exception_to_string(exc), prefix="  "))
 | 
					 | 
				
			||||||
            except EOFError:
 | 
					 | 
				
			||||||
                break
 | 
					 | 
				
			||||||
            except KeyboardInterrupt:
 | 
					 | 
				
			||||||
                self.stdout.write()
 | 
					 | 
				
			||||||
                sysexit(0)
 | 
					 | 
				
			||||||
        self.stdout.write()
 | 
					 | 
				
			||||||
@ -126,7 +126,7 @@ class Command(BaseCommand):
 | 
				
			|||||||
        def_name_perm = f"model_{model_path}_permissions"
 | 
					        def_name_perm = f"model_{model_path}_permissions"
 | 
				
			||||||
        def_path_perm = f"#/$defs/{def_name_perm}"
 | 
					        def_path_perm = f"#/$defs/{def_name_perm}"
 | 
				
			||||||
        self.schema["$defs"][def_name_perm] = self.model_permissions(model)
 | 
					        self.schema["$defs"][def_name_perm] = self.model_permissions(model)
 | 
				
			||||||
        template = {
 | 
					        return {
 | 
				
			||||||
            "type": "object",
 | 
					            "type": "object",
 | 
				
			||||||
            "required": ["model", "identifiers"],
 | 
					            "required": ["model", "identifiers"],
 | 
				
			||||||
            "properties": {
 | 
					            "properties": {
 | 
				
			||||||
@ -143,11 +143,6 @@ class Command(BaseCommand):
 | 
				
			|||||||
                "identifiers": {"$ref": def_path},
 | 
					                "identifiers": {"$ref": def_path},
 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        # Meta models don't require identifiers, as there's no matching database model to find
 | 
					 | 
				
			||||||
        if issubclass(model, BaseMetaModel):
 | 
					 | 
				
			||||||
            del template["properties"]["identifiers"]
 | 
					 | 
				
			||||||
            template["required"].remove("identifiers")
 | 
					 | 
				
			||||||
        return template
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def field_to_jsonschema(self, field: Field) -> dict:
 | 
					    def field_to_jsonschema(self, field: Field) -> dict:
 | 
				
			||||||
        """Convert a single field to json schema"""
 | 
					        """Convert a single field to json schema"""
 | 
				
			||||||
 | 
				
			|||||||
@ -146,10 +146,6 @@ entries:
 | 
				
			|||||||
                  ]
 | 
					                  ]
 | 
				
			||||||
              ]
 | 
					              ]
 | 
				
			||||||
              nested_context: !Context context2
 | 
					              nested_context: !Context context2
 | 
				
			||||||
              at_index_sequence: !AtIndex [!Context sequence, 0]
 | 
					 | 
				
			||||||
              at_index_sequence_default: !AtIndex [!Context sequence, 100, "non existent"]
 | 
					 | 
				
			||||||
              at_index_mapping: !AtIndex [!Context mapping, "key2"]
 | 
					 | 
				
			||||||
              at_index_mapping_default: !AtIndex [!Context mapping, "invalid", "non existent"]
 | 
					 | 
				
			||||||
      identifiers:
 | 
					      identifiers:
 | 
				
			||||||
          name: test
 | 
					          name: test
 | 
				
			||||||
      conditions:
 | 
					      conditions:
 | 
				
			||||||
 | 
				
			|||||||
@ -27,8 +27,7 @@ def blueprint_tester(file_name: Path) -> Callable:
 | 
				
			|||||||
        base = Path("blueprints/")
 | 
					        base = Path("blueprints/")
 | 
				
			||||||
        rel_path = Path(file_name).relative_to(base)
 | 
					        rel_path = Path(file_name).relative_to(base)
 | 
				
			||||||
        importer = Importer.from_string(BlueprintInstance(path=str(rel_path)).retrieve())
 | 
					        importer = Importer.from_string(BlueprintInstance(path=str(rel_path)).retrieve())
 | 
				
			||||||
        validation, logs = importer.validate()
 | 
					        self.assertTrue(importer.validate()[0])
 | 
				
			||||||
        self.assertTrue(validation, logs)
 | 
					 | 
				
			||||||
        self.assertTrue(importer.apply())
 | 
					        self.assertTrue(importer.apply())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    return tester
 | 
					    return tester
 | 
				
			||||||
 | 
				
			|||||||
@ -215,10 +215,6 @@ class TestBlueprintsV1(TransactionTestCase):
 | 
				
			|||||||
                    },
 | 
					                    },
 | 
				
			||||||
                    "nested_context": "context-nested-value",
 | 
					                    "nested_context": "context-nested-value",
 | 
				
			||||||
                    "env_null": None,
 | 
					                    "env_null": None,
 | 
				
			||||||
                    "at_index_sequence": "foo",
 | 
					 | 
				
			||||||
                    "at_index_sequence_default": "non existent",
 | 
					 | 
				
			||||||
                    "at_index_mapping": 2,
 | 
					 | 
				
			||||||
                    "at_index_mapping_default": "non existent",
 | 
					 | 
				
			||||||
                }
 | 
					                }
 | 
				
			||||||
            ).exists()
 | 
					            ).exists()
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
 | 
				
			|||||||
@ -24,10 +24,6 @@ from authentik.lib.sentry import SentryIgnoredException
 | 
				
			|||||||
from authentik.policies.models import PolicyBindingModel
 | 
					from authentik.policies.models import PolicyBindingModel
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class UNSET:
 | 
					 | 
				
			||||||
    """Used to test whether a key has not been set."""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def get_attrs(obj: SerializerModel) -> dict[str, Any]:
 | 
					def get_attrs(obj: SerializerModel) -> dict[str, Any]:
 | 
				
			||||||
    """Get object's attributes via their serializer, and convert it to a normal dict"""
 | 
					    """Get object's attributes via their serializer, and convert it to a normal dict"""
 | 
				
			||||||
    serializer: Serializer = obj.serializer(obj)
 | 
					    serializer: Serializer = obj.serializer(obj)
 | 
				
			||||||
@ -202,9 +198,6 @@ class Blueprint:
 | 
				
			|||||||
class YAMLTag:
 | 
					class YAMLTag:
 | 
				
			||||||
    """Base class for all YAML Tags"""
 | 
					    """Base class for all YAML Tags"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def __repr__(self) -> str:
 | 
					 | 
				
			||||||
        return str(self.resolve(BlueprintEntry(""), Blueprint()))
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def resolve(self, entry: BlueprintEntry, blueprint: Blueprint) -> Any:
 | 
					    def resolve(self, entry: BlueprintEntry, blueprint: Blueprint) -> Any:
 | 
				
			||||||
        """Implement yaml tag logic"""
 | 
					        """Implement yaml tag logic"""
 | 
				
			||||||
        raise NotImplementedError
 | 
					        raise NotImplementedError
 | 
				
			||||||
@ -563,53 +556,6 @@ class Value(EnumeratedItem):
 | 
				
			|||||||
            raise EntryInvalidError.from_entry(f"Empty/invalid context: {context}", entry) from exc
 | 
					            raise EntryInvalidError.from_entry(f"Empty/invalid context: {context}", entry) from exc
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class AtIndex(YAMLTag):
 | 
					 | 
				
			||||||
    """Get value at index of a sequence or mapping"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    obj: YAMLTag | dict | list | tuple
 | 
					 | 
				
			||||||
    attribute: int | str | YAMLTag
 | 
					 | 
				
			||||||
    default: Any | UNSET
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def __init__(self, loader: "BlueprintLoader", node: SequenceNode) -> None:
 | 
					 | 
				
			||||||
        super().__init__()
 | 
					 | 
				
			||||||
        self.obj = loader.construct_object(node.value[0])
 | 
					 | 
				
			||||||
        self.attribute = loader.construct_object(node.value[1])
 | 
					 | 
				
			||||||
        if len(node.value) == 2:  # noqa: PLR2004
 | 
					 | 
				
			||||||
            self.default = UNSET
 | 
					 | 
				
			||||||
        else:
 | 
					 | 
				
			||||||
            self.default = loader.construct_object(node.value[2])
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def resolve(self, entry: BlueprintEntry, blueprint: Blueprint) -> Any:
 | 
					 | 
				
			||||||
        if isinstance(self.obj, YAMLTag):
 | 
					 | 
				
			||||||
            obj = self.obj.resolve(entry, blueprint)
 | 
					 | 
				
			||||||
        else:
 | 
					 | 
				
			||||||
            obj = self.obj
 | 
					 | 
				
			||||||
        if isinstance(self.attribute, YAMLTag):
 | 
					 | 
				
			||||||
            attribute = self.attribute.resolve(entry, blueprint)
 | 
					 | 
				
			||||||
        else:
 | 
					 | 
				
			||||||
            attribute = self.attribute
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        if isinstance(obj, list | tuple):
 | 
					 | 
				
			||||||
            try:
 | 
					 | 
				
			||||||
                return obj[attribute]
 | 
					 | 
				
			||||||
            except TypeError as exc:
 | 
					 | 
				
			||||||
                raise EntryInvalidError.from_entry(
 | 
					 | 
				
			||||||
                    f"Invalid index for list: {attribute}", entry
 | 
					 | 
				
			||||||
                ) from exc
 | 
					 | 
				
			||||||
            except IndexError as exc:
 | 
					 | 
				
			||||||
                if self.default is UNSET:
 | 
					 | 
				
			||||||
                    raise EntryInvalidError.from_entry(
 | 
					 | 
				
			||||||
                        f"Index out of range: {attribute}", entry
 | 
					 | 
				
			||||||
                    ) from exc
 | 
					 | 
				
			||||||
                return self.default
 | 
					 | 
				
			||||||
        if attribute in obj:
 | 
					 | 
				
			||||||
            return obj[attribute]
 | 
					 | 
				
			||||||
        else:
 | 
					 | 
				
			||||||
            if self.default is UNSET:
 | 
					 | 
				
			||||||
                raise EntryInvalidError.from_entry(f"Key does not exist: {attribute}", entry)
 | 
					 | 
				
			||||||
            return self.default
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class BlueprintDumper(SafeDumper):
 | 
					class BlueprintDumper(SafeDumper):
 | 
				
			||||||
    """Dump dataclasses to yaml"""
 | 
					    """Dump dataclasses to yaml"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -660,7 +606,6 @@ class BlueprintLoader(SafeLoader):
 | 
				
			|||||||
        self.add_constructor("!Enumerate", Enumerate)
 | 
					        self.add_constructor("!Enumerate", Enumerate)
 | 
				
			||||||
        self.add_constructor("!Value", Value)
 | 
					        self.add_constructor("!Value", Value)
 | 
				
			||||||
        self.add_constructor("!Index", Index)
 | 
					        self.add_constructor("!Index", Index)
 | 
				
			||||||
        self.add_constructor("!AtIndex", AtIndex)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class EntryInvalidError(SentryIgnoredException):
 | 
					class EntryInvalidError(SentryIgnoredException):
 | 
				
			||||||
 | 
				
			|||||||
@ -65,12 +65,7 @@ from authentik.lib.utils.reflection import get_apps
 | 
				
			|||||||
from authentik.outposts.models import OutpostServiceConnection
 | 
					from authentik.outposts.models import OutpostServiceConnection
 | 
				
			||||||
from authentik.policies.models import Policy, PolicyBindingModel
 | 
					from authentik.policies.models import Policy, PolicyBindingModel
 | 
				
			||||||
from authentik.policies.reputation.models import Reputation
 | 
					from authentik.policies.reputation.models import Reputation
 | 
				
			||||||
from authentik.providers.oauth2.models import (
 | 
					from authentik.providers.oauth2.models import AccessToken, AuthorizationCode, RefreshToken
 | 
				
			||||||
    AccessToken,
 | 
					 | 
				
			||||||
    AuthorizationCode,
 | 
					 | 
				
			||||||
    DeviceToken,
 | 
					 | 
				
			||||||
    RefreshToken,
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
from authentik.providers.scim.models import SCIMProviderGroup, SCIMProviderUser
 | 
					from authentik.providers.scim.models import SCIMProviderGroup, SCIMProviderUser
 | 
				
			||||||
from authentik.rbac.models import Role
 | 
					from authentik.rbac.models import Role
 | 
				
			||||||
from authentik.sources.scim.models import SCIMSourceGroup, SCIMSourceUser
 | 
					from authentik.sources.scim.models import SCIMSourceGroup, SCIMSourceUser
 | 
				
			||||||
@ -130,7 +125,6 @@ def excluded_models() -> list[type[Model]]:
 | 
				
			|||||||
        MicrosoftEntraProviderGroup,
 | 
					        MicrosoftEntraProviderGroup,
 | 
				
			||||||
        EndpointDevice,
 | 
					        EndpointDevice,
 | 
				
			||||||
        EndpointDeviceConnection,
 | 
					        EndpointDeviceConnection,
 | 
				
			||||||
        DeviceToken,
 | 
					 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -299,11 +293,7 @@ class Importer:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
        serializer_kwargs = {}
 | 
					        serializer_kwargs = {}
 | 
				
			||||||
        model_instance = existing_models.first()
 | 
					        model_instance = existing_models.first()
 | 
				
			||||||
        if (
 | 
					        if not isinstance(model(), BaseMetaModel) and model_instance:
 | 
				
			||||||
            not isinstance(model(), BaseMetaModel)
 | 
					 | 
				
			||||||
            and model_instance
 | 
					 | 
				
			||||||
            and entry.state != BlueprintEntryDesiredState.MUST_CREATED
 | 
					 | 
				
			||||||
        ):
 | 
					 | 
				
			||||||
            self.logger.debug(
 | 
					            self.logger.debug(
 | 
				
			||||||
                "Initialise serializer with instance",
 | 
					                "Initialise serializer with instance",
 | 
				
			||||||
                model=model,
 | 
					                model=model,
 | 
				
			||||||
@ -313,12 +303,11 @@ class Importer:
 | 
				
			|||||||
            serializer_kwargs["instance"] = model_instance
 | 
					            serializer_kwargs["instance"] = model_instance
 | 
				
			||||||
            serializer_kwargs["partial"] = True
 | 
					            serializer_kwargs["partial"] = True
 | 
				
			||||||
        elif model_instance and entry.state == BlueprintEntryDesiredState.MUST_CREATED:
 | 
					        elif model_instance and entry.state == BlueprintEntryDesiredState.MUST_CREATED:
 | 
				
			||||||
            msg = (
 | 
					 | 
				
			||||||
                f"State is set to {BlueprintEntryDesiredState.MUST_CREATED.value} "
 | 
					 | 
				
			||||||
                "and object exists already",
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
            raise EntryInvalidError.from_entry(
 | 
					            raise EntryInvalidError.from_entry(
 | 
				
			||||||
                ValidationError({k: msg for k in entry.identifiers.keys()}, "unique"),
 | 
					                (
 | 
				
			||||||
 | 
					                    f"State is set to {BlueprintEntryDesiredState.MUST_CREATED} "
 | 
				
			||||||
 | 
					                    "and object exists already",
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
                entry,
 | 
					                entry,
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
        else:
 | 
					        else:
 | 
				
			||||||
 | 
				
			|||||||
@ -159,7 +159,7 @@ def blueprints_discovery(self: SystemTask, path: str | None = None):
 | 
				
			|||||||
        check_blueprint_v1_file(blueprint)
 | 
					        check_blueprint_v1_file(blueprint)
 | 
				
			||||||
        count += 1
 | 
					        count += 1
 | 
				
			||||||
    self.set_status(
 | 
					    self.set_status(
 | 
				
			||||||
        TaskStatus.SUCCESSFUL, _("Successfully imported {count} files.".format(count=count))
 | 
					        TaskStatus.SUCCESSFUL, _("Successfully imported %(count)d files." % {"count": count})
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -14,10 +14,10 @@ from rest_framework.response import Response
 | 
				
			|||||||
from rest_framework.validators import UniqueValidator
 | 
					from rest_framework.validators import UniqueValidator
 | 
				
			||||||
from rest_framework.viewsets import ModelViewSet
 | 
					from rest_framework.viewsets import ModelViewSet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import SecretKeyFilter
 | 
				
			||||||
from authentik.brands.models import Brand
 | 
					from authentik.brands.models import Brand
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
from authentik.core.api.utils import ModelSerializer, PassiveSerializer
 | 
					from authentik.core.api.utils import ModelSerializer, PassiveSerializer
 | 
				
			||||||
from authentik.rbac.filters import SecretKeyFilter
 | 
					 | 
				
			||||||
from authentik.tenants.utils import get_current_tenant
 | 
					from authentik.tenants.utils import get_current_tenant
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -84,8 +84,8 @@ class CurrentBrandSerializer(PassiveSerializer):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    matched_domain = CharField(source="domain")
 | 
					    matched_domain = CharField(source="domain")
 | 
				
			||||||
    branding_title = CharField()
 | 
					    branding_title = CharField()
 | 
				
			||||||
    branding_logo = CharField(source="branding_logo_url")
 | 
					    branding_logo = CharField()
 | 
				
			||||||
    branding_favicon = CharField(source="branding_favicon_url")
 | 
					    branding_favicon = CharField()
 | 
				
			||||||
    ui_footer_links = ListField(
 | 
					    ui_footer_links = ListField(
 | 
				
			||||||
        child=FooterLinkSerializer(),
 | 
					        child=FooterLinkSerializer(),
 | 
				
			||||||
        read_only=True,
 | 
					        read_only=True,
 | 
				
			||||||
 | 
				
			|||||||
@ -4,7 +4,7 @@ from collections.abc import Callable
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from django.http.request import HttpRequest
 | 
					from django.http.request import HttpRequest
 | 
				
			||||||
from django.http.response import HttpResponse
 | 
					from django.http.response import HttpResponse
 | 
				
			||||||
from django.utils.translation import override
 | 
					from django.utils.translation import activate
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik.brands.utils import get_brand_for_request
 | 
					from authentik.brands.utils import get_brand_for_request
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -18,14 +18,10 @@ class BrandMiddleware:
 | 
				
			|||||||
        self.get_response = get_response
 | 
					        self.get_response = get_response
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def __call__(self, request: HttpRequest) -> HttpResponse:
 | 
					    def __call__(self, request: HttpRequest) -> HttpResponse:
 | 
				
			||||||
        locale_to_set = None
 | 
					 | 
				
			||||||
        if not hasattr(request, "brand"):
 | 
					        if not hasattr(request, "brand"):
 | 
				
			||||||
            brand = get_brand_for_request(request)
 | 
					            brand = get_brand_for_request(request)
 | 
				
			||||||
            request.brand = brand
 | 
					            request.brand = brand
 | 
				
			||||||
            locale = brand.default_locale
 | 
					            locale = brand.default_locale
 | 
				
			||||||
            if locale != "":
 | 
					            if locale != "":
 | 
				
			||||||
                locale_to_set = locale
 | 
					                activate(locale)
 | 
				
			||||||
        if locale_to_set:
 | 
					 | 
				
			||||||
            with override(locale_to_set):
 | 
					 | 
				
			||||||
                return self.get_response(request)
 | 
					 | 
				
			||||||
        return self.get_response(request)
 | 
					        return self.get_response(request)
 | 
				
			||||||
 | 
				
			|||||||
@ -10,7 +10,6 @@ from structlog.stdlib import get_logger
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from authentik.crypto.models import CertificateKeyPair
 | 
					from authentik.crypto.models import CertificateKeyPair
 | 
				
			||||||
from authentik.flows.models import Flow
 | 
					from authentik.flows.models import Flow
 | 
				
			||||||
from authentik.lib.config import CONFIG
 | 
					 | 
				
			||||||
from authentik.lib.models import SerializerModel
 | 
					from authentik.lib.models import SerializerModel
 | 
				
			||||||
 | 
					
 | 
				
			||||||
LOGGER = get_logger()
 | 
					LOGGER = get_logger()
 | 
				
			||||||
@ -72,18 +71,6 @@ class Brand(SerializerModel):
 | 
				
			|||||||
    )
 | 
					    )
 | 
				
			||||||
    attributes = models.JSONField(default=dict, blank=True)
 | 
					    attributes = models.JSONField(default=dict, blank=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def branding_logo_url(self) -> str:
 | 
					 | 
				
			||||||
        """Get branding_logo with the correct prefix"""
 | 
					 | 
				
			||||||
        if self.branding_logo.startswith("/static"):
 | 
					 | 
				
			||||||
            return CONFIG.get("web.path", "/")[:-1] + self.branding_logo
 | 
					 | 
				
			||||||
        return self.branding_logo
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def branding_favicon_url(self) -> str:
 | 
					 | 
				
			||||||
        """Get branding_favicon with the correct prefix"""
 | 
					 | 
				
			||||||
        if self.branding_favicon.startswith("/static"):
 | 
					 | 
				
			||||||
            return CONFIG.get("web.path", "/")[:-1] + self.branding_favicon
 | 
					 | 
				
			||||||
        return self.branding_favicon
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    @property
 | 
					    @property
 | 
				
			||||||
    def serializer(self) -> Serializer:
 | 
					    def serializer(self) -> Serializer:
 | 
				
			||||||
        from authentik.brands.api import BrandSerializer
 | 
					        from authentik.brands.api import BrandSerializer
 | 
				
			||||||
 | 
				
			|||||||
@ -1,58 +0,0 @@
 | 
				
			|||||||
"""Application Roles API Viewset"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.http import HttpRequest
 | 
					 | 
				
			||||||
from django.utils.translation import gettext_lazy as _
 | 
					 | 
				
			||||||
from rest_framework.exceptions import ValidationError
 | 
					 | 
				
			||||||
from rest_framework.viewsets import ModelViewSet
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from authentik.blueprints.v1.importer import SERIALIZER_CONTEXT_BLUEPRINT
 | 
					 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					 | 
				
			||||||
from authentik.core.api.utils import ModelSerializer
 | 
					 | 
				
			||||||
from authentik.core.models import (
 | 
					 | 
				
			||||||
    Application,
 | 
					 | 
				
			||||||
    ApplicationEntitlement,
 | 
					 | 
				
			||||||
)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class ApplicationEntitlementSerializer(ModelSerializer):
 | 
					 | 
				
			||||||
    """ApplicationEntitlement Serializer"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def validate_app(self, app: Application) -> Application:
 | 
					 | 
				
			||||||
        """Ensure user has permission to view"""
 | 
					 | 
				
			||||||
        request: HttpRequest = self.context.get("request")
 | 
					 | 
				
			||||||
        if not request and SERIALIZER_CONTEXT_BLUEPRINT in self.context:
 | 
					 | 
				
			||||||
            return app
 | 
					 | 
				
			||||||
        user = request.user
 | 
					 | 
				
			||||||
        if user.has_perm("view_application", app) or user.has_perm(
 | 
					 | 
				
			||||||
            "authentik_core.view_application"
 | 
					 | 
				
			||||||
        ):
 | 
					 | 
				
			||||||
            return app
 | 
					 | 
				
			||||||
        raise ValidationError(_("User does not have access to application."), code="invalid")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    class Meta:
 | 
					 | 
				
			||||||
        model = ApplicationEntitlement
 | 
					 | 
				
			||||||
        fields = [
 | 
					 | 
				
			||||||
            "pbm_uuid",
 | 
					 | 
				
			||||||
            "name",
 | 
					 | 
				
			||||||
            "app",
 | 
					 | 
				
			||||||
            "attributes",
 | 
					 | 
				
			||||||
        ]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class ApplicationEntitlementViewSet(UsedByMixin, ModelViewSet):
 | 
					 | 
				
			||||||
    """ApplicationEntitlement Viewset"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    queryset = ApplicationEntitlement.objects.all()
 | 
					 | 
				
			||||||
    serializer_class = ApplicationEntitlementSerializer
 | 
					 | 
				
			||||||
    search_fields = [
 | 
					 | 
				
			||||||
        "pbm_uuid",
 | 
					 | 
				
			||||||
        "name",
 | 
					 | 
				
			||||||
        "app",
 | 
					 | 
				
			||||||
        "attributes",
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
    filterset_fields = [
 | 
					 | 
				
			||||||
        "pbm_uuid",
 | 
					 | 
				
			||||||
        "name",
 | 
					 | 
				
			||||||
        "app",
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
    ordering = ["name"]
 | 
					 | 
				
			||||||
@ -209,7 +209,7 @@ class ApplicationViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
    @extend_schema(
 | 
					    @extend_schema(
 | 
				
			||||||
        parameters=[
 | 
					        parameters=[
 | 
				
			||||||
            OpenApiParameter(
 | 
					            OpenApiParameter(
 | 
				
			||||||
                name="list_rbac",
 | 
					                name="superuser_full_list",
 | 
				
			||||||
                location=OpenApiParameter.QUERY,
 | 
					                location=OpenApiParameter.QUERY,
 | 
				
			||||||
                type=OpenApiTypes.BOOL,
 | 
					                type=OpenApiTypes.BOOL,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
@ -229,8 +229,10 @@ class ApplicationViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
        """Custom list method that checks Policy based access instead of guardian"""
 | 
					        """Custom list method that checks Policy based access instead of guardian"""
 | 
				
			||||||
        should_cache = request.query_params.get("search", "") == ""
 | 
					        should_cache = request.query_params.get("search", "") == ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        list_rbac = str(request.query_params.get("list_rbac", "false")).lower() == "true"
 | 
					        superuser_full_list = (
 | 
				
			||||||
        if list_rbac:
 | 
					            str(request.query_params.get("superuser_full_list", "false")).lower() == "true"
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					        if superuser_full_list and request.user.is_superuser:
 | 
				
			||||||
            return super().list(request)
 | 
					            return super().list(request)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        only_with_launch_url = str(
 | 
					        only_with_launch_url = str(
 | 
				
			||||||
 | 
				
			|||||||
@ -2,12 +2,16 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from typing import TypedDict
 | 
					from typing import TypedDict
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from django_filters.rest_framework import DjangoFilterBackend
 | 
				
			||||||
 | 
					from guardian.utils import get_anonymous_user
 | 
				
			||||||
from rest_framework import mixins
 | 
					from rest_framework import mixins
 | 
				
			||||||
from rest_framework.fields import SerializerMethodField
 | 
					from rest_framework.fields import SerializerMethodField
 | 
				
			||||||
 | 
					from rest_framework.filters import OrderingFilter, SearchFilter
 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
from rest_framework.viewsets import GenericViewSet
 | 
					from rest_framework.viewsets import GenericViewSet
 | 
				
			||||||
from ua_parser import user_agent_parser
 | 
					from ua_parser import user_agent_parser
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import OwnerSuperuserPermissions
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
from authentik.core.api.utils import ModelSerializer
 | 
					from authentik.core.api.utils import ModelSerializer
 | 
				
			||||||
from authentik.core.models import AuthenticatedSession
 | 
					from authentik.core.models import AuthenticatedSession
 | 
				
			||||||
@ -106,4 +110,11 @@ class AuthenticatedSessionViewSet(
 | 
				
			|||||||
    search_fields = ["user__username", "last_ip", "last_user_agent"]
 | 
					    search_fields = ["user__username", "last_ip", "last_user_agent"]
 | 
				
			||||||
    filterset_fields = ["user__username", "last_ip", "last_user_agent"]
 | 
					    filterset_fields = ["user__username", "last_ip", "last_user_agent"]
 | 
				
			||||||
    ordering = ["user__username"]
 | 
					    ordering = ["user__username"]
 | 
				
			||||||
    owner_field = "user"
 | 
					    permission_classes = [OwnerSuperuserPermissions]
 | 
				
			||||||
 | 
					    filter_backends = [DjangoFilterBackend, OrderingFilter, SearchFilter]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_queryset(self):
 | 
				
			||||||
 | 
					        user = self.request.user if self.request else get_anonymous_user()
 | 
				
			||||||
 | 
					        if user.is_superuser:
 | 
				
			||||||
 | 
					            return super().get_queryset()
 | 
				
			||||||
 | 
					        return super().get_queryset().filter(user=user.pk)
 | 
				
			||||||
 | 
				
			|||||||
@ -1,6 +1,5 @@
 | 
				
			|||||||
"""Authenticator Devices API Views"""
 | 
					"""Authenticator Devices API Views"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from django.utils.translation import gettext_lazy as _
 | 
					 | 
				
			||||||
from drf_spectacular.types import OpenApiTypes
 | 
					from drf_spectacular.types import OpenApiTypes
 | 
				
			||||||
from drf_spectacular.utils import OpenApiParameter, extend_schema
 | 
					from drf_spectacular.utils import OpenApiParameter, extend_schema
 | 
				
			||||||
from rest_framework.fields import (
 | 
					from rest_framework.fields import (
 | 
				
			||||||
@ -41,11 +40,7 @@ class DeviceSerializer(MetaNameSerializer):
 | 
				
			|||||||
    def get_extra_description(self, instance: Device) -> str:
 | 
					    def get_extra_description(self, instance: Device) -> str:
 | 
				
			||||||
        """Get extra description"""
 | 
					        """Get extra description"""
 | 
				
			||||||
        if isinstance(instance, WebAuthnDevice):
 | 
					        if isinstance(instance, WebAuthnDevice):
 | 
				
			||||||
            return (
 | 
					            return instance.device_type.description
 | 
				
			||||||
                instance.device_type.description
 | 
					 | 
				
			||||||
                if instance.device_type
 | 
					 | 
				
			||||||
                else _("Extra description not available")
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
        if isinstance(instance, EndpointDevice):
 | 
					        if isinstance(instance, EndpointDevice):
 | 
				
			||||||
            return instance.data.get("deviceSignals", {}).get("deviceModel")
 | 
					            return instance.data.get("deviceSignals", {}).get("deviceModel")
 | 
				
			||||||
        return ""
 | 
					        return ""
 | 
				
			||||||
 | 
				
			|||||||
@ -2,16 +2,19 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from collections.abc import Iterable
 | 
					from collections.abc import Iterable
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from django_filters.rest_framework import DjangoFilterBackend
 | 
				
			||||||
from drf_spectacular.utils import OpenApiResponse, extend_schema
 | 
					from drf_spectacular.utils import OpenApiResponse, extend_schema
 | 
				
			||||||
from rest_framework import mixins
 | 
					from rest_framework import mixins
 | 
				
			||||||
from rest_framework.decorators import action
 | 
					from rest_framework.decorators import action
 | 
				
			||||||
from rest_framework.fields import CharField, ReadOnlyField, SerializerMethodField
 | 
					from rest_framework.fields import CharField, ReadOnlyField, SerializerMethodField
 | 
				
			||||||
 | 
					from rest_framework.filters import OrderingFilter, SearchFilter
 | 
				
			||||||
from rest_framework.parsers import MultiPartParser
 | 
					from rest_framework.parsers import MultiPartParser
 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
from rest_framework.response import Response
 | 
					from rest_framework.response import Response
 | 
				
			||||||
from rest_framework.viewsets import GenericViewSet
 | 
					from rest_framework.viewsets import GenericViewSet
 | 
				
			||||||
from structlog.stdlib import get_logger
 | 
					from structlog.stdlib import get_logger
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import OwnerFilter, OwnerSuperuserPermissions
 | 
				
			||||||
from authentik.blueprints.v1.importer import SERIALIZER_CONTEXT_BLUEPRINT
 | 
					from authentik.blueprints.v1.importer import SERIALIZER_CONTEXT_BLUEPRINT
 | 
				
			||||||
from authentik.core.api.object_types import TypesMixin
 | 
					from authentik.core.api.object_types import TypesMixin
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
@ -156,9 +159,9 @@ class SourceViewSet(
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class UserSourceConnectionSerializer(SourceSerializer):
 | 
					class UserSourceConnectionSerializer(SourceSerializer):
 | 
				
			||||||
    """User source connection"""
 | 
					    """OAuth Source Serializer"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    source_obj = SourceSerializer(read_only=True, source="source")
 | 
					    source = SourceSerializer(read_only=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        model = UserSourceConnection
 | 
					        model = UserSourceConnection
 | 
				
			||||||
@ -166,10 +169,10 @@ class UserSourceConnectionSerializer(SourceSerializer):
 | 
				
			|||||||
            "pk",
 | 
					            "pk",
 | 
				
			||||||
            "user",
 | 
					            "user",
 | 
				
			||||||
            "source",
 | 
					            "source",
 | 
				
			||||||
            "source_obj",
 | 
					 | 
				
			||||||
            "created",
 | 
					            "created",
 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
        extra_kwargs = {
 | 
					        extra_kwargs = {
 | 
				
			||||||
 | 
					            "user": {"read_only": True},
 | 
				
			||||||
            "created": {"read_only": True},
 | 
					            "created": {"read_only": True},
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -186,16 +189,17 @@ class UserSourceConnectionViewSet(
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    queryset = UserSourceConnection.objects.all()
 | 
					    queryset = UserSourceConnection.objects.all()
 | 
				
			||||||
    serializer_class = UserSourceConnectionSerializer
 | 
					    serializer_class = UserSourceConnectionSerializer
 | 
				
			||||||
 | 
					    permission_classes = [OwnerSuperuserPermissions]
 | 
				
			||||||
    filterset_fields = ["user", "source__slug"]
 | 
					    filterset_fields = ["user", "source__slug"]
 | 
				
			||||||
    search_fields = ["source__slug"]
 | 
					    search_fields = ["source__slug"]
 | 
				
			||||||
 | 
					    filter_backends = [OwnerFilter, DjangoFilterBackend, OrderingFilter, SearchFilter]
 | 
				
			||||||
    ordering = ["source__slug", "pk"]
 | 
					    ordering = ["source__slug", "pk"]
 | 
				
			||||||
    owner_field = "user"
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class GroupSourceConnectionSerializer(SourceSerializer):
 | 
					class GroupSourceConnectionSerializer(SourceSerializer):
 | 
				
			||||||
    """Group Source Connection"""
 | 
					    """Group Source Connection Serializer"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    source_obj = SourceSerializer(read_only=True)
 | 
					    source = SourceSerializer(read_only=True)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        model = GroupSourceConnection
 | 
					        model = GroupSourceConnection
 | 
				
			||||||
@ -203,11 +207,12 @@ class GroupSourceConnectionSerializer(SourceSerializer):
 | 
				
			|||||||
            "pk",
 | 
					            "pk",
 | 
				
			||||||
            "group",
 | 
					            "group",
 | 
				
			||||||
            "source",
 | 
					            "source",
 | 
				
			||||||
            "source_obj",
 | 
					 | 
				
			||||||
            "identifier",
 | 
					            "identifier",
 | 
				
			||||||
            "created",
 | 
					            "created",
 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
        extra_kwargs = {
 | 
					        extra_kwargs = {
 | 
				
			||||||
 | 
					            "group": {"read_only": True},
 | 
				
			||||||
 | 
					            "identifier": {"read_only": True},
 | 
				
			||||||
            "created": {"read_only": True},
 | 
					            "created": {"read_only": True},
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -224,7 +229,8 @@ class GroupSourceConnectionViewSet(
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    queryset = GroupSourceConnection.objects.all()
 | 
					    queryset = GroupSourceConnection.objects.all()
 | 
				
			||||||
    serializer_class = GroupSourceConnectionSerializer
 | 
					    serializer_class = GroupSourceConnectionSerializer
 | 
				
			||||||
 | 
					    permission_classes = [OwnerSuperuserPermissions]
 | 
				
			||||||
    filterset_fields = ["group", "source__slug"]
 | 
					    filterset_fields = ["group", "source__slug"]
 | 
				
			||||||
    search_fields = ["source__slug"]
 | 
					    search_fields = ["source__slug"]
 | 
				
			||||||
 | 
					    filter_backends = [OwnerFilter, DjangoFilterBackend, OrderingFilter, SearchFilter]
 | 
				
			||||||
    ordering = ["source__slug", "pk"]
 | 
					    ordering = ["source__slug", "pk"]
 | 
				
			||||||
    owner_field = "user"
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -3,15 +3,18 @@
 | 
				
			|||||||
from typing import Any
 | 
					from typing import Any
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from django.utils.timezone import now
 | 
					from django.utils.timezone import now
 | 
				
			||||||
 | 
					from django_filters.rest_framework import DjangoFilterBackend
 | 
				
			||||||
from drf_spectacular.utils import OpenApiResponse, extend_schema, inline_serializer
 | 
					from drf_spectacular.utils import OpenApiResponse, extend_schema, inline_serializer
 | 
				
			||||||
from guardian.shortcuts import assign_perm
 | 
					from guardian.shortcuts import assign_perm, get_anonymous_user
 | 
				
			||||||
from rest_framework.decorators import action
 | 
					from rest_framework.decorators import action
 | 
				
			||||||
from rest_framework.exceptions import ValidationError
 | 
					from rest_framework.exceptions import ValidationError
 | 
				
			||||||
from rest_framework.fields import CharField
 | 
					from rest_framework.fields import CharField
 | 
				
			||||||
 | 
					from rest_framework.filters import OrderingFilter, SearchFilter
 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
from rest_framework.response import Response
 | 
					from rest_framework.response import Response
 | 
				
			||||||
from rest_framework.viewsets import ModelViewSet
 | 
					from rest_framework.viewsets import ModelViewSet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import OwnerSuperuserPermissions
 | 
				
			||||||
from authentik.blueprints.api import ManagedSerializer
 | 
					from authentik.blueprints.api import ManagedSerializer
 | 
				
			||||||
from authentik.blueprints.v1.importer import SERIALIZER_CONTEXT_BLUEPRINT
 | 
					from authentik.blueprints.v1.importer import SERIALIZER_CONTEXT_BLUEPRINT
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
@ -135,11 +138,16 @@ class TokenViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
        "managed",
 | 
					        "managed",
 | 
				
			||||||
    ]
 | 
					    ]
 | 
				
			||||||
    ordering = ["identifier", "expires"]
 | 
					    ordering = ["identifier", "expires"]
 | 
				
			||||||
    owner_field = "user"
 | 
					    permission_classes = [OwnerSuperuserPermissions]
 | 
				
			||||||
    rbac_allow_create_without_perm = True
 | 
					    filter_backends = [DjangoFilterBackend, OrderingFilter, SearchFilter]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    def get_queryset(self):
 | 
				
			||||||
 | 
					        user = self.request.user if self.request else get_anonymous_user()
 | 
				
			||||||
 | 
					        if user.is_superuser:
 | 
				
			||||||
 | 
					            return super().get_queryset()
 | 
				
			||||||
 | 
					        return super().get_queryset().filter(user=user.pk)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def perform_create(self, serializer: TokenSerializer):
 | 
					    def perform_create(self, serializer: TokenSerializer):
 | 
				
			||||||
        # TODO: better permission check
 | 
					 | 
				
			||||||
        if not self.request.user.is_superuser:
 | 
					        if not self.request.user.is_superuser:
 | 
				
			||||||
            instance = serializer.save(
 | 
					            instance = serializer.save(
 | 
				
			||||||
                user=self.request.user,
 | 
					                user=self.request.user,
 | 
				
			||||||
 | 
				
			|||||||
@ -1,12 +1,10 @@
 | 
				
			|||||||
"""transactional application and provider creation"""
 | 
					"""transactional application and provider creation"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from django.apps import apps
 | 
					from django.apps import apps
 | 
				
			||||||
from django.db.models import Model
 | 
					 | 
				
			||||||
from django.utils.translation import gettext as _
 | 
					 | 
				
			||||||
from drf_spectacular.utils import PolymorphicProxySerializer, extend_schema, extend_schema_field
 | 
					from drf_spectacular.utils import PolymorphicProxySerializer, extend_schema, extend_schema_field
 | 
				
			||||||
from rest_framework.exceptions import PermissionDenied, ValidationError
 | 
					from rest_framework.exceptions import ValidationError
 | 
				
			||||||
from rest_framework.fields import BooleanField, CharField, ChoiceField, DictField, ListField
 | 
					from rest_framework.fields import BooleanField, CharField, ChoiceField, DictField, ListField
 | 
				
			||||||
from rest_framework.permissions import IsAuthenticated
 | 
					from rest_framework.permissions import IsAdminUser
 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
from rest_framework.response import Response
 | 
					from rest_framework.response import Response
 | 
				
			||||||
from rest_framework.views import APIView
 | 
					from rest_framework.views import APIView
 | 
				
			||||||
@ -22,9 +20,8 @@ from authentik.blueprints.v1.common import (
 | 
				
			|||||||
from authentik.blueprints.v1.importer import Importer
 | 
					from authentik.blueprints.v1.importer import Importer
 | 
				
			||||||
from authentik.core.api.applications import ApplicationSerializer
 | 
					from authentik.core.api.applications import ApplicationSerializer
 | 
				
			||||||
from authentik.core.api.utils import PassiveSerializer
 | 
					from authentik.core.api.utils import PassiveSerializer
 | 
				
			||||||
from authentik.core.models import Application, Provider
 | 
					from authentik.core.models import Provider
 | 
				
			||||||
from authentik.lib.utils.reflection import all_subclasses
 | 
					from authentik.lib.utils.reflection import all_subclasses
 | 
				
			||||||
from authentik.policies.api.bindings import PolicyBindingSerializer
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
def get_provider_serializer_mapping():
 | 
					def get_provider_serializer_mapping():
 | 
				
			||||||
@ -48,20 +45,6 @@ class TransactionProviderField(DictField):
 | 
				
			|||||||
    """Dictionary field which can hold provider creation data"""
 | 
					    """Dictionary field which can hold provider creation data"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class TransactionPolicyBindingSerializer(PolicyBindingSerializer):
 | 
					 | 
				
			||||||
    """PolicyBindingSerializer which does not require target as target is set implicitly"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def validate(self, attrs):
 | 
					 | 
				
			||||||
        # As the PolicyBindingSerializer checks that the correct things can be bound to a target
 | 
					 | 
				
			||||||
        # but we don't have a target here as that's set by the blueprint, pass in an empty app
 | 
					 | 
				
			||||||
        # which will have the correct allowed combination of group/user/policy.
 | 
					 | 
				
			||||||
        attrs["target"] = Application()
 | 
					 | 
				
			||||||
        return super().validate(attrs)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    class Meta(PolicyBindingSerializer.Meta):
 | 
					 | 
				
			||||||
        fields = [x for x in PolicyBindingSerializer.Meta.fields if x != "target"]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class TransactionApplicationSerializer(PassiveSerializer):
 | 
					class TransactionApplicationSerializer(PassiveSerializer):
 | 
				
			||||||
    """Serializer for creating a provider and an application in one transaction"""
 | 
					    """Serializer for creating a provider and an application in one transaction"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -69,8 +52,6 @@ class TransactionApplicationSerializer(PassiveSerializer):
 | 
				
			|||||||
    provider_model = ChoiceField(choices=list(get_provider_serializer_mapping().keys()))
 | 
					    provider_model = ChoiceField(choices=list(get_provider_serializer_mapping().keys()))
 | 
				
			||||||
    provider = TransactionProviderField()
 | 
					    provider = TransactionProviderField()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    policy_bindings = TransactionPolicyBindingSerializer(many=True, required=False)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    _provider_model: type[Provider] = None
 | 
					    _provider_model: type[Provider] = None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def validate_provider_model(self, fq_model_name: str) -> str:
 | 
					    def validate_provider_model(self, fq_model_name: str) -> str:
 | 
				
			||||||
@ -115,19 +96,6 @@ class TransactionApplicationSerializer(PassiveSerializer):
 | 
				
			|||||||
                id="app",
 | 
					                id="app",
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        for binding in attrs.get("policy_bindings", []):
 | 
					 | 
				
			||||||
            binding["target"] = KeyOf(None, ScalarNode(tag="", value="app"))
 | 
					 | 
				
			||||||
            for key, value in binding.items():
 | 
					 | 
				
			||||||
                if not isinstance(value, Model):
 | 
					 | 
				
			||||||
                    continue
 | 
					 | 
				
			||||||
                binding[key] = value.pk
 | 
					 | 
				
			||||||
            blueprint.entries.append(
 | 
					 | 
				
			||||||
                BlueprintEntry(
 | 
					 | 
				
			||||||
                    model="authentik_policies.policybinding",
 | 
					 | 
				
			||||||
                    state=BlueprintEntryDesiredState.MUST_CREATED,
 | 
					 | 
				
			||||||
                    identifiers=binding,
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
        importer = Importer(blueprint, {})
 | 
					        importer = Importer(blueprint, {})
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            valid, _ = importer.validate(raise_validation_errors=True)
 | 
					            valid, _ = importer.validate(raise_validation_errors=True)
 | 
				
			||||||
@ -152,7 +120,8 @@ class TransactionApplicationResponseSerializer(PassiveSerializer):
 | 
				
			|||||||
class TransactionalApplicationView(APIView):
 | 
					class TransactionalApplicationView(APIView):
 | 
				
			||||||
    """Create provider and application and attach them in a single transaction"""
 | 
					    """Create provider and application and attach them in a single transaction"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    permission_classes = [IsAuthenticated]
 | 
					    # TODO: Migrate to a more specific permission
 | 
				
			||||||
 | 
					    permission_classes = [IsAdminUser]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @extend_schema(
 | 
					    @extend_schema(
 | 
				
			||||||
        request=TransactionApplicationSerializer(),
 | 
					        request=TransactionApplicationSerializer(),
 | 
				
			||||||
@ -164,23 +133,8 @@ class TransactionalApplicationView(APIView):
 | 
				
			|||||||
        """Convert data into a blueprint, validate it and apply it"""
 | 
					        """Convert data into a blueprint, validate it and apply it"""
 | 
				
			||||||
        data = TransactionApplicationSerializer(data=request.data)
 | 
					        data = TransactionApplicationSerializer(data=request.data)
 | 
				
			||||||
        data.is_valid(raise_exception=True)
 | 
					        data.is_valid(raise_exception=True)
 | 
				
			||||||
        blueprint: Blueprint = data.validated_data
 | 
					
 | 
				
			||||||
        for entry in blueprint.entries:
 | 
					        importer = Importer(data.validated_data, {})
 | 
				
			||||||
            full_model = entry.get_model(blueprint)
 | 
					 | 
				
			||||||
            app, __, model = full_model.partition(".")
 | 
					 | 
				
			||||||
            if not request.user.has_perm(f"{app}.add_{model}"):
 | 
					 | 
				
			||||||
                raise PermissionDenied(
 | 
					 | 
				
			||||||
                    {
 | 
					 | 
				
			||||||
                        entry.id: _(
 | 
					 | 
				
			||||||
                            "User lacks permission to create {model}".format_map(
 | 
					 | 
				
			||||||
                                {
 | 
					 | 
				
			||||||
                                    "model": full_model,
 | 
					 | 
				
			||||||
                                }
 | 
					 | 
				
			||||||
                            )
 | 
					 | 
				
			||||||
                        )
 | 
					 | 
				
			||||||
                    }
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
        importer = Importer(blueprint, {})
 | 
					 | 
				
			||||||
        applied = importer.apply()
 | 
					        applied = importer.apply()
 | 
				
			||||||
        response = {"applied": False, "logs": []}
 | 
					        response = {"applied": False, "logs": []}
 | 
				
			||||||
        response["applied"] = applied
 | 
					        response["applied"] = applied
 | 
				
			||||||
 | 
				
			|||||||
@ -585,7 +585,7 @@ class UserViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
        """Set password for user"""
 | 
					        """Set password for user"""
 | 
				
			||||||
        user: User = self.get_object()
 | 
					        user: User = self.get_object()
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
            user.set_password(request.data.get("password"), request=request)
 | 
					            user.set_password(request.data.get("password"))
 | 
				
			||||||
            user.save()
 | 
					            user.save()
 | 
				
			||||||
        except (ValidationError, IntegrityError) as exc:
 | 
					        except (ValidationError, IntegrityError) as exc:
 | 
				
			||||||
            LOGGER.debug("Failed to set password", exc=exc)
 | 
					            LOGGER.debug("Failed to set password", exc=exc)
 | 
				
			||||||
@ -666,12 +666,7 @@ class UserViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    @permission_required("authentik_core.impersonate")
 | 
					    @permission_required("authentik_core.impersonate")
 | 
				
			||||||
    @extend_schema(
 | 
					    @extend_schema(
 | 
				
			||||||
        request=inline_serializer(
 | 
					        request=OpenApiTypes.NONE,
 | 
				
			||||||
            "ImpersonationSerializer",
 | 
					 | 
				
			||||||
            {
 | 
					 | 
				
			||||||
                "reason": CharField(required=True),
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        responses={
 | 
					        responses={
 | 
				
			||||||
            "204": OpenApiResponse(description="Successfully started impersonation"),
 | 
					            "204": OpenApiResponse(description="Successfully started impersonation"),
 | 
				
			||||||
            "401": OpenApiResponse(description="Access denied"),
 | 
					            "401": OpenApiResponse(description="Access denied"),
 | 
				
			||||||
@ -684,7 +679,6 @@ class UserViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
            LOGGER.debug("User attempted to impersonate", user=request.user)
 | 
					            LOGGER.debug("User attempted to impersonate", user=request.user)
 | 
				
			||||||
            return Response(status=401)
 | 
					            return Response(status=401)
 | 
				
			||||||
        user_to_be = self.get_object()
 | 
					        user_to_be = self.get_object()
 | 
				
			||||||
        reason = request.data.get("reason", "")
 | 
					 | 
				
			||||||
        # Check both object-level perms and global perms
 | 
					        # Check both object-level perms and global perms
 | 
				
			||||||
        if not request.user.has_perm(
 | 
					        if not request.user.has_perm(
 | 
				
			||||||
            "authentik_core.impersonate", user_to_be
 | 
					            "authentik_core.impersonate", user_to_be
 | 
				
			||||||
@ -694,16 +688,11 @@ class UserViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
        if user_to_be.pk == self.request.user.pk:
 | 
					        if user_to_be.pk == self.request.user.pk:
 | 
				
			||||||
            LOGGER.debug("User attempted to impersonate themselves", user=request.user)
 | 
					            LOGGER.debug("User attempted to impersonate themselves", user=request.user)
 | 
				
			||||||
            return Response(status=401)
 | 
					            return Response(status=401)
 | 
				
			||||||
        if not reason and request.tenant.impersonation_require_reason:
 | 
					 | 
				
			||||||
            LOGGER.debug(
 | 
					 | 
				
			||||||
                "User attempted to impersonate without providing a reason", user=request.user
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
            return Response(status=401)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
        request.session[SESSION_KEY_IMPERSONATE_ORIGINAL_USER] = request.user
 | 
					        request.session[SESSION_KEY_IMPERSONATE_ORIGINAL_USER] = request.user
 | 
				
			||||||
        request.session[SESSION_KEY_IMPERSONATE_USER] = user_to_be
 | 
					        request.session[SESSION_KEY_IMPERSONATE_USER] = user_to_be
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        Event.new(EventAction.IMPERSONATION_STARTED, reason=reason).from_http(request, user_to_be)
 | 
					        Event.new(EventAction.IMPERSONATION_STARTED).from_http(request, user_to_be)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        return Response(status=201)
 | 
					        return Response(status=201)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -44,12 +44,13 @@ class TokenBackend(InbuiltBackend):
 | 
				
			|||||||
        self, request: HttpRequest, username: str | None, password: str | None, **kwargs: Any
 | 
					        self, request: HttpRequest, username: str | None, password: str | None, **kwargs: Any
 | 
				
			||||||
    ) -> User | None:
 | 
					    ) -> User | None:
 | 
				
			||||||
        try:
 | 
					        try:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            user = User._default_manager.get_by_natural_key(username)
 | 
					            user = User._default_manager.get_by_natural_key(username)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        except User.DoesNotExist:
 | 
					        except User.DoesNotExist:
 | 
				
			||||||
            # Run the default password hasher once to reduce the timing
 | 
					            # Run the default password hasher once to reduce the timing
 | 
				
			||||||
            # difference between an existing and a nonexistent user (#20760).
 | 
					            # difference between an existing and a nonexistent user (#20760).
 | 
				
			||||||
            User().set_password(password, request=request)
 | 
					            User().set_password(password)
 | 
				
			||||||
            return None
 | 
					            return None
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        tokens = Token.filter_not_expired(
 | 
					        tokens = Token.filter_not_expired(
 | 
				
			||||||
 | 
				
			|||||||
@ -58,7 +58,6 @@ class PropertyMappingEvaluator(BaseEvaluator):
 | 
				
			|||||||
            self._context["user"] = user
 | 
					            self._context["user"] = user
 | 
				
			||||||
        if request:
 | 
					        if request:
 | 
				
			||||||
            req.http_request = request
 | 
					            req.http_request = request
 | 
				
			||||||
            self._context["http_request"] = request
 | 
					 | 
				
			||||||
        req.context.update(**kwargs)
 | 
					        req.context.update(**kwargs)
 | 
				
			||||||
        self._context["request"] = req
 | 
					        self._context["request"] = req
 | 
				
			||||||
        self._context.update(**kwargs)
 | 
					        self._context.update(**kwargs)
 | 
				
			||||||
 | 
				
			|||||||
@ -17,9 +17,7 @@ from authentik.events.middleware import should_log_model
 | 
				
			|||||||
from authentik.events.models import Event, EventAction
 | 
					from authentik.events.models import Event, EventAction
 | 
				
			||||||
from authentik.events.utils import model_to_dict
 | 
					from authentik.events.utils import model_to_dict
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					BANNER_TEXT = f"""### authentik shell ({get_full_version()})
 | 
				
			||||||
def get_banner_text(shell_type="shell") -> str:
 | 
					 | 
				
			||||||
    return f"""### authentik {shell_type} ({get_full_version()})
 | 
					 | 
				
			||||||
### Node {platform.node()} | Arch {platform.machine()} | Python {platform.python_version()} """
 | 
					### Node {platform.node()} | Arch {platform.machine()} | Python {platform.python_version()} """
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -116,4 +114,4 @@ class Command(BaseCommand):
 | 
				
			|||||||
            readline.parse_and_bind("tab: complete")
 | 
					            readline.parse_and_bind("tab: complete")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        # Run interactive shell
 | 
					        # Run interactive shell
 | 
				
			||||||
        code.interact(banner=get_banner_text(), local=namespace)
 | 
					        code.interact(banner=BANNER_TEXT, local=namespace)
 | 
				
			||||||
 | 
				
			|||||||
@ -5,7 +5,7 @@ from contextvars import ContextVar
 | 
				
			|||||||
from uuid import uuid4
 | 
					from uuid import uuid4
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from django.http import HttpRequest, HttpResponse
 | 
					from django.http import HttpRequest, HttpResponse
 | 
				
			||||||
from django.utils.translation import override
 | 
					from django.utils.translation import activate
 | 
				
			||||||
from sentry_sdk.api import set_tag
 | 
					from sentry_sdk.api import set_tag
 | 
				
			||||||
from structlog.contextvars import STRUCTLOG_KEY_PREFIX
 | 
					from structlog.contextvars import STRUCTLOG_KEY_PREFIX
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -31,20 +31,16 @@ class ImpersonateMiddleware:
 | 
				
			|||||||
    def __call__(self, request: HttpRequest) -> HttpResponse:
 | 
					    def __call__(self, request: HttpRequest) -> HttpResponse:
 | 
				
			||||||
        # No permission checks are done here, they need to be checked before
 | 
					        # No permission checks are done here, they need to be checked before
 | 
				
			||||||
        # SESSION_KEY_IMPERSONATE_USER is set.
 | 
					        # SESSION_KEY_IMPERSONATE_USER is set.
 | 
				
			||||||
        locale_to_set = None
 | 
					 | 
				
			||||||
        if request.user.is_authenticated:
 | 
					        if request.user.is_authenticated:
 | 
				
			||||||
            locale = request.user.locale(request)
 | 
					            locale = request.user.locale(request)
 | 
				
			||||||
            if locale != "":
 | 
					            if locale != "":
 | 
				
			||||||
                locale_to_set = locale
 | 
					                activate(locale)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if SESSION_KEY_IMPERSONATE_USER in request.session:
 | 
					        if SESSION_KEY_IMPERSONATE_USER in request.session:
 | 
				
			||||||
            request.user = request.session[SESSION_KEY_IMPERSONATE_USER]
 | 
					            request.user = request.session[SESSION_KEY_IMPERSONATE_USER]
 | 
				
			||||||
            # Ensure that the user is active, otherwise nothing will work
 | 
					            # Ensure that the user is active, otherwise nothing will work
 | 
				
			||||||
            request.user.is_active = True
 | 
					            request.user.is_active = True
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if locale_to_set:
 | 
					 | 
				
			||||||
            with override(locale_to_set):
 | 
					 | 
				
			||||||
                return self.get_response(request)
 | 
					 | 
				
			||||||
        return self.get_response(request)
 | 
					        return self.get_response(request)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -1,45 +0,0 @@
 | 
				
			|||||||
# Generated by Django 5.0.9 on 2024-11-20 15:16
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import django.db.models.deletion
 | 
					 | 
				
			||||||
from django.db import migrations, models
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Migration(migrations.Migration):
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    dependencies = [
 | 
					 | 
				
			||||||
        ("authentik_core", "0040_provider_invalidation_flow"),
 | 
					 | 
				
			||||||
        ("authentik_policies", "0011_policybinding_failure_result_and_more"),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    operations = [
 | 
					 | 
				
			||||||
        migrations.CreateModel(
 | 
					 | 
				
			||||||
            name="ApplicationEntitlement",
 | 
					 | 
				
			||||||
            fields=[
 | 
					 | 
				
			||||||
                (
 | 
					 | 
				
			||||||
                    "policybindingmodel_ptr",
 | 
					 | 
				
			||||||
                    models.OneToOneField(
 | 
					 | 
				
			||||||
                        auto_created=True,
 | 
					 | 
				
			||||||
                        on_delete=django.db.models.deletion.CASCADE,
 | 
					 | 
				
			||||||
                        parent_link=True,
 | 
					 | 
				
			||||||
                        primary_key=True,
 | 
					 | 
				
			||||||
                        serialize=False,
 | 
					 | 
				
			||||||
                        to="authentik_policies.policybindingmodel",
 | 
					 | 
				
			||||||
                    ),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                ("attributes", models.JSONField(blank=True, default=dict)),
 | 
					 | 
				
			||||||
                ("name", models.TextField()),
 | 
					 | 
				
			||||||
                (
 | 
					 | 
				
			||||||
                    "app",
 | 
					 | 
				
			||||||
                    models.ForeignKey(
 | 
					 | 
				
			||||||
                        on_delete=django.db.models.deletion.CASCADE, to="authentik_core.application"
 | 
					 | 
				
			||||||
                    ),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            options={
 | 
					 | 
				
			||||||
                "verbose_name": "Application Entitlement",
 | 
					 | 
				
			||||||
                "verbose_name_plural": "Application Entitlements",
 | 
					 | 
				
			||||||
                "unique_together": {("app", "name")},
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
            bases=("authentik_policies.policybindingmodel", models.Model),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
@ -1,45 +0,0 @@
 | 
				
			|||||||
# Generated by Django 5.0.10 on 2025-01-13 18:05
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.db import migrations, models
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Migration(migrations.Migration):
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    dependencies = [
 | 
					 | 
				
			||||||
        ("authentik_core", "0041_applicationentitlement"),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    operations = [
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="authenticatedsession",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expires"], name="authentik_c_expires_08251d_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="authenticatedsession",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expiring"], name="authentik_c_expirin_9cd839_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="authenticatedsession",
 | 
					 | 
				
			||||||
            index=models.Index(
 | 
					 | 
				
			||||||
                fields=["expiring", "expires"], name="authentik_c_expirin_195a84_idx"
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="authenticatedsession",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["session_key"], name="authentik_c_session_d0f005_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="token",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expires"], name="authentik_c_expires_a62b4b_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="token",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expiring"], name="authentik_c_expirin_a1b838_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="token",
 | 
					 | 
				
			||||||
            index=models.Index(
 | 
					 | 
				
			||||||
                fields=["expiring", "expires"], name="authentik_c_expirin_ba04d9_idx"
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
@ -1,57 +0,0 @@
 | 
				
			|||||||
# Generated by Django 5.0.10 on 2025-01-08 17:39
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.db import migrations
 | 
					 | 
				
			||||||
from django.apps.registry import Apps
 | 
					 | 
				
			||||||
from django.db.backends.base.schema import BaseDatabaseSchemaEditor
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
def migrate_user_debug_attribute(apps: Apps, schema_editor: BaseDatabaseSchemaEditor):
 | 
					 | 
				
			||||||
    from django.apps import apps as real_apps
 | 
					 | 
				
			||||||
    from django.contrib.auth.management import create_permissions
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    db_alias = schema_editor.connection.alias
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    User = apps.get_model("authentik_core", "User")
 | 
					 | 
				
			||||||
    USER_ATTRIBUTE_DEBUG = "goauthentik.io/user/debug"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    # Permissions are only created _after_ migrations are run
 | 
					 | 
				
			||||||
    # - https://github.com/django/django/blob/43cdfa8b20e567a801b7d0a09ec67ddd062d5ea4/django/contrib/auth/apps.py#L19
 | 
					 | 
				
			||||||
    # - https://stackoverflow.com/a/72029063/1870445
 | 
					 | 
				
			||||||
    create_permissions(real_apps.get_app_config("authentik_core"), using=db_alias)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    Permission = apps.get_model("auth", "Permission")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    new_prem = Permission.objects.using(db_alias).get(codename="user_view_debug")
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    db_alias = schema_editor.connection.alias
 | 
					 | 
				
			||||||
    for user in User.objects.using(db_alias).filter(
 | 
					 | 
				
			||||||
        **{f"attributes__{USER_ATTRIBUTE_DEBUG}": True}
 | 
					 | 
				
			||||||
    ):
 | 
					 | 
				
			||||||
        user.permissions.add(new_prem)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Migration(migrations.Migration):
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    dependencies = [
 | 
					 | 
				
			||||||
        ("authentik_core", "0042_authenticatedsession_authentik_c_expires_08251d_idx_and_more"),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    operations = [
 | 
					 | 
				
			||||||
        migrations.AlterModelOptions(
 | 
					 | 
				
			||||||
            name="user",
 | 
					 | 
				
			||||||
            options={
 | 
					 | 
				
			||||||
                "permissions": [
 | 
					 | 
				
			||||||
                    ("reset_user_password", "Reset Password"),
 | 
					 | 
				
			||||||
                    ("impersonate", "Can impersonate other users"),
 | 
					 | 
				
			||||||
                    ("assign_user_permissions", "Can assign permissions to users"),
 | 
					 | 
				
			||||||
                    ("unassign_user_permissions", "Can unassign permissions from users"),
 | 
					 | 
				
			||||||
                    ("preview_user", "Can preview user data sent to providers"),
 | 
					 | 
				
			||||||
                    ("view_user_applications", "View applications the user has access to"),
 | 
					 | 
				
			||||||
                    ("user_view_debug", "User receives additional details for error messages"),
 | 
					 | 
				
			||||||
                ],
 | 
					 | 
				
			||||||
                "verbose_name": "User",
 | 
					 | 
				
			||||||
                "verbose_name_plural": "Users",
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.RunPython(migrate_user_debug_attribute),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
@ -41,6 +41,7 @@ from authentik.tenants.models import DEFAULT_TOKEN_DURATION, DEFAULT_TOKEN_LENGT
 | 
				
			|||||||
from authentik.tenants.utils import get_current_tenant, get_unique_identifier
 | 
					from authentik.tenants.utils import get_current_tenant, get_unique_identifier
 | 
				
			||||||
 | 
					
 | 
				
			||||||
LOGGER = get_logger()
 | 
					LOGGER = get_logger()
 | 
				
			||||||
 | 
					USER_ATTRIBUTE_DEBUG = "goauthentik.io/user/debug"
 | 
				
			||||||
USER_ATTRIBUTE_GENERATED = "goauthentik.io/user/generated"
 | 
					USER_ATTRIBUTE_GENERATED = "goauthentik.io/user/generated"
 | 
				
			||||||
USER_ATTRIBUTE_EXPIRES = "goauthentik.io/user/expires"
 | 
					USER_ATTRIBUTE_EXPIRES = "goauthentik.io/user/expires"
 | 
				
			||||||
USER_ATTRIBUTE_DELETE_ON_LOGOUT = "goauthentik.io/user/delete-on-logout"
 | 
					USER_ATTRIBUTE_DELETE_ON_LOGOUT = "goauthentik.io/user/delete-on-logout"
 | 
				
			||||||
@ -281,7 +282,6 @@ class User(SerializerModel, GuardianUserMixin, AttributesMixin, AbstractUser):
 | 
				
			|||||||
            ("unassign_user_permissions", _("Can unassign permissions from users")),
 | 
					            ("unassign_user_permissions", _("Can unassign permissions from users")),
 | 
				
			||||||
            ("preview_user", _("Can preview user data sent to providers")),
 | 
					            ("preview_user", _("Can preview user data sent to providers")),
 | 
				
			||||||
            ("view_user_applications", _("View applications the user has access to")),
 | 
					            ("view_user_applications", _("View applications the user has access to")),
 | 
				
			||||||
            ("user_view_debug", _("User receives additional details for error messages")),
 | 
					 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
        indexes = [
 | 
					        indexes = [
 | 
				
			||||||
            models.Index(fields=["last_login"]),
 | 
					            models.Index(fields=["last_login"]),
 | 
				
			||||||
@ -314,32 +314,6 @@ class User(SerializerModel, GuardianUserMixin, AttributesMixin, AbstractUser):
 | 
				
			|||||||
        always_merger.merge(final_attributes, self.attributes)
 | 
					        always_merger.merge(final_attributes, self.attributes)
 | 
				
			||||||
        return final_attributes
 | 
					        return final_attributes
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def app_entitlements(self, app: "Application | None") -> QuerySet["ApplicationEntitlement"]:
 | 
					 | 
				
			||||||
        """Get all entitlements this user has for `app`."""
 | 
					 | 
				
			||||||
        if not app:
 | 
					 | 
				
			||||||
            return []
 | 
					 | 
				
			||||||
        all_groups = self.all_groups()
 | 
					 | 
				
			||||||
        qs = app.applicationentitlement_set.filter(
 | 
					 | 
				
			||||||
            Q(
 | 
					 | 
				
			||||||
                Q(bindings__user=self) | Q(bindings__group__in=all_groups),
 | 
					 | 
				
			||||||
                bindings__negate=False,
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
            | Q(
 | 
					 | 
				
			||||||
                Q(~Q(bindings__user=self), bindings__user__isnull=False)
 | 
					 | 
				
			||||||
                | Q(~Q(bindings__group__in=all_groups), bindings__group__isnull=False),
 | 
					 | 
				
			||||||
                bindings__negate=True,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            bindings__enabled=True,
 | 
					 | 
				
			||||||
        ).order_by("name")
 | 
					 | 
				
			||||||
        return qs
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def app_entitlements_attributes(self, app: "Application | None") -> dict:
 | 
					 | 
				
			||||||
        """Get a dictionary containing all merged attributes from app entitlements for `app`."""
 | 
					 | 
				
			||||||
        final_attributes = {}
 | 
					 | 
				
			||||||
        for attrs in self.app_entitlements(app).values_list("attributes", flat=True):
 | 
					 | 
				
			||||||
            always_merger.merge(final_attributes, attrs)
 | 
					 | 
				
			||||||
        return final_attributes
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    @property
 | 
					    @property
 | 
				
			||||||
    def serializer(self) -> Serializer:
 | 
					    def serializer(self) -> Serializer:
 | 
				
			||||||
        from authentik.core.api.users import UserSerializer
 | 
					        from authentik.core.api.users import UserSerializer
 | 
				
			||||||
@ -356,13 +330,13 @@ class User(SerializerModel, GuardianUserMixin, AttributesMixin, AbstractUser):
 | 
				
			|||||||
        """superuser == staff user"""
 | 
					        """superuser == staff user"""
 | 
				
			||||||
        return self.is_superuser  # type: ignore
 | 
					        return self.is_superuser  # type: ignore
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def set_password(self, raw_password, signal=True, sender=None, request=None):
 | 
					    def set_password(self, raw_password, signal=True, sender=None):
 | 
				
			||||||
        if self.pk and signal:
 | 
					        if self.pk and signal:
 | 
				
			||||||
            from authentik.core.signals import password_changed
 | 
					            from authentik.core.signals import password_changed
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            if not sender:
 | 
					            if not sender:
 | 
				
			||||||
                sender = self
 | 
					                sender = self
 | 
				
			||||||
            password_changed.send(sender=sender, user=self, password=raw_password, request=request)
 | 
					            password_changed.send(sender=sender, user=self, password=raw_password)
 | 
				
			||||||
        self.password_change_date = now()
 | 
					        self.password_change_date = now()
 | 
				
			||||||
        return super().set_password(raw_password)
 | 
					        return super().set_password(raw_password)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -607,31 +581,6 @@ class Application(SerializerModel, PolicyBindingModel):
 | 
				
			|||||||
        verbose_name_plural = _("Applications")
 | 
					        verbose_name_plural = _("Applications")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ApplicationEntitlement(AttributesMixin, SerializerModel, PolicyBindingModel):
 | 
					 | 
				
			||||||
    """Application-scoped entitlement to control authorization in an application"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    name = models.TextField()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    app = models.ForeignKey(Application, on_delete=models.CASCADE)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    class Meta:
 | 
					 | 
				
			||||||
        verbose_name = _("Application Entitlement")
 | 
					 | 
				
			||||||
        verbose_name_plural = _("Application Entitlements")
 | 
					 | 
				
			||||||
        unique_together = (("app", "name"),)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def __str__(self):
 | 
					 | 
				
			||||||
        return f"Application Entitlement {self.name} for app {self.app_id}"
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    @property
 | 
					 | 
				
			||||||
    def serializer(self) -> type[Serializer]:
 | 
					 | 
				
			||||||
        from authentik.core.api.application_entitlements import ApplicationEntitlementSerializer
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        return ApplicationEntitlementSerializer
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def supported_policy_binding_targets(self):
 | 
					 | 
				
			||||||
        return ["group", "user"]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class SourceUserMatchingModes(models.TextChoices):
 | 
					class SourceUserMatchingModes(models.TextChoices):
 | 
				
			||||||
    """Different modes a source can handle new/returning users"""
 | 
					    """Different modes a source can handle new/returning users"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -846,11 +795,6 @@ class ExpiringModel(models.Model):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        abstract = True
 | 
					        abstract = True
 | 
				
			||||||
        indexes = [
 | 
					 | 
				
			||||||
            models.Index(fields=["expires"]),
 | 
					 | 
				
			||||||
            models.Index(fields=["expiring"]),
 | 
					 | 
				
			||||||
            models.Index(fields=["expiring", "expires"]),
 | 
					 | 
				
			||||||
        ]
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def expire_action(self, *args, **kwargs):
 | 
					    def expire_action(self, *args, **kwargs):
 | 
				
			||||||
        """Handler which is called when this object is expired. By
 | 
					        """Handler which is called when this object is expired. By
 | 
				
			||||||
@ -906,7 +850,7 @@ class Token(SerializerModel, ManagedModel, ExpiringModel):
 | 
				
			|||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        verbose_name = _("Token")
 | 
					        verbose_name = _("Token")
 | 
				
			||||||
        verbose_name_plural = _("Tokens")
 | 
					        verbose_name_plural = _("Tokens")
 | 
				
			||||||
        indexes = ExpiringModel.Meta.indexes + [
 | 
					        indexes = [
 | 
				
			||||||
            models.Index(fields=["identifier"]),
 | 
					            models.Index(fields=["identifier"]),
 | 
				
			||||||
            models.Index(fields=["key"]),
 | 
					            models.Index(fields=["key"]),
 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
@ -1006,9 +950,6 @@ class AuthenticatedSession(ExpiringModel):
 | 
				
			|||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        verbose_name = _("Authenticated Session")
 | 
					        verbose_name = _("Authenticated Session")
 | 
				
			||||||
        verbose_name_plural = _("Authenticated Sessions")
 | 
					        verbose_name_plural = _("Authenticated Sessions")
 | 
				
			||||||
        indexes = ExpiringModel.Meta.indexes + [
 | 
					 | 
				
			||||||
            models.Index(fields=["session_key"]),
 | 
					 | 
				
			||||||
        ]
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def __str__(self) -> str:
 | 
					    def __str__(self) -> str:
 | 
				
			||||||
        return f"Authenticated Session {self.session_key[:10]}"
 | 
					        return f"Authenticated Session {self.session_key[:10]}"
 | 
				
			||||||
 | 
				
			|||||||
@ -129,11 +129,6 @@ class SourceFlowManager:
 | 
				
			|||||||
            )
 | 
					            )
 | 
				
			||||||
            new_connection.user = self.request.user
 | 
					            new_connection.user = self.request.user
 | 
				
			||||||
            new_connection = self.update_user_connection(new_connection, **kwargs)
 | 
					            new_connection = self.update_user_connection(new_connection, **kwargs)
 | 
				
			||||||
            if existing := self.user_connection_type.objects.filter(
 | 
					 | 
				
			||||||
                source=self.source, identifier=self.identifier
 | 
					 | 
				
			||||||
            ).first():
 | 
					 | 
				
			||||||
                existing = self.update_user_connection(existing)
 | 
					 | 
				
			||||||
                return Action.AUTH, existing
 | 
					 | 
				
			||||||
            return Action.LINK, new_connection
 | 
					            return Action.LINK, new_connection
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        action, connection = self.matcher.get_user_action(self.identifier, self.user_properties)
 | 
					        action, connection = self.matcher.get_user_action(self.identifier, self.user_properties)
 | 
				
			||||||
@ -238,7 +233,13 @@ class SourceFlowManager:
 | 
				
			|||||||
                self.request.GET,
 | 
					                self.request.GET,
 | 
				
			||||||
                flow_slug=flow_slug,
 | 
					                flow_slug=flow_slug,
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
        flow_context.setdefault(PLAN_CONTEXT_REDIRECT, final_redirect)
 | 
					        # Ensure redirect is carried through when user was trying to
 | 
				
			||||||
 | 
					        # authorize application
 | 
				
			||||||
 | 
					        final_redirect = self.request.session.get(SESSION_KEY_GET, {}).get(
 | 
				
			||||||
 | 
					            NEXT_ARG_NAME, "authentik_core:if-user"
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					        if PLAN_CONTEXT_REDIRECT not in flow_context:
 | 
				
			||||||
 | 
					            flow_context[PLAN_CONTEXT_REDIRECT] = final_redirect
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if not flow:
 | 
					        if not flow:
 | 
				
			||||||
            return bad_request_message(
 | 
					            return bad_request_message(
 | 
				
			||||||
@ -259,7 +260,12 @@ class SourceFlowManager:
 | 
				
			|||||||
        if stages:
 | 
					        if stages:
 | 
				
			||||||
            for stage in stages:
 | 
					            for stage in stages:
 | 
				
			||||||
                plan.append_stage(stage)
 | 
					                plan.append_stage(stage)
 | 
				
			||||||
        return plan.to_redirect(self.request, flow)
 | 
					        self.request.session[SESSION_KEY_PLAN] = plan
 | 
				
			||||||
 | 
					        return redirect_with_qs(
 | 
				
			||||||
 | 
					            "authentik_core:if-flow",
 | 
				
			||||||
 | 
					            self.request.GET,
 | 
				
			||||||
 | 
					            flow_slug=flow.slug,
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def handle_auth(
 | 
					    def handle_auth(
 | 
				
			||||||
        self,
 | 
					        self,
 | 
				
			||||||
 | 
				
			|||||||
@ -9,9 +9,6 @@
 | 
				
			|||||||
        versionFamily: "{{ version_family }}",
 | 
					        versionFamily: "{{ version_family }}",
 | 
				
			||||||
        versionSubdomain: "{{ version_subdomain }}",
 | 
					        versionSubdomain: "{{ version_subdomain }}",
 | 
				
			||||||
        build: "{{ build }}",
 | 
					        build: "{{ build }}",
 | 
				
			||||||
        api: {
 | 
					 | 
				
			||||||
            base: "{{ base_url }}",
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
    };
 | 
					    };
 | 
				
			||||||
    window.addEventListener("DOMContentLoaded", function () {
 | 
					    window.addEventListener("DOMContentLoaded", function () {
 | 
				
			||||||
        {% for message in messages %}
 | 
					        {% for message in messages %}
 | 
				
			||||||
 | 
				
			|||||||
@ -9,14 +9,14 @@
 | 
				
			|||||||
        <meta charset="UTF-8">
 | 
					        <meta charset="UTF-8">
 | 
				
			||||||
        <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
 | 
					        <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
 | 
				
			||||||
        <title>{% block title %}{% trans title|default:brand.branding_title %}{% endblock %}</title>
 | 
					        <title>{% block title %}{% trans title|default:brand.branding_title %}{% endblock %}</title>
 | 
				
			||||||
        <link rel="icon" href="{{ brand.branding_favicon_url }}">
 | 
					        <link rel="icon" href="{{ brand.branding_favicon }}">
 | 
				
			||||||
        <link rel="shortcut icon" href="{{ brand.branding_favicon_url }}">
 | 
					        <link rel="shortcut icon" href="{{ brand.branding_favicon }}">
 | 
				
			||||||
        {% block head_before %}
 | 
					        {% block head_before %}
 | 
				
			||||||
        {% endblock %}
 | 
					        {% endblock %}
 | 
				
			||||||
        <link rel="stylesheet" type="text/css" href="{% static 'dist/authentik.css' %}">
 | 
					        <link rel="stylesheet" type="text/css" href="{% static 'dist/authentik.css' %}">
 | 
				
			||||||
        <link rel="stylesheet" type="text/css" href="{% static 'dist/custom.css' %}" data-inject>
 | 
					        <link rel="stylesheet" type="text/css" href="{% static 'dist/custom.css' %}" data-inject>
 | 
				
			||||||
        <script src="{% versioned_script 'dist/poly-%v.js' %}" type="module"></script>
 | 
					        {% versioned_script "dist/poly-%v.js" %}
 | 
				
			||||||
        <script src="{% versioned_script 'dist/standalone/loading/index-%v.js' %}" type="module"></script>
 | 
					        {% versioned_script "dist/standalone/loading/index-%v.js" %}
 | 
				
			||||||
        {% block head %}
 | 
					        {% block head %}
 | 
				
			||||||
        {% endblock %}
 | 
					        {% endblock %}
 | 
				
			||||||
        <meta name="sentry-trace" content="{{ sentry_trace }}" />
 | 
					        <meta name="sentry-trace" content="{{ sentry_trace }}" />
 | 
				
			||||||
 | 
				
			|||||||
@ -3,7 +3,7 @@
 | 
				
			|||||||
{% load authentik_core %}
 | 
					{% load authentik_core %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{% block head %}
 | 
					{% block head %}
 | 
				
			||||||
<script src="{% versioned_script 'dist/admin/AdminInterface-%v.js' %}" type="module"></script>
 | 
					{% versioned_script "dist/admin/AdminInterface-%v.js" %}
 | 
				
			||||||
<meta name="theme-color" content="#18191a" media="(prefers-color-scheme: dark)">
 | 
					<meta name="theme-color" content="#18191a" media="(prefers-color-scheme: dark)">
 | 
				
			||||||
<meta name="theme-color" content="#ffffff" media="(prefers-color-scheme: light)">
 | 
					<meta name="theme-color" content="#ffffff" media="(prefers-color-scheme: light)">
 | 
				
			||||||
{% include "base/header_js.html" %}
 | 
					{% include "base/header_js.html" %}
 | 
				
			||||||
 | 
				
			|||||||
@ -3,7 +3,7 @@
 | 
				
			|||||||
{% load authentik_core %}
 | 
					{% load authentik_core %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{% block head %}
 | 
					{% block head %}
 | 
				
			||||||
<script src="{% versioned_script 'dist/user/UserInterface-%v.js' %}" type="module"></script>
 | 
					{% versioned_script "dist/user/UserInterface-%v.js" %}
 | 
				
			||||||
<meta name="theme-color" content="#1c1e21" media="(prefers-color-scheme: light)">
 | 
					<meta name="theme-color" content="#1c1e21" media="(prefers-color-scheme: light)">
 | 
				
			||||||
<meta name="theme-color" content="#1c1e21" media="(prefers-color-scheme: dark)">
 | 
					<meta name="theme-color" content="#1c1e21" media="(prefers-color-scheme: dark)">
 | 
				
			||||||
{% include "base/header_js.html" %}
 | 
					{% include "base/header_js.html" %}
 | 
				
			||||||
 | 
				
			|||||||
@ -4,7 +4,7 @@
 | 
				
			|||||||
{% load i18n %}
 | 
					{% load i18n %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{% block head_before %}
 | 
					{% block head_before %}
 | 
				
			||||||
<link rel="prefetch" href="{% static 'dist/assets/images/flow_background.jpg' %}" />
 | 
					<link rel="prefetch" href="/static/dist/assets/images/flow_background.jpg" />
 | 
				
			||||||
<link rel="stylesheet" type="text/css" href="{% static 'dist/patternfly.min.css' %}">
 | 
					<link rel="stylesheet" type="text/css" href="{% static 'dist/patternfly.min.css' %}">
 | 
				
			||||||
<link rel="stylesheet" type="text/css" href="{% static 'dist/theme-dark.css' %}" media="(prefers-color-scheme: dark)">
 | 
					<link rel="stylesheet" type="text/css" href="{% static 'dist/theme-dark.css' %}" media="(prefers-color-scheme: dark)">
 | 
				
			||||||
{% include "base/header_js.html" %}
 | 
					{% include "base/header_js.html" %}
 | 
				
			||||||
@ -13,7 +13,7 @@
 | 
				
			|||||||
{% block head %}
 | 
					{% block head %}
 | 
				
			||||||
<style>
 | 
					<style>
 | 
				
			||||||
:root {
 | 
					:root {
 | 
				
			||||||
    --ak-flow-background: url("{% static 'dist/assets/images/flow_background.jpg' %}");
 | 
					    --ak-flow-background: url("/static/dist/assets/images/flow_background.jpg");
 | 
				
			||||||
    --pf-c-background-image--BackgroundImage: var(--ak-flow-background);
 | 
					    --pf-c-background-image--BackgroundImage: var(--ak-flow-background);
 | 
				
			||||||
    --pf-c-background-image--BackgroundImage-2x: var(--ak-flow-background);
 | 
					    --pf-c-background-image--BackgroundImage-2x: var(--ak-flow-background);
 | 
				
			||||||
    --pf-c-background-image--BackgroundImage--sm: var(--ak-flow-background);
 | 
					    --pf-c-background-image--BackgroundImage--sm: var(--ak-flow-background);
 | 
				
			||||||
@ -50,7 +50,7 @@
 | 
				
			|||||||
    <div class="ak-login-container">
 | 
					    <div class="ak-login-container">
 | 
				
			||||||
        <main class="pf-c-login__main">
 | 
					        <main class="pf-c-login__main">
 | 
				
			||||||
            <div class="pf-c-login__main-header pf-c-brand ak-brand">
 | 
					            <div class="pf-c-login__main-header pf-c-brand ak-brand">
 | 
				
			||||||
                <img src="{{ brand.branding_logo_url }}" alt="authentik Logo" />
 | 
					                <img src="{{ brand.branding_logo }}" alt="authentik Logo" />
 | 
				
			||||||
            </div>
 | 
					            </div>
 | 
				
			||||||
            <header class="pf-c-login__main-header">
 | 
					            <header class="pf-c-login__main-header">
 | 
				
			||||||
                <h1 class="pf-c-title pf-m-3xl">
 | 
					                <h1 class="pf-c-title pf-m-3xl">
 | 
				
			||||||
 | 
				
			|||||||
@ -2,6 +2,7 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
from django import template
 | 
					from django import template
 | 
				
			||||||
from django.templatetags.static import static as static_loader
 | 
					from django.templatetags.static import static as static_loader
 | 
				
			||||||
 | 
					from django.utils.safestring import mark_safe
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik import get_full_version
 | 
					from authentik import get_full_version
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -11,4 +12,10 @@ register = template.Library()
 | 
				
			|||||||
@register.simple_tag()
 | 
					@register.simple_tag()
 | 
				
			||||||
def versioned_script(path: str) -> str:
 | 
					def versioned_script(path: str) -> str:
 | 
				
			||||||
    """Wrapper around {% static %} tag that supports setting the version"""
 | 
					    """Wrapper around {% static %} tag that supports setting the version"""
 | 
				
			||||||
    return static_loader(path.replace("%v", get_full_version()))
 | 
					    returned_lines = [
 | 
				
			||||||
 | 
					        (
 | 
				
			||||||
 | 
					            f'<script src="{static_loader(path.replace("%v", get_full_version()))}'
 | 
				
			||||||
 | 
					            '" type="module"></script>'
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					    ]
 | 
				
			||||||
 | 
					    return mark_safe("".join(returned_lines))  # nosec
 | 
				
			||||||
 | 
				
			|||||||
@ -1,153 +0,0 @@
 | 
				
			|||||||
"""Test Application Entitlements API"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.urls import reverse
 | 
					 | 
				
			||||||
from guardian.shortcuts import assign_perm
 | 
					 | 
				
			||||||
from rest_framework.test import APITestCase
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from authentik.core.models import Application, ApplicationEntitlement, Group
 | 
					 | 
				
			||||||
from authentik.core.tests.utils import create_test_admin_user, create_test_flow, create_test_user
 | 
					 | 
				
			||||||
from authentik.lib.generators import generate_id
 | 
					 | 
				
			||||||
from authentik.policies.dummy.models import DummyPolicy
 | 
					 | 
				
			||||||
from authentik.policies.models import PolicyBinding
 | 
					 | 
				
			||||||
from authentik.providers.oauth2.models import OAuth2Provider
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class TestApplicationEntitlements(APITestCase):
 | 
					 | 
				
			||||||
    """Test application entitlements"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def setUp(self) -> None:
 | 
					 | 
				
			||||||
        self.user = create_test_user()
 | 
					 | 
				
			||||||
        self.other_user = create_test_user()
 | 
					 | 
				
			||||||
        self.provider = OAuth2Provider.objects.create(
 | 
					 | 
				
			||||||
            name="test",
 | 
					 | 
				
			||||||
            authorization_flow=create_test_flow(),
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.app: Application = Application.objects.create(
 | 
					 | 
				
			||||||
            name=generate_id(),
 | 
					 | 
				
			||||||
            slug=generate_id(),
 | 
					 | 
				
			||||||
            provider=self.provider,
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_user(self):
 | 
					 | 
				
			||||||
        """Test user-direct assignment"""
 | 
					 | 
				
			||||||
        ent = ApplicationEntitlement.objects.create(app=self.app, name=generate_id())
 | 
					 | 
				
			||||||
        PolicyBinding.objects.create(target=ent, user=self.user, order=0)
 | 
					 | 
				
			||||||
        ents = self.user.app_entitlements(self.app)
 | 
					 | 
				
			||||||
        self.assertEqual(len(ents), 1)
 | 
					 | 
				
			||||||
        self.assertEqual(ents[0].name, ent.name)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_group(self):
 | 
					 | 
				
			||||||
        """Test direct group"""
 | 
					 | 
				
			||||||
        group = Group.objects.create(name=generate_id())
 | 
					 | 
				
			||||||
        self.user.ak_groups.add(group)
 | 
					 | 
				
			||||||
        ent = ApplicationEntitlement.objects.create(app=self.app, name=generate_id())
 | 
					 | 
				
			||||||
        PolicyBinding.objects.create(target=ent, group=group, order=0)
 | 
					 | 
				
			||||||
        ents = self.user.app_entitlements(self.app)
 | 
					 | 
				
			||||||
        self.assertEqual(len(ents), 1)
 | 
					 | 
				
			||||||
        self.assertEqual(ents[0].name, ent.name)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_group_indirect(self):
 | 
					 | 
				
			||||||
        """Test indirect group"""
 | 
					 | 
				
			||||||
        parent = Group.objects.create(name=generate_id())
 | 
					 | 
				
			||||||
        group = Group.objects.create(name=generate_id(), parent=parent)
 | 
					 | 
				
			||||||
        self.user.ak_groups.add(group)
 | 
					 | 
				
			||||||
        ent = ApplicationEntitlement.objects.create(app=self.app, name=generate_id())
 | 
					 | 
				
			||||||
        PolicyBinding.objects.create(target=ent, group=parent, order=0)
 | 
					 | 
				
			||||||
        ents = self.user.app_entitlements(self.app)
 | 
					 | 
				
			||||||
        self.assertEqual(len(ents), 1)
 | 
					 | 
				
			||||||
        self.assertEqual(ents[0].name, ent.name)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_negate_user(self):
 | 
					 | 
				
			||||||
        """Test with negate flag"""
 | 
					 | 
				
			||||||
        ent = ApplicationEntitlement.objects.create(app=self.app, name=generate_id())
 | 
					 | 
				
			||||||
        PolicyBinding.objects.create(target=ent, user=self.other_user, order=0, negate=True)
 | 
					 | 
				
			||||||
        ents = self.user.app_entitlements(self.app)
 | 
					 | 
				
			||||||
        self.assertEqual(len(ents), 1)
 | 
					 | 
				
			||||||
        self.assertEqual(ents[0].name, ent.name)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_negate_group(self):
 | 
					 | 
				
			||||||
        """Test with negate flag"""
 | 
					 | 
				
			||||||
        other_group = Group.objects.create(name=generate_id())
 | 
					 | 
				
			||||||
        ent = ApplicationEntitlement.objects.create(app=self.app, name=generate_id())
 | 
					 | 
				
			||||||
        PolicyBinding.objects.create(target=ent, group=other_group, order=0, negate=True)
 | 
					 | 
				
			||||||
        ents = self.user.app_entitlements(self.app)
 | 
					 | 
				
			||||||
        self.assertEqual(len(ents), 1)
 | 
					 | 
				
			||||||
        self.assertEqual(ents[0].name, ent.name)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_api_perms_global(self):
 | 
					 | 
				
			||||||
        """Test API creation with global permissions"""
 | 
					 | 
				
			||||||
        assign_perm("authentik_core.add_applicationentitlement", self.user)
 | 
					 | 
				
			||||||
        assign_perm("authentik_core.view_application", self.user)
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
        res = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:applicationentitlement-list"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "name": generate_id(),
 | 
					 | 
				
			||||||
                "app": self.app.pk,
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertEqual(res.status_code, 201)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_api_perms_scoped(self):
 | 
					 | 
				
			||||||
        """Test API creation with scoped permissions"""
 | 
					 | 
				
			||||||
        assign_perm("authentik_core.add_applicationentitlement", self.user)
 | 
					 | 
				
			||||||
        assign_perm("authentik_core.view_application", self.user, self.app)
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
        res = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:applicationentitlement-list"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "name": generate_id(),
 | 
					 | 
				
			||||||
                "app": self.app.pk,
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertEqual(res.status_code, 201)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_api_perms_missing(self):
 | 
					 | 
				
			||||||
        """Test API creation with no permissions"""
 | 
					 | 
				
			||||||
        assign_perm("authentik_core.add_applicationentitlement", self.user)
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
        res = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:applicationentitlement-list"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "name": generate_id(),
 | 
					 | 
				
			||||||
                "app": self.app.pk,
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertEqual(res.status_code, 400)
 | 
					 | 
				
			||||||
        self.assertJSONEqual(res.content, {"app": ["User does not have access to application."]})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_api_bindings_policy(self):
 | 
					 | 
				
			||||||
        """Test that API doesn't allow policies to be bound to this"""
 | 
					 | 
				
			||||||
        ent = ApplicationEntitlement.objects.create(app=self.app, name=generate_id())
 | 
					 | 
				
			||||||
        policy = DummyPolicy.objects.create(name=generate_id())
 | 
					 | 
				
			||||||
        admin = create_test_admin_user()
 | 
					 | 
				
			||||||
        self.client.force_login(admin)
 | 
					 | 
				
			||||||
        response = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:policybinding-list"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "target": ent.pbm_uuid,
 | 
					 | 
				
			||||||
                "policy": policy.pk,
 | 
					 | 
				
			||||||
                "order": 0,
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertJSONEqual(
 | 
					 | 
				
			||||||
            response.content.decode(),
 | 
					 | 
				
			||||||
            {"non_field_errors": ["One of 'group', 'user' must be set."]},
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_api_bindings_group(self):
 | 
					 | 
				
			||||||
        """Test that API doesn't allow policies to be bound to this"""
 | 
					 | 
				
			||||||
        ent = ApplicationEntitlement.objects.create(app=self.app, name=generate_id())
 | 
					 | 
				
			||||||
        group = Group.objects.create(name=generate_id())
 | 
					 | 
				
			||||||
        admin = create_test_admin_user()
 | 
					 | 
				
			||||||
        self.client.force_login(admin)
 | 
					 | 
				
			||||||
        response = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:policybinding-list"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "target": ent.pbm_uuid,
 | 
					 | 
				
			||||||
                "group": group.pk,
 | 
					 | 
				
			||||||
                "order": 0,
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertEqual(response.status_code, 201)
 | 
					 | 
				
			||||||
        self.assertTrue(PolicyBinding.objects.filter(target=ent.pbm_uuid).exists())
 | 
					 | 
				
			||||||
@ -12,7 +12,7 @@ from authentik.core.tests.utils import create_test_admin_user, create_test_flow
 | 
				
			|||||||
from authentik.lib.generators import generate_id
 | 
					from authentik.lib.generators import generate_id
 | 
				
			||||||
from authentik.policies.dummy.models import DummyPolicy
 | 
					from authentik.policies.dummy.models import DummyPolicy
 | 
				
			||||||
from authentik.policies.models import PolicyBinding
 | 
					from authentik.policies.models import PolicyBinding
 | 
				
			||||||
from authentik.providers.oauth2.models import OAuth2Provider, RedirectURI, RedirectURIMatchingMode
 | 
					from authentik.providers.oauth2.models import OAuth2Provider
 | 
				
			||||||
from authentik.providers.proxy.models import ProxyProvider
 | 
					from authentik.providers.proxy.models import ProxyProvider
 | 
				
			||||||
from authentik.providers.saml.models import SAMLProvider
 | 
					from authentik.providers.saml.models import SAMLProvider
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -24,7 +24,7 @@ class TestApplicationsAPI(APITestCase):
 | 
				
			|||||||
        self.user = create_test_admin_user()
 | 
					        self.user = create_test_admin_user()
 | 
				
			||||||
        self.provider = OAuth2Provider.objects.create(
 | 
					        self.provider = OAuth2Provider.objects.create(
 | 
				
			||||||
            name="test",
 | 
					            name="test",
 | 
				
			||||||
            redirect_uris=[RedirectURI(RedirectURIMatchingMode.STRICT, "http://some-other-domain")],
 | 
					            redirect_uris="http://some-other-domain",
 | 
				
			||||||
            authorization_flow=create_test_flow(),
 | 
					            authorization_flow=create_test_flow(),
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.allowed: Application = Application.objects.create(
 | 
					        self.allowed: Application = Application.objects.create(
 | 
				
			||||||
 | 
				
			|||||||
@ -29,8 +29,7 @@ class TestImpersonation(APITestCase):
 | 
				
			|||||||
            reverse(
 | 
					            reverse(
 | 
				
			||||||
                "authentik_api:user-impersonate",
 | 
					                "authentik_api:user-impersonate",
 | 
				
			||||||
                kwargs={"pk": self.other_user.pk},
 | 
					                kwargs={"pk": self.other_user.pk},
 | 
				
			||||||
            ),
 | 
					            )
 | 
				
			||||||
            data={"reason": "some reason"},
 | 
					 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        response = self.client.get(reverse("authentik_api:user-me"))
 | 
					        response = self.client.get(reverse("authentik_api:user-me"))
 | 
				
			||||||
@ -56,8 +55,7 @@ class TestImpersonation(APITestCase):
 | 
				
			|||||||
            reverse(
 | 
					            reverse(
 | 
				
			||||||
                "authentik_api:user-impersonate",
 | 
					                "authentik_api:user-impersonate",
 | 
				
			||||||
                kwargs={"pk": self.other_user.pk},
 | 
					                kwargs={"pk": self.other_user.pk},
 | 
				
			||||||
            ),
 | 
					            )
 | 
				
			||||||
            data={"reason": "some reason"},
 | 
					 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.assertEqual(response.status_code, 201)
 | 
					        self.assertEqual(response.status_code, 201)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -77,8 +75,7 @@ class TestImpersonation(APITestCase):
 | 
				
			|||||||
            reverse(
 | 
					            reverse(
 | 
				
			||||||
                "authentik_api:user-impersonate",
 | 
					                "authentik_api:user-impersonate",
 | 
				
			||||||
                kwargs={"pk": self.other_user.pk},
 | 
					                kwargs={"pk": self.other_user.pk},
 | 
				
			||||||
            ),
 | 
					            )
 | 
				
			||||||
            data={"reason": "some reason"},
 | 
					 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.assertEqual(response.status_code, 201)
 | 
					        self.assertEqual(response.status_code, 201)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -92,8 +89,7 @@ class TestImpersonation(APITestCase):
 | 
				
			|||||||
        self.client.force_login(self.other_user)
 | 
					        self.client.force_login(self.other_user)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        response = self.client.post(
 | 
					        response = self.client.post(
 | 
				
			||||||
            reverse("authentik_api:user-impersonate", kwargs={"pk": self.user.pk}),
 | 
					            reverse("authentik_api:user-impersonate", kwargs={"pk": self.user.pk})
 | 
				
			||||||
            data={"reason": "some reason"},
 | 
					 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.assertEqual(response.status_code, 403)
 | 
					        self.assertEqual(response.status_code, 403)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -109,8 +105,7 @@ class TestImpersonation(APITestCase):
 | 
				
			|||||||
        self.client.force_login(self.user)
 | 
					        self.client.force_login(self.user)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        response = self.client.post(
 | 
					        response = self.client.post(
 | 
				
			||||||
            reverse("authentik_api:user-impersonate", kwargs={"pk": self.other_user.pk}),
 | 
					            reverse("authentik_api:user-impersonate", kwargs={"pk": self.other_user.pk})
 | 
				
			||||||
            data={"reason": "some reason"},
 | 
					 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.assertEqual(response.status_code, 401)
 | 
					        self.assertEqual(response.status_code, 401)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -123,22 +118,7 @@ class TestImpersonation(APITestCase):
 | 
				
			|||||||
        self.client.force_login(self.user)
 | 
					        self.client.force_login(self.user)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        response = self.client.post(
 | 
					        response = self.client.post(
 | 
				
			||||||
            reverse("authentik_api:user-impersonate", kwargs={"pk": self.user.pk}),
 | 
					            reverse("authentik_api:user-impersonate", kwargs={"pk": self.user.pk})
 | 
				
			||||||
            data={"reason": "some reason"},
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertEqual(response.status_code, 401)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        response = self.client.get(reverse("authentik_api:user-me"))
 | 
					 | 
				
			||||||
        response_body = loads(response.content.decode())
 | 
					 | 
				
			||||||
        self.assertEqual(response_body["user"]["username"], self.user.username)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_impersonate_reason_required(self):
 | 
					 | 
				
			||||||
        """test impersonation that user must provide reason"""
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        response = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:user-impersonate", kwargs={"pk": self.user.pk}),
 | 
					 | 
				
			||||||
            data={"reason": ""},
 | 
					 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.assertEqual(response.status_code, 401)
 | 
					        self.assertEqual(response.status_code, 401)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -81,22 +81,6 @@ class TestSourceFlowManager(TestCase):
 | 
				
			|||||||
            reverse("authentik_core:if-user") + "#/settings;page-sources",
 | 
					            reverse("authentik_core:if-user") + "#/settings;page-sources",
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def test_authenticated_auth(self):
 | 
					 | 
				
			||||||
        """Test authenticated user linking"""
 | 
					 | 
				
			||||||
        user = User.objects.create(username="foo", email="foo@bar.baz")
 | 
					 | 
				
			||||||
        UserOAuthSourceConnection.objects.create(
 | 
					 | 
				
			||||||
            user=user, source=self.source, identifier=self.identifier
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        request = get_request("/", user=user)
 | 
					 | 
				
			||||||
        flow_manager = OAuthSourceFlowManager(
 | 
					 | 
				
			||||||
            self.source, request, self.identifier, {"info": {}}, {}
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        action, connection = flow_manager.get_action()
 | 
					 | 
				
			||||||
        self.assertEqual(action, Action.AUTH)
 | 
					 | 
				
			||||||
        self.assertIsNotNone(connection.pk)
 | 
					 | 
				
			||||||
        response = flow_manager.get_flow()
 | 
					 | 
				
			||||||
        self.assertEqual(response.status_code, 302)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_unauthenticated_link(self):
 | 
					    def test_unauthenticated_link(self):
 | 
				
			||||||
        """Test un-authenticated user linking"""
 | 
					        """Test un-authenticated user linking"""
 | 
				
			||||||
        flow_manager = OAuthSourceFlowManager(
 | 
					        flow_manager = OAuthSourceFlowManager(
 | 
				
			||||||
 | 
				
			|||||||
@ -1,13 +1,11 @@
 | 
				
			|||||||
"""Test Transactional API"""
 | 
					"""Test Transactional API"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from django.urls import reverse
 | 
					from django.urls import reverse
 | 
				
			||||||
from guardian.shortcuts import assign_perm
 | 
					 | 
				
			||||||
from rest_framework.test import APITestCase
 | 
					from rest_framework.test import APITestCase
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik.core.models import Application, Group
 | 
					from authentik.core.models import Application
 | 
				
			||||||
from authentik.core.tests.utils import create_test_flow, create_test_user
 | 
					from authentik.core.tests.utils import create_test_admin_user, create_test_flow
 | 
				
			||||||
from authentik.lib.generators import generate_id
 | 
					from authentik.lib.generators import generate_id
 | 
				
			||||||
from authentik.policies.models import PolicyBinding
 | 
					 | 
				
			||||||
from authentik.providers.oauth2.models import OAuth2Provider
 | 
					from authentik.providers.oauth2.models import OAuth2Provider
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -15,9 +13,7 @@ class TestTransactionalApplicationsAPI(APITestCase):
 | 
				
			|||||||
    """Test Transactional API"""
 | 
					    """Test Transactional API"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def setUp(self) -> None:
 | 
					    def setUp(self) -> None:
 | 
				
			||||||
        self.user = create_test_user()
 | 
					        self.user = create_test_admin_user()
 | 
				
			||||||
        assign_perm("authentik_core.add_application", self.user)
 | 
					 | 
				
			||||||
        assign_perm("authentik_providers_oauth2.add_oauth2provider", self.user)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def test_create_transactional(self):
 | 
					    def test_create_transactional(self):
 | 
				
			||||||
        """Test transactional Application + provider creation"""
 | 
					        """Test transactional Application + provider creation"""
 | 
				
			||||||
@ -35,7 +31,6 @@ class TestTransactionalApplicationsAPI(APITestCase):
 | 
				
			|||||||
                    "name": uid,
 | 
					                    "name": uid,
 | 
				
			||||||
                    "authorization_flow": str(create_test_flow().pk),
 | 
					                    "authorization_flow": str(create_test_flow().pk),
 | 
				
			||||||
                    "invalidation_flow": str(create_test_flow().pk),
 | 
					                    "invalidation_flow": str(create_test_flow().pk),
 | 
				
			||||||
                    "redirect_uris": [],
 | 
					 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
@ -46,66 +41,6 @@ class TestTransactionalApplicationsAPI(APITestCase):
 | 
				
			|||||||
        self.assertIsNotNone(app)
 | 
					        self.assertIsNotNone(app)
 | 
				
			||||||
        self.assertEqual(app.provider.pk, provider.pk)
 | 
					        self.assertEqual(app.provider.pk, provider.pk)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def test_create_transactional_permission_denied(self):
 | 
					 | 
				
			||||||
        """Test transactional Application + provider creation (missing permissions)"""
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
        uid = generate_id()
 | 
					 | 
				
			||||||
        response = self.client.put(
 | 
					 | 
				
			||||||
            reverse("authentik_api:core-transactional-application"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "app": {
 | 
					 | 
				
			||||||
                    "name": uid,
 | 
					 | 
				
			||||||
                    "slug": uid,
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
                "provider_model": "authentik_providers_saml.samlprovider",
 | 
					 | 
				
			||||||
                "provider": {
 | 
					 | 
				
			||||||
                    "name": uid,
 | 
					 | 
				
			||||||
                    "authorization_flow": str(create_test_flow().pk),
 | 
					 | 
				
			||||||
                    "invalidation_flow": str(create_test_flow().pk),
 | 
					 | 
				
			||||||
                    "acs_url": "https://goauthentik.io",
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertJSONEqual(
 | 
					 | 
				
			||||||
            response.content.decode(),
 | 
					 | 
				
			||||||
            {"provider": "User lacks permission to create authentik_providers_saml.samlprovider"},
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_create_transactional_bindings(self):
 | 
					 | 
				
			||||||
        """Test transactional Application + provider creation"""
 | 
					 | 
				
			||||||
        assign_perm("authentik_policies.add_policybinding", self.user)
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
        uid = generate_id()
 | 
					 | 
				
			||||||
        group = Group.objects.create(name=generate_id())
 | 
					 | 
				
			||||||
        authorization_flow = create_test_flow()
 | 
					 | 
				
			||||||
        response = self.client.put(
 | 
					 | 
				
			||||||
            reverse("authentik_api:core-transactional-application"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "app": {
 | 
					 | 
				
			||||||
                    "name": uid,
 | 
					 | 
				
			||||||
                    "slug": uid,
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
                "provider_model": "authentik_providers_oauth2.oauth2provider",
 | 
					 | 
				
			||||||
                "provider": {
 | 
					 | 
				
			||||||
                    "name": uid,
 | 
					 | 
				
			||||||
                    "authorization_flow": str(authorization_flow.pk),
 | 
					 | 
				
			||||||
                    "invalidation_flow": str(authorization_flow.pk),
 | 
					 | 
				
			||||||
                    "redirect_uris": [],
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
                "policy_bindings": [{"group": group.pk, "order": 0}],
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertJSONEqual(response.content.decode(), {"applied": True, "logs": []})
 | 
					 | 
				
			||||||
        provider = OAuth2Provider.objects.filter(name=uid).first()
 | 
					 | 
				
			||||||
        self.assertIsNotNone(provider)
 | 
					 | 
				
			||||||
        app = Application.objects.filter(slug=uid).first()
 | 
					 | 
				
			||||||
        self.assertIsNotNone(app)
 | 
					 | 
				
			||||||
        self.assertEqual(app.provider.pk, provider.pk)
 | 
					 | 
				
			||||||
        binding = PolicyBinding.objects.filter(target=app).first()
 | 
					 | 
				
			||||||
        self.assertIsNotNone(binding)
 | 
					 | 
				
			||||||
        self.assertEqual(binding.target, app)
 | 
					 | 
				
			||||||
        self.assertEqual(binding.group, group)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_create_transactional_invalid(self):
 | 
					    def test_create_transactional_invalid(self):
 | 
				
			||||||
        """Test transactional Application + provider creation"""
 | 
					        """Test transactional Application + provider creation"""
 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					        self.client.force_login(self.user)
 | 
				
			||||||
@ -122,7 +57,6 @@ class TestTransactionalApplicationsAPI(APITestCase):
 | 
				
			|||||||
                    "name": uid,
 | 
					                    "name": uid,
 | 
				
			||||||
                    "authorization_flow": "",
 | 
					                    "authorization_flow": "",
 | 
				
			||||||
                    "invalidation_flow": "",
 | 
					                    "invalidation_flow": "",
 | 
				
			||||||
                    "redirect_uris": [],
 | 
					 | 
				
			||||||
                },
 | 
					                },
 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
@ -135,32 +69,3 @@ class TestTransactionalApplicationsAPI(APITestCase):
 | 
				
			|||||||
                }
 | 
					                }
 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_create_transactional_duplicate_name_provider(self):
 | 
					 | 
				
			||||||
        """Test transactional Application + provider creation"""
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
        uid = generate_id()
 | 
					 | 
				
			||||||
        OAuth2Provider.objects.create(
 | 
					 | 
				
			||||||
            name=uid,
 | 
					 | 
				
			||||||
            authorization_flow=create_test_flow(),
 | 
					 | 
				
			||||||
            invalidation_flow=create_test_flow(),
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        response = self.client.put(
 | 
					 | 
				
			||||||
            reverse("authentik_api:core-transactional-application"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "app": {
 | 
					 | 
				
			||||||
                    "name": uid,
 | 
					 | 
				
			||||||
                    "slug": uid,
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
                "provider_model": "authentik_providers_oauth2.oauth2provider",
 | 
					 | 
				
			||||||
                "provider": {
 | 
					 | 
				
			||||||
                    "name": uid,
 | 
					 | 
				
			||||||
                    "authorization_flow": str(create_test_flow().pk),
 | 
					 | 
				
			||||||
                    "invalidation_flow": str(create_test_flow().pk),
 | 
					 | 
				
			||||||
                },
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertJSONEqual(
 | 
					 | 
				
			||||||
            response.content.decode(),
 | 
					 | 
				
			||||||
            {"provider": {"name": ["State is set to must_created and object exists already"]}},
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -6,7 +6,6 @@ from django.conf import settings
 | 
				
			|||||||
from django.contrib.auth.decorators import login_required
 | 
					from django.contrib.auth.decorators import login_required
 | 
				
			||||||
from django.urls import path
 | 
					from django.urls import path
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik.core.api.application_entitlements import ApplicationEntitlementViewSet
 | 
					 | 
				
			||||||
from authentik.core.api.applications import ApplicationViewSet
 | 
					from authentik.core.api.applications import ApplicationViewSet
 | 
				
			||||||
from authentik.core.api.authenticated_sessions import AuthenticatedSessionViewSet
 | 
					from authentik.core.api.authenticated_sessions import AuthenticatedSessionViewSet
 | 
				
			||||||
from authentik.core.api.devices import AdminDeviceViewSet, DeviceViewSet
 | 
					from authentik.core.api.devices import AdminDeviceViewSet, DeviceViewSet
 | 
				
			||||||
@ -70,7 +69,6 @@ urlpatterns = [
 | 
				
			|||||||
api_urlpatterns = [
 | 
					api_urlpatterns = [
 | 
				
			||||||
    ("core/authenticated_sessions", AuthenticatedSessionViewSet),
 | 
					    ("core/authenticated_sessions", AuthenticatedSessionViewSet),
 | 
				
			||||||
    ("core/applications", ApplicationViewSet),
 | 
					    ("core/applications", ApplicationViewSet),
 | 
				
			||||||
    ("core/application_entitlements", ApplicationEntitlementViewSet),
 | 
					 | 
				
			||||||
    path(
 | 
					    path(
 | 
				
			||||||
        "core/transactional/applications/",
 | 
					        "core/transactional/applications/",
 | 
				
			||||||
        TransactionalApplicationView.as_view(),
 | 
					        TransactionalApplicationView.as_view(),
 | 
				
			||||||
 | 
				
			|||||||
@ -17,8 +17,10 @@ from authentik.flows.planner import PLAN_CONTEXT_APPLICATION, FlowPlanner
 | 
				
			|||||||
from authentik.flows.stage import ChallengeStageView
 | 
					from authentik.flows.stage import ChallengeStageView
 | 
				
			||||||
from authentik.flows.views.executor import (
 | 
					from authentik.flows.views.executor import (
 | 
				
			||||||
    SESSION_KEY_APPLICATION_PRE,
 | 
					    SESSION_KEY_APPLICATION_PRE,
 | 
				
			||||||
 | 
					    SESSION_KEY_PLAN,
 | 
				
			||||||
    ToDefaultFlow,
 | 
					    ToDefaultFlow,
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					from authentik.lib.utils.urls import redirect_with_qs
 | 
				
			||||||
from authentik.stages.consent.stage import (
 | 
					from authentik.stages.consent.stage import (
 | 
				
			||||||
    PLAN_CONTEXT_CONSENT_HEADER,
 | 
					    PLAN_CONTEXT_CONSENT_HEADER,
 | 
				
			||||||
    PLAN_CONTEXT_CONSENT_PERMISSIONS,
 | 
					    PLAN_CONTEXT_CONSENT_PERMISSIONS,
 | 
				
			||||||
@ -56,7 +58,8 @@ class RedirectToAppLaunch(View):
 | 
				
			|||||||
        except FlowNonApplicableException:
 | 
					        except FlowNonApplicableException:
 | 
				
			||||||
            raise Http404 from None
 | 
					            raise Http404 from None
 | 
				
			||||||
        plan.insert_stage(in_memory_stage(RedirectToAppStage))
 | 
					        plan.insert_stage(in_memory_stage(RedirectToAppStage))
 | 
				
			||||||
        return plan.to_redirect(request, flow)
 | 
					        request.session[SESSION_KEY_PLAN] = plan
 | 
				
			||||||
 | 
					        return redirect_with_qs("authentik_core:if-flow", request.GET, flow_slug=flow.slug)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class RedirectToAppStage(ChallengeStageView):
 | 
					class RedirectToAppStage(ChallengeStageView):
 | 
				
			||||||
 | 
				
			|||||||
@ -16,7 +16,6 @@ from authentik.api.v3.config import ConfigView
 | 
				
			|||||||
from authentik.brands.api import CurrentBrandSerializer
 | 
					from authentik.brands.api import CurrentBrandSerializer
 | 
				
			||||||
from authentik.brands.models import Brand
 | 
					from authentik.brands.models import Brand
 | 
				
			||||||
from authentik.core.models import UserTypes
 | 
					from authentik.core.models import UserTypes
 | 
				
			||||||
from authentik.lib.config import CONFIG
 | 
					 | 
				
			||||||
from authentik.policies.denied import AccessDeniedResponse
 | 
					from authentik.policies.denied import AccessDeniedResponse
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -52,7 +51,6 @@ class InterfaceView(TemplateView):
 | 
				
			|||||||
        kwargs["version_subdomain"] = f"version-{LOCAL_VERSION.major}-{LOCAL_VERSION.minor}"
 | 
					        kwargs["version_subdomain"] = f"version-{LOCAL_VERSION.major}-{LOCAL_VERSION.minor}"
 | 
				
			||||||
        kwargs["build"] = get_build_hash()
 | 
					        kwargs["build"] = get_build_hash()
 | 
				
			||||||
        kwargs["url_kwargs"] = self.kwargs
 | 
					        kwargs["url_kwargs"] = self.kwargs
 | 
				
			||||||
        kwargs["base_url"] = self.request.build_absolute_uri(CONFIG.get("web.path", "/"))
 | 
					 | 
				
			||||||
        return super().get_context_data(**kwargs)
 | 
					        return super().get_context_data(**kwargs)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -24,10 +24,10 @@ from rest_framework.fields import (
 | 
				
			|||||||
from rest_framework.filters import OrderingFilter, SearchFilter
 | 
					from rest_framework.filters import OrderingFilter, SearchFilter
 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
from rest_framework.response import Response
 | 
					from rest_framework.response import Response
 | 
				
			||||||
from rest_framework.validators import UniqueValidator
 | 
					 | 
				
			||||||
from rest_framework.viewsets import ModelViewSet
 | 
					from rest_framework.viewsets import ModelViewSet
 | 
				
			||||||
from structlog.stdlib import get_logger
 | 
					from structlog.stdlib import get_logger
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import SecretKeyFilter
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
from authentik.core.api.utils import ModelSerializer, PassiveSerializer
 | 
					from authentik.core.api.utils import ModelSerializer, PassiveSerializer
 | 
				
			||||||
from authentik.crypto.apps import MANAGED_KEY
 | 
					from authentik.crypto.apps import MANAGED_KEY
 | 
				
			||||||
@ -35,7 +35,7 @@ from authentik.crypto.builder import CertificateBuilder, PrivateKeyAlg
 | 
				
			|||||||
from authentik.crypto.models import CertificateKeyPair
 | 
					from authentik.crypto.models import CertificateKeyPair
 | 
				
			||||||
from authentik.events.models import Event, EventAction
 | 
					from authentik.events.models import Event, EventAction
 | 
				
			||||||
from authentik.rbac.decorators import permission_required
 | 
					from authentik.rbac.decorators import permission_required
 | 
				
			||||||
from authentik.rbac.filters import ObjectFilter, SecretKeyFilter
 | 
					from authentik.rbac.filters import ObjectFilter
 | 
				
			||||||
 | 
					
 | 
				
			||||||
LOGGER = get_logger()
 | 
					LOGGER = get_logger()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -181,10 +181,7 @@ class CertificateDataSerializer(PassiveSerializer):
 | 
				
			|||||||
class CertificateGenerationSerializer(PassiveSerializer):
 | 
					class CertificateGenerationSerializer(PassiveSerializer):
 | 
				
			||||||
    """Certificate generation parameters"""
 | 
					    """Certificate generation parameters"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    common_name = CharField(
 | 
					    common_name = CharField()
 | 
				
			||||||
        validators=[UniqueValidator(queryset=CertificateKeyPair.objects.all())],
 | 
					 | 
				
			||||||
        source="name",
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    subject_alt_name = CharField(required=False, allow_blank=True, label=_("Subject-alt name"))
 | 
					    subject_alt_name = CharField(required=False, allow_blank=True, label=_("Subject-alt name"))
 | 
				
			||||||
    validity_days = IntegerField(initial=365)
 | 
					    validity_days = IntegerField(initial=365)
 | 
				
			||||||
    alg = ChoiceField(default=PrivateKeyAlg.RSA, choices=PrivateKeyAlg.choices)
 | 
					    alg = ChoiceField(default=PrivateKeyAlg.RSA, choices=PrivateKeyAlg.choices)
 | 
				
			||||||
@ -245,10 +242,11 @@ class CertificateKeyPairViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
    def generate(self, request: Request) -> Response:
 | 
					    def generate(self, request: Request) -> Response:
 | 
				
			||||||
        """Generate a new, self-signed certificate-key pair"""
 | 
					        """Generate a new, self-signed certificate-key pair"""
 | 
				
			||||||
        data = CertificateGenerationSerializer(data=request.data)
 | 
					        data = CertificateGenerationSerializer(data=request.data)
 | 
				
			||||||
        data.is_valid(raise_exception=True)
 | 
					        if not data.is_valid():
 | 
				
			||||||
 | 
					            return Response(data.errors, status=400)
 | 
				
			||||||
        raw_san = data.validated_data.get("subject_alt_name", "")
 | 
					        raw_san = data.validated_data.get("subject_alt_name", "")
 | 
				
			||||||
        sans = raw_san.split(",") if raw_san != "" else []
 | 
					        sans = raw_san.split(",") if raw_san != "" else []
 | 
				
			||||||
        builder = CertificateBuilder(data.validated_data["name"])
 | 
					        builder = CertificateBuilder(data.validated_data["common_name"])
 | 
				
			||||||
        builder.alg = data.validated_data["alg"]
 | 
					        builder.alg = data.validated_data["alg"]
 | 
				
			||||||
        builder.build(
 | 
					        builder.build(
 | 
				
			||||||
            subject_alt_names=sans,
 | 
					            subject_alt_names=sans,
 | 
				
			||||||
 | 
				
			|||||||
@ -85,5 +85,5 @@ def certificate_discovery(self: SystemTask):
 | 
				
			|||||||
        if dirty:
 | 
					        if dirty:
 | 
				
			||||||
            cert.save()
 | 
					            cert.save()
 | 
				
			||||||
    self.set_status(
 | 
					    self.set_status(
 | 
				
			||||||
        TaskStatus.SUCCESSFUL, _("Successfully imported {count} files.".format(count=discovered))
 | 
					        TaskStatus.SUCCESSFUL, _("Successfully imported %(count)d files." % {"count": discovered})
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
 | 
				
			|||||||
@ -18,7 +18,7 @@ from authentik.crypto.models import CertificateKeyPair
 | 
				
			|||||||
from authentik.crypto.tasks import MANAGED_DISCOVERED, certificate_discovery
 | 
					from authentik.crypto.tasks import MANAGED_DISCOVERED, certificate_discovery
 | 
				
			||||||
from authentik.lib.config import CONFIG
 | 
					from authentik.lib.config import CONFIG
 | 
				
			||||||
from authentik.lib.generators import generate_id, generate_key
 | 
					from authentik.lib.generators import generate_id, generate_key
 | 
				
			||||||
from authentik.providers.oauth2.models import OAuth2Provider, RedirectURI, RedirectURIMatchingMode
 | 
					from authentik.providers.oauth2.models import OAuth2Provider
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class TestCrypto(APITestCase):
 | 
					class TestCrypto(APITestCase):
 | 
				
			||||||
@ -89,17 +89,6 @@ class TestCrypto(APITestCase):
 | 
				
			|||||||
        self.assertIsInstance(ext[1], DNSName)
 | 
					        self.assertIsInstance(ext[1], DNSName)
 | 
				
			||||||
        self.assertEqual(ext[1].value, "baz")
 | 
					        self.assertEqual(ext[1].value, "baz")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def test_builder_api_duplicate(self):
 | 
					 | 
				
			||||||
        """Test Builder (via API)"""
 | 
					 | 
				
			||||||
        cert = create_test_cert()
 | 
					 | 
				
			||||||
        self.client.force_login(create_test_admin_user())
 | 
					 | 
				
			||||||
        res = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:certificatekeypair-generate"),
 | 
					 | 
				
			||||||
            data={"common_name": cert.name, "subject_alt_name": "bar,baz", "validity_days": 3},
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertEqual(res.status_code, 400)
 | 
					 | 
				
			||||||
        self.assertJSONEqual(res.content, {"common_name": ["This field must be unique."]})
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def test_builder_api_empty_san(self):
 | 
					    def test_builder_api_empty_san(self):
 | 
				
			||||||
        """Test Builder (via API)"""
 | 
					        """Test Builder (via API)"""
 | 
				
			||||||
        self.client.force_login(create_test_admin_user())
 | 
					        self.client.force_login(create_test_admin_user())
 | 
				
			||||||
@ -274,7 +263,7 @@ class TestCrypto(APITestCase):
 | 
				
			|||||||
            client_id="test",
 | 
					            client_id="test",
 | 
				
			||||||
            client_secret=generate_key(),
 | 
					            client_secret=generate_key(),
 | 
				
			||||||
            authorization_flow=create_test_flow(),
 | 
					            authorization_flow=create_test_flow(),
 | 
				
			||||||
            redirect_uris=[RedirectURI(RedirectURIMatchingMode.STRICT, "http://localhost")],
 | 
					            redirect_uris="http://localhost",
 | 
				
			||||||
            signing_key=keypair,
 | 
					            signing_key=keypair,
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        response = self.client.get(
 | 
					        response = self.client.get(
 | 
				
			||||||
@ -306,7 +295,7 @@ class TestCrypto(APITestCase):
 | 
				
			|||||||
            client_id="test",
 | 
					            client_id="test",
 | 
				
			||||||
            client_secret=generate_key(),
 | 
					            client_secret=generate_key(),
 | 
				
			||||||
            authorization_flow=create_test_flow(),
 | 
					            authorization_flow=create_test_flow(),
 | 
				
			||||||
            redirect_uris=[RedirectURI(RedirectURIMatchingMode.STRICT, "http://localhost")],
 | 
					            redirect_uris="http://localhost",
 | 
				
			||||||
            signing_key=keypair,
 | 
					            signing_key=keypair,
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        response = self.client.get(
 | 
					        response = self.client.get(
 | 
				
			||||||
 | 
				
			|||||||
@ -6,7 +6,6 @@ from django.http import HttpRequest, HttpResponse, JsonResponse
 | 
				
			|||||||
from django.urls import resolve
 | 
					from django.urls import resolve
 | 
				
			||||||
from structlog.stdlib import BoundLogger, get_logger
 | 
					from structlog.stdlib import BoundLogger, get_logger
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik.core.api.users import UserViewSet
 | 
					 | 
				
			||||||
from authentik.enterprise.api import LicenseViewSet
 | 
					from authentik.enterprise.api import LicenseViewSet
 | 
				
			||||||
from authentik.enterprise.license import LicenseKey
 | 
					from authentik.enterprise.license import LicenseKey
 | 
				
			||||||
from authentik.enterprise.models import LicenseUsageStatus
 | 
					from authentik.enterprise.models import LicenseUsageStatus
 | 
				
			||||||
@ -60,9 +59,6 @@ class EnterpriseMiddleware:
 | 
				
			|||||||
        # Flow executor is mounted as an API path but explicitly allowed
 | 
					        # Flow executor is mounted as an API path but explicitly allowed
 | 
				
			||||||
        if request.resolver_match._func_path == class_to_path(FlowExecutorView):
 | 
					        if request.resolver_match._func_path == class_to_path(FlowExecutorView):
 | 
				
			||||||
            return True
 | 
					            return True
 | 
				
			||||||
        # Always allow making changes to users, even in case the license has ben exceeded
 | 
					 | 
				
			||||||
        if request.resolver_match._func_path == class_to_path(UserViewSet):
 | 
					 | 
				
			||||||
            return True
 | 
					 | 
				
			||||||
        # Only apply these restrictions to the API
 | 
					        # Only apply these restrictions to the API
 | 
				
			||||||
        if "authentik_api" not in request.resolver_match.app_names:
 | 
					        if "authentik_api" not in request.resolver_match.app_names:
 | 
				
			||||||
            return True
 | 
					            return True
 | 
				
			||||||
 | 
				
			|||||||
@ -1,27 +0,0 @@
 | 
				
			|||||||
# Generated by Django 5.0.10 on 2025-01-13 18:05
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.db import migrations, models
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Migration(migrations.Migration):
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    dependencies = [
 | 
					 | 
				
			||||||
        ("authentik_enterprise", "0003_remove_licenseusage_within_limits_and_more"),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    operations = [
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="licenseusage",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expires"], name="authentik_e_expires_3f2956_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="licenseusage",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expiring"], name="authentik_e_expirin_11d3d7_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="licenseusage",
 | 
					 | 
				
			||||||
            index=models.Index(
 | 
					 | 
				
			||||||
                fields=["expiring", "expires"], name="authentik_e_expirin_4d558f_idx"
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
@ -93,4 +93,3 @@ class LicenseUsage(ExpiringModel):
 | 
				
			|||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        verbose_name = _("License Usage")
 | 
					        verbose_name = _("License Usage")
 | 
				
			||||||
        verbose_name_plural = _("License Usage Records")
 | 
					        verbose_name_plural = _("License Usage Records")
 | 
				
			||||||
        indexes = ExpiringModel.Meta.indexes
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -1,8 +1,11 @@
 | 
				
			|||||||
"""RAC Provider API Views"""
 | 
					"""RAC Provider API Views"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from django_filters.rest_framework.backends import DjangoFilterBackend
 | 
				
			||||||
from rest_framework import mixins
 | 
					from rest_framework import mixins
 | 
				
			||||||
 | 
					from rest_framework.filters import OrderingFilter, SearchFilter
 | 
				
			||||||
from rest_framework.viewsets import GenericViewSet
 | 
					from rest_framework.viewsets import GenericViewSet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import OwnerFilter, OwnerSuperuserPermissions
 | 
				
			||||||
from authentik.core.api.groups import GroupMemberSerializer
 | 
					from authentik.core.api.groups import GroupMemberSerializer
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
from authentik.core.api.utils import ModelSerializer
 | 
					from authentik.core.api.utils import ModelSerializer
 | 
				
			||||||
@ -31,6 +34,12 @@ class ConnectionTokenSerializer(EnterpriseRequiredMixin, ModelSerializer):
 | 
				
			|||||||
        ]
 | 
					        ]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					class ConnectionTokenOwnerFilter(OwnerFilter):
 | 
				
			||||||
 | 
					    """Owner filter for connection tokens (checks session's user)"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    owner_key = "session__user"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class ConnectionTokenViewSet(
 | 
					class ConnectionTokenViewSet(
 | 
				
			||||||
    mixins.RetrieveModelMixin,
 | 
					    mixins.RetrieveModelMixin,
 | 
				
			||||||
    mixins.UpdateModelMixin,
 | 
					    mixins.UpdateModelMixin,
 | 
				
			||||||
@ -46,4 +55,10 @@ class ConnectionTokenViewSet(
 | 
				
			|||||||
    filterset_fields = ["endpoint", "session__user", "provider"]
 | 
					    filterset_fields = ["endpoint", "session__user", "provider"]
 | 
				
			||||||
    search_fields = ["endpoint__name", "provider__name"]
 | 
					    search_fields = ["endpoint__name", "provider__name"]
 | 
				
			||||||
    ordering = ["endpoint__name", "provider__name"]
 | 
					    ordering = ["endpoint__name", "provider__name"]
 | 
				
			||||||
    owner_field = "session__user"
 | 
					    permission_classes = [OwnerSuperuserPermissions]
 | 
				
			||||||
 | 
					    filter_backends = [
 | 
				
			||||||
 | 
					        ConnectionTokenOwnerFilter,
 | 
				
			||||||
 | 
					        DjangoFilterBackend,
 | 
				
			||||||
 | 
					        OrderingFilter,
 | 
				
			||||||
 | 
					        SearchFilter,
 | 
				
			||||||
 | 
					    ]
 | 
				
			||||||
 | 
				
			|||||||
@ -96,7 +96,7 @@ class EndpointViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
                OpenApiTypes.STR,
 | 
					                OpenApiTypes.STR,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
            OpenApiParameter(
 | 
					            OpenApiParameter(
 | 
				
			||||||
                name="list_rbac",
 | 
					                name="superuser_full_list",
 | 
				
			||||||
                location=OpenApiParameter.QUERY,
 | 
					                location=OpenApiParameter.QUERY,
 | 
				
			||||||
                type=OpenApiTypes.BOOL,
 | 
					                type=OpenApiTypes.BOOL,
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
@ -110,8 +110,8 @@ class EndpointViewSet(UsedByMixin, ModelViewSet):
 | 
				
			|||||||
        """List accessible endpoints"""
 | 
					        """List accessible endpoints"""
 | 
				
			||||||
        should_cache = request.GET.get("search", "") == ""
 | 
					        should_cache = request.GET.get("search", "") == ""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        list_rbac = str(request.GET.get("list_rbac", "false")).lower() == "true"
 | 
					        superuser_full_list = str(request.GET.get("superuser_full_list", "false")).lower() == "true"
 | 
				
			||||||
        if list_rbac:
 | 
					        if superuser_full_list and request.user.is_superuser:
 | 
				
			||||||
            return super().list(request)
 | 
					            return super().list(request)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        queryset = self._filter_queryset_for_list(self.get_queryset())
 | 
					        queryset = self._filter_queryset_for_list(self.get_queryset())
 | 
				
			||||||
 | 
				
			|||||||
@ -16,28 +16,13 @@ class RACProviderSerializer(EnterpriseRequiredMixin, ProviderSerializer):
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        model = RACProvider
 | 
					        model = RACProvider
 | 
				
			||||||
        fields = [
 | 
					        fields = ProviderSerializer.Meta.fields + [
 | 
				
			||||||
            "pk",
 | 
					 | 
				
			||||||
            "name",
 | 
					 | 
				
			||||||
            "authentication_flow",
 | 
					 | 
				
			||||||
            "authorization_flow",
 | 
					 | 
				
			||||||
            "property_mappings",
 | 
					 | 
				
			||||||
            "component",
 | 
					 | 
				
			||||||
            "assigned_application_slug",
 | 
					 | 
				
			||||||
            "assigned_application_name",
 | 
					 | 
				
			||||||
            "assigned_backchannel_application_slug",
 | 
					 | 
				
			||||||
            "assigned_backchannel_application_name",
 | 
					 | 
				
			||||||
            "verbose_name",
 | 
					 | 
				
			||||||
            "verbose_name_plural",
 | 
					 | 
				
			||||||
            "meta_model_name",
 | 
					 | 
				
			||||||
            "settings",
 | 
					            "settings",
 | 
				
			||||||
            "outpost_set",
 | 
					            "outpost_set",
 | 
				
			||||||
            "connection_expiry",
 | 
					            "connection_expiry",
 | 
				
			||||||
            "delete_token_on_disconnect",
 | 
					            "delete_token_on_disconnect",
 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
        extra_kwargs = {
 | 
					        extra_kwargs = ProviderSerializer.Meta.extra_kwargs
 | 
				
			||||||
            "authorization_flow": {"required": True, "allow_null": False},
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class RACProviderViewSet(UsedByMixin, ModelViewSet):
 | 
					class RACProviderViewSet(UsedByMixin, ModelViewSet):
 | 
				
			||||||
 | 
				
			|||||||
@ -1,28 +0,0 @@
 | 
				
			|||||||
# Generated by Django 5.0.10 on 2025-01-13 18:05
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.db import migrations, models
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Migration(migrations.Migration):
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    dependencies = [
 | 
					 | 
				
			||||||
        ("authentik_core", "0042_authenticatedsession_authentik_c_expires_08251d_idx_and_more"),
 | 
					 | 
				
			||||||
        ("authentik_providers_rac", "0005_alter_racpropertymapping_options"),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    operations = [
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="connectiontoken",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expires"], name="authentik_p_expires_91f148_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="connectiontoken",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expiring"], name="authentik_p_expirin_59a5a7_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="connectiontoken",
 | 
					 | 
				
			||||||
            index=models.Index(
 | 
					 | 
				
			||||||
                fields=["expiring", "expires"], name="authentik_p_expirin_aed3ca_idx"
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
@ -159,9 +159,9 @@ class ConnectionToken(ExpiringModel):
 | 
				
			|||||||
            default_settings["port"] = str(port)
 | 
					            default_settings["port"] = str(port)
 | 
				
			||||||
        else:
 | 
					        else:
 | 
				
			||||||
            default_settings["hostname"] = self.endpoint.host
 | 
					            default_settings["hostname"] = self.endpoint.host
 | 
				
			||||||
        if self.endpoint.protocol == Protocols.RDP:
 | 
					        default_settings["client-name"] = "authentik"
 | 
				
			||||||
            default_settings["resize-method"] = "display-update"
 | 
					        # default_settings["enable-drive"] = "true"
 | 
				
			||||||
        default_settings["client-name"] = f"authentik - {self.session.user}"
 | 
					        # default_settings["drive-name"] = "authentik"
 | 
				
			||||||
        settings = {}
 | 
					        settings = {}
 | 
				
			||||||
        always_merger.merge(settings, default_settings)
 | 
					        always_merger.merge(settings, default_settings)
 | 
				
			||||||
        always_merger.merge(settings, self.endpoint.provider.settings)
 | 
					        always_merger.merge(settings, self.endpoint.provider.settings)
 | 
				
			||||||
@ -211,4 +211,3 @@ class ConnectionToken(ExpiringModel):
 | 
				
			|||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        verbose_name = _("RAC Connection token")
 | 
					        verbose_name = _("RAC Connection token")
 | 
				
			||||||
        verbose_name_plural = _("RAC Connection tokens")
 | 
					        verbose_name_plural = _("RAC Connection tokens")
 | 
				
			||||||
        indexes = ExpiringModel.Meta.indexes
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -3,11 +3,11 @@
 | 
				
			|||||||
{% load authentik_core %}
 | 
					{% load authentik_core %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
{% block head %}
 | 
					{% block head %}
 | 
				
			||||||
<script src="{% versioned_script 'dist/enterprise/rac/index-%v.js' %}" type="module"></script>
 | 
					{% versioned_script "dist/enterprise/rac/index-%v.js" %}
 | 
				
			||||||
<meta name="theme-color" content="#18191a" media="(prefers-color-scheme: dark)">
 | 
					<meta name="theme-color" content="#18191a" media="(prefers-color-scheme: dark)">
 | 
				
			||||||
<meta name="theme-color" content="#ffffff" media="(prefers-color-scheme: light)">
 | 
					<meta name="theme-color" content="#ffffff" media="(prefers-color-scheme: light)">
 | 
				
			||||||
<link rel="icon" href="{{ tenant.branding_favicon_url }}">
 | 
					<link rel="icon" href="{{ tenant.branding_favicon }}">
 | 
				
			||||||
<link rel="shortcut icon" href="{{ tenant.branding_favicon_url }}">
 | 
					<link rel="shortcut icon" href="{{ tenant.branding_favicon }}">
 | 
				
			||||||
{% include "base/header_js.html" %}
 | 
					{% include "base/header_js.html" %}
 | 
				
			||||||
{% endblock %}
 | 
					{% endblock %}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -1,46 +0,0 @@
 | 
				
			|||||||
"""Test RAC Provider"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from datetime import timedelta
 | 
					 | 
				
			||||||
from time import mktime
 | 
					 | 
				
			||||||
from unittest.mock import MagicMock, patch
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.urls import reverse
 | 
					 | 
				
			||||||
from django.utils.timezone import now
 | 
					 | 
				
			||||||
from rest_framework.test import APITestCase
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from authentik.core.tests.utils import create_test_admin_user, create_test_flow
 | 
					 | 
				
			||||||
from authentik.enterprise.license import LicenseKey
 | 
					 | 
				
			||||||
from authentik.enterprise.models import License
 | 
					 | 
				
			||||||
from authentik.lib.generators import generate_id
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class TestAPI(APITestCase):
 | 
					 | 
				
			||||||
    """Test Provider API"""
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def setUp(self) -> None:
 | 
					 | 
				
			||||||
        self.user = create_test_admin_user()
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    @patch(
 | 
					 | 
				
			||||||
        "authentik.enterprise.license.LicenseKey.validate",
 | 
					 | 
				
			||||||
        MagicMock(
 | 
					 | 
				
			||||||
            return_value=LicenseKey(
 | 
					 | 
				
			||||||
                aud="",
 | 
					 | 
				
			||||||
                exp=int(mktime((now() + timedelta(days=3000)).timetuple())),
 | 
					 | 
				
			||||||
                name=generate_id(),
 | 
					 | 
				
			||||||
                internal_users=100,
 | 
					 | 
				
			||||||
                external_users=100,
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    def test_create(self):
 | 
					 | 
				
			||||||
        """Test creation of RAC Provider"""
 | 
					 | 
				
			||||||
        License.objects.create(key=generate_id())
 | 
					 | 
				
			||||||
        self.client.force_login(self.user)
 | 
					 | 
				
			||||||
        response = self.client.post(
 | 
					 | 
				
			||||||
            reverse("authentik_api:racprovider-list"),
 | 
					 | 
				
			||||||
            data={
 | 
					 | 
				
			||||||
                "name": generate_id(),
 | 
					 | 
				
			||||||
                "authorization_flow": create_test_flow().pk,
 | 
					 | 
				
			||||||
            },
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        self.assertEqual(response.status_code, 201)
 | 
					 | 
				
			||||||
@ -68,6 +68,7 @@ class TestEndpointsAPI(APITestCase):
 | 
				
			|||||||
                            "name": self.provider.name,
 | 
					                            "name": self.provider.name,
 | 
				
			||||||
                            "authentication_flow": None,
 | 
					                            "authentication_flow": None,
 | 
				
			||||||
                            "authorization_flow": None,
 | 
					                            "authorization_flow": None,
 | 
				
			||||||
 | 
					                            "invalidation_flow": None,
 | 
				
			||||||
                            "property_mappings": [],
 | 
					                            "property_mappings": [],
 | 
				
			||||||
                            "connection_expiry": "hours=8",
 | 
					                            "connection_expiry": "hours=8",
 | 
				
			||||||
                            "delete_token_on_disconnect": False,
 | 
					                            "delete_token_on_disconnect": False,
 | 
				
			||||||
@ -120,6 +121,7 @@ class TestEndpointsAPI(APITestCase):
 | 
				
			|||||||
                            "name": self.provider.name,
 | 
					                            "name": self.provider.name,
 | 
				
			||||||
                            "authentication_flow": None,
 | 
					                            "authentication_flow": None,
 | 
				
			||||||
                            "authorization_flow": None,
 | 
					                            "authorization_flow": None,
 | 
				
			||||||
 | 
					                            "invalidation_flow": None,
 | 
				
			||||||
                            "property_mappings": [],
 | 
					                            "property_mappings": [],
 | 
				
			||||||
                            "component": "ak-provider-rac-form",
 | 
					                            "component": "ak-provider-rac-form",
 | 
				
			||||||
                            "assigned_application_slug": self.app.slug,
 | 
					                            "assigned_application_slug": self.app.slug,
 | 
				
			||||||
@ -149,6 +151,7 @@ class TestEndpointsAPI(APITestCase):
 | 
				
			|||||||
                            "name": self.provider.name,
 | 
					                            "name": self.provider.name,
 | 
				
			||||||
                            "authentication_flow": None,
 | 
					                            "authentication_flow": None,
 | 
				
			||||||
                            "authorization_flow": None,
 | 
					                            "authorization_flow": None,
 | 
				
			||||||
 | 
					                            "invalidation_flow": None,
 | 
				
			||||||
                            "property_mappings": [],
 | 
					                            "property_mappings": [],
 | 
				
			||||||
                            "component": "ak-provider-rac-form",
 | 
					                            "component": "ak-provider-rac-form",
 | 
				
			||||||
                            "assigned_application_slug": self.app.slug,
 | 
					                            "assigned_application_slug": self.app.slug,
 | 
				
			||||||
 | 
				
			|||||||
@ -50,10 +50,9 @@ class TestModels(TransactionTestCase):
 | 
				
			|||||||
            {
 | 
					            {
 | 
				
			||||||
                "hostname": self.endpoint.host.split(":")[0],
 | 
					                "hostname": self.endpoint.host.split(":")[0],
 | 
				
			||||||
                "port": "1324",
 | 
					                "port": "1324",
 | 
				
			||||||
                "client-name": f"authentik - {self.user}",
 | 
					                "client-name": "authentik",
 | 
				
			||||||
                "drive-path": path,
 | 
					                "drive-path": path,
 | 
				
			||||||
                "create-drive-path": "true",
 | 
					                "create-drive-path": "true",
 | 
				
			||||||
                "resize-method": "display-update",
 | 
					 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        # Set settings in provider
 | 
					        # Set settings in provider
 | 
				
			||||||
@ -64,11 +63,10 @@ class TestModels(TransactionTestCase):
 | 
				
			|||||||
            {
 | 
					            {
 | 
				
			||||||
                "hostname": self.endpoint.host.split(":")[0],
 | 
					                "hostname": self.endpoint.host.split(":")[0],
 | 
				
			||||||
                "port": "1324",
 | 
					                "port": "1324",
 | 
				
			||||||
                "client-name": f"authentik - {self.user}",
 | 
					                "client-name": "authentik",
 | 
				
			||||||
                "drive-path": path,
 | 
					                "drive-path": path,
 | 
				
			||||||
                "create-drive-path": "true",
 | 
					                "create-drive-path": "true",
 | 
				
			||||||
                "level": "provider",
 | 
					                "level": "provider",
 | 
				
			||||||
                "resize-method": "display-update",
 | 
					 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        # Set settings in endpoint
 | 
					        # Set settings in endpoint
 | 
				
			||||||
@ -81,11 +79,10 @@ class TestModels(TransactionTestCase):
 | 
				
			|||||||
            {
 | 
					            {
 | 
				
			||||||
                "hostname": self.endpoint.host.split(":")[0],
 | 
					                "hostname": self.endpoint.host.split(":")[0],
 | 
				
			||||||
                "port": "1324",
 | 
					                "port": "1324",
 | 
				
			||||||
                "client-name": f"authentik - {self.user}",
 | 
					                "client-name": "authentik",
 | 
				
			||||||
                "drive-path": path,
 | 
					                "drive-path": path,
 | 
				
			||||||
                "create-drive-path": "true",
 | 
					                "create-drive-path": "true",
 | 
				
			||||||
                "level": "endpoint",
 | 
					                "level": "endpoint",
 | 
				
			||||||
                "resize-method": "display-update",
 | 
					 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        # Set settings in token
 | 
					        # Set settings in token
 | 
				
			||||||
@ -98,11 +95,10 @@ class TestModels(TransactionTestCase):
 | 
				
			|||||||
            {
 | 
					            {
 | 
				
			||||||
                "hostname": self.endpoint.host.split(":")[0],
 | 
					                "hostname": self.endpoint.host.split(":")[0],
 | 
				
			||||||
                "port": "1324",
 | 
					                "port": "1324",
 | 
				
			||||||
                "client-name": f"authentik - {self.user}",
 | 
					                "client-name": "authentik",
 | 
				
			||||||
                "drive-path": path,
 | 
					                "drive-path": path,
 | 
				
			||||||
                "create-drive-path": "true",
 | 
					                "create-drive-path": "true",
 | 
				
			||||||
                "level": "token",
 | 
					                "level": "token",
 | 
				
			||||||
                "resize-method": "display-update",
 | 
					 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        # Set settings in property mapping (provider)
 | 
					        # Set settings in property mapping (provider)
 | 
				
			||||||
@ -118,11 +114,10 @@ class TestModels(TransactionTestCase):
 | 
				
			|||||||
            {
 | 
					            {
 | 
				
			||||||
                "hostname": self.endpoint.host.split(":")[0],
 | 
					                "hostname": self.endpoint.host.split(":")[0],
 | 
				
			||||||
                "port": "1324",
 | 
					                "port": "1324",
 | 
				
			||||||
                "client-name": f"authentik - {self.user}",
 | 
					                "client-name": "authentik",
 | 
				
			||||||
                "drive-path": path,
 | 
					                "drive-path": path,
 | 
				
			||||||
                "create-drive-path": "true",
 | 
					                "create-drive-path": "true",
 | 
				
			||||||
                "level": "property_mapping_provider",
 | 
					                "level": "property_mapping_provider",
 | 
				
			||||||
                "resize-method": "display-update",
 | 
					 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        # Set settings in property mapping (endpoint)
 | 
					        # Set settings in property mapping (endpoint)
 | 
				
			||||||
@ -140,12 +135,11 @@ class TestModels(TransactionTestCase):
 | 
				
			|||||||
            {
 | 
					            {
 | 
				
			||||||
                "hostname": self.endpoint.host.split(":")[0],
 | 
					                "hostname": self.endpoint.host.split(":")[0],
 | 
				
			||||||
                "port": "1324",
 | 
					                "port": "1324",
 | 
				
			||||||
                "client-name": f"authentik - {self.user}",
 | 
					                "client-name": "authentik",
 | 
				
			||||||
                "drive-path": path,
 | 
					                "drive-path": path,
 | 
				
			||||||
                "create-drive-path": "true",
 | 
					                "create-drive-path": "true",
 | 
				
			||||||
                "level": "property_mapping_endpoint",
 | 
					                "level": "property_mapping_endpoint",
 | 
				
			||||||
                "foo": "true",
 | 
					                "foo": "true",
 | 
				
			||||||
                "bar": "6",
 | 
					                "bar": "6",
 | 
				
			||||||
                "resize-method": "display-update",
 | 
					 | 
				
			||||||
            },
 | 
					            },
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
 | 
				
			|||||||
@ -18,7 +18,9 @@ from authentik.flows.exceptions import FlowNonApplicableException
 | 
				
			|||||||
from authentik.flows.models import in_memory_stage
 | 
					from authentik.flows.models import in_memory_stage
 | 
				
			||||||
from authentik.flows.planner import PLAN_CONTEXT_APPLICATION, FlowPlanner
 | 
					from authentik.flows.planner import PLAN_CONTEXT_APPLICATION, FlowPlanner
 | 
				
			||||||
from authentik.flows.stage import RedirectStage
 | 
					from authentik.flows.stage import RedirectStage
 | 
				
			||||||
 | 
					from authentik.flows.views.executor import SESSION_KEY_PLAN
 | 
				
			||||||
from authentik.lib.utils.time import timedelta_from_string
 | 
					from authentik.lib.utils.time import timedelta_from_string
 | 
				
			||||||
 | 
					from authentik.lib.utils.urls import redirect_with_qs
 | 
				
			||||||
from authentik.policies.engine import PolicyEngine
 | 
					from authentik.policies.engine import PolicyEngine
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -54,7 +56,12 @@ class RACStartView(EnterprisePolicyAccessView):
 | 
				
			|||||||
                provider=self.provider,
 | 
					                provider=self.provider,
 | 
				
			||||||
            )
 | 
					            )
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        return plan.to_redirect(request, self.provider.authorization_flow)
 | 
					        request.session[SESSION_KEY_PLAN] = plan
 | 
				
			||||||
 | 
					        return redirect_with_qs(
 | 
				
			||||||
 | 
					            "authentik_core:if-flow",
 | 
				
			||||||
 | 
					            request.GET,
 | 
				
			||||||
 | 
					            flow_slug=self.provider.authorization_flow.slug,
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class RACInterface(InterfaceView):
 | 
					class RACInterface(InterfaceView):
 | 
				
			||||||
 | 
				
			|||||||
@ -1,11 +1,14 @@
 | 
				
			|||||||
"""AuthenticatorEndpointGDTCStage API Views"""
 | 
					"""AuthenticatorEndpointGDTCStage API Views"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from django_filters.rest_framework.backends import DjangoFilterBackend
 | 
				
			||||||
from rest_framework import mixins
 | 
					from rest_framework import mixins
 | 
				
			||||||
 | 
					from rest_framework.filters import OrderingFilter, SearchFilter
 | 
				
			||||||
from rest_framework.permissions import IsAdminUser
 | 
					from rest_framework.permissions import IsAdminUser
 | 
				
			||||||
from rest_framework.serializers import ModelSerializer
 | 
					from rest_framework.serializers import ModelSerializer
 | 
				
			||||||
from rest_framework.viewsets import GenericViewSet, ModelViewSet
 | 
					from rest_framework.viewsets import GenericViewSet, ModelViewSet
 | 
				
			||||||
from structlog.stdlib import get_logger
 | 
					from structlog.stdlib import get_logger
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import OwnerFilter, OwnerPermissions
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
from authentik.enterprise.api import EnterpriseRequiredMixin
 | 
					from authentik.enterprise.api import EnterpriseRequiredMixin
 | 
				
			||||||
from authentik.enterprise.stages.authenticator_endpoint_gdtc.models import (
 | 
					from authentik.enterprise.stages.authenticator_endpoint_gdtc.models import (
 | 
				
			||||||
@ -64,7 +67,8 @@ class EndpointDeviceViewSet(
 | 
				
			|||||||
    search_fields = ["name"]
 | 
					    search_fields = ["name"]
 | 
				
			||||||
    filterset_fields = ["name"]
 | 
					    filterset_fields = ["name"]
 | 
				
			||||||
    ordering = ["name"]
 | 
					    ordering = ["name"]
 | 
				
			||||||
    owner_field = "user"
 | 
					    permission_classes = [OwnerPermissions]
 | 
				
			||||||
 | 
					    filter_backends = [OwnerFilter, DjangoFilterBackend, OrderingFilter, SearchFilter]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class EndpointAdminDeviceViewSet(ModelViewSet):
 | 
					class EndpointAdminDeviceViewSet(ModelViewSet):
 | 
				
			||||||
 | 
				
			|||||||
@ -4,9 +4,7 @@ from typing import Any
 | 
				
			|||||||
from django.http import HttpRequest, HttpResponse, HttpResponseRedirect
 | 
					from django.http import HttpRequest, HttpResponse, HttpResponseRedirect
 | 
				
			||||||
from django.template.response import TemplateResponse
 | 
					from django.template.response import TemplateResponse
 | 
				
			||||||
from django.urls import reverse
 | 
					from django.urls import reverse
 | 
				
			||||||
from django.utils.decorators import method_decorator
 | 
					 | 
				
			||||||
from django.views import View
 | 
					from django.views import View
 | 
				
			||||||
from django.views.decorators.clickjacking import xframe_options_sameorigin
 | 
					 | 
				
			||||||
from googleapiclient.discovery import build
 | 
					from googleapiclient.discovery import build
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from authentik.enterprise.stages.authenticator_endpoint_gdtc.models import (
 | 
					from authentik.enterprise.stages.authenticator_endpoint_gdtc.models import (
 | 
				
			||||||
@ -28,7 +26,6 @@ HEADER_ACCESS_CHALLENGE_RESPONSE = "X-Verified-Access-Challenge-Response"
 | 
				
			|||||||
DEVICE_TRUST_VERIFIED_ACCESS = "VerifiedAccess"
 | 
					DEVICE_TRUST_VERIFIED_ACCESS = "VerifiedAccess"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@method_decorator(xframe_options_sameorigin, name="dispatch")
 | 
					 | 
				
			||||||
class GoogleChromeDeviceTrustConnector(View):
 | 
					class GoogleChromeDeviceTrustConnector(View):
 | 
				
			||||||
    """Google Chrome Device-trust connector based endpoint authenticator"""
 | 
					    """Google Chrome Device-trust connector based endpoint authenticator"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -215,49 +215,3 @@ class TestReadOnly(FlowTestCase):
 | 
				
			|||||||
            {"detail": "Request denied due to expired/invalid license.", "code": "denied_license"},
 | 
					            {"detail": "Request denied due to expired/invalid license.", "code": "denied_license"},
 | 
				
			||||||
        )
 | 
					        )
 | 
				
			||||||
        self.assertEqual(response.status_code, 400)
 | 
					        self.assertEqual(response.status_code, 400)
 | 
				
			||||||
 | 
					 | 
				
			||||||
    @patch(
 | 
					 | 
				
			||||||
        "authentik.enterprise.license.LicenseKey.validate",
 | 
					 | 
				
			||||||
        MagicMock(
 | 
					 | 
				
			||||||
            return_value=LicenseKey(
 | 
					 | 
				
			||||||
                aud="",
 | 
					 | 
				
			||||||
                exp=expiry_valid,
 | 
					 | 
				
			||||||
                name=generate_id(),
 | 
					 | 
				
			||||||
                internal_users=100,
 | 
					 | 
				
			||||||
                external_users=100,
 | 
					 | 
				
			||||||
            )
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    @patch(
 | 
					 | 
				
			||||||
        "authentik.enterprise.license.LicenseKey.get_internal_user_count",
 | 
					 | 
				
			||||||
        MagicMock(return_value=1000),
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    @patch(
 | 
					 | 
				
			||||||
        "authentik.enterprise.license.LicenseKey.get_external_user_count",
 | 
					 | 
				
			||||||
        MagicMock(return_value=1000),
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    @patch(
 | 
					 | 
				
			||||||
        "authentik.enterprise.license.LicenseKey.record_usage",
 | 
					 | 
				
			||||||
        MagicMock(),
 | 
					 | 
				
			||||||
    )
 | 
					 | 
				
			||||||
    def test_manage_users(self):
 | 
					 | 
				
			||||||
        """Test that managing users is still possible"""
 | 
					 | 
				
			||||||
        License.objects.create(key=generate_id())
 | 
					 | 
				
			||||||
        usage = LicenseUsage.objects.create(
 | 
					 | 
				
			||||||
            internal_user_count=100,
 | 
					 | 
				
			||||||
            external_user_count=100,
 | 
					 | 
				
			||||||
            status=LicenseUsageStatus.VALID,
 | 
					 | 
				
			||||||
        )
 | 
					 | 
				
			||||||
        usage.record_date = now() - timedelta(weeks=THRESHOLD_READ_ONLY_WEEKS + 1)
 | 
					 | 
				
			||||||
        usage.save(update_fields=["record_date"])
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        admin = create_test_admin_user()
 | 
					 | 
				
			||||||
        self.client.force_login(admin)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        # Reading is always allowed
 | 
					 | 
				
			||||||
        response = self.client.get(reverse("authentik_api:user-list"))
 | 
					 | 
				
			||||||
        self.assertEqual(response.status_code, 200)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
        # Writing should also be allowed
 | 
					 | 
				
			||||||
        response = self.client.patch(reverse("authentik_api:user-detail", kwargs={"pk": admin.pk}))
 | 
					 | 
				
			||||||
        self.assertEqual(response.status_code, 200)
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -1,15 +1,17 @@
 | 
				
			|||||||
"""Notification API Views"""
 | 
					"""Notification API Views"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from django_filters.rest_framework import DjangoFilterBackend
 | 
				
			||||||
from drf_spectacular.types import OpenApiTypes
 | 
					from drf_spectacular.types import OpenApiTypes
 | 
				
			||||||
from drf_spectacular.utils import OpenApiResponse, extend_schema
 | 
					from drf_spectacular.utils import OpenApiResponse, extend_schema
 | 
				
			||||||
from rest_framework import mixins
 | 
					from rest_framework import mixins
 | 
				
			||||||
from rest_framework.decorators import action
 | 
					from rest_framework.decorators import action
 | 
				
			||||||
from rest_framework.fields import ReadOnlyField
 | 
					from rest_framework.fields import ReadOnlyField
 | 
				
			||||||
from rest_framework.permissions import IsAuthenticated
 | 
					from rest_framework.filters import OrderingFilter, SearchFilter
 | 
				
			||||||
from rest_framework.request import Request
 | 
					from rest_framework.request import Request
 | 
				
			||||||
from rest_framework.response import Response
 | 
					from rest_framework.response import Response
 | 
				
			||||||
from rest_framework.viewsets import GenericViewSet
 | 
					from rest_framework.viewsets import GenericViewSet
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					from authentik.api.authorization import OwnerFilter, OwnerPermissions
 | 
				
			||||||
from authentik.core.api.used_by import UsedByMixin
 | 
					from authentik.core.api.used_by import UsedByMixin
 | 
				
			||||||
from authentik.core.api.utils import ModelSerializer
 | 
					from authentik.core.api.utils import ModelSerializer
 | 
				
			||||||
from authentik.events.api.events import EventSerializer
 | 
					from authentik.events.api.events import EventSerializer
 | 
				
			||||||
@ -55,7 +57,8 @@ class NotificationViewSet(
 | 
				
			|||||||
        "seen",
 | 
					        "seen",
 | 
				
			||||||
        "user",
 | 
					        "user",
 | 
				
			||||||
    ]
 | 
					    ]
 | 
				
			||||||
    owner_field = "user"
 | 
					    permission_classes = [OwnerPermissions]
 | 
				
			||||||
 | 
					    filter_backends = [OwnerFilter, DjangoFilterBackend, OrderingFilter, SearchFilter]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    @extend_schema(
 | 
					    @extend_schema(
 | 
				
			||||||
        request=OpenApiTypes.NONE,
 | 
					        request=OpenApiTypes.NONE,
 | 
				
			||||||
@ -63,7 +66,7 @@ class NotificationViewSet(
 | 
				
			|||||||
            204: OpenApiResponse(description="Marked tasks as read successfully."),
 | 
					            204: OpenApiResponse(description="Marked tasks as read successfully."),
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
    )
 | 
					    )
 | 
				
			||||||
    @action(detail=False, methods=["post"], permission_classes=[IsAuthenticated])
 | 
					    @action(detail=False, methods=["post"])
 | 
				
			||||||
    def mark_all_seen(self, request: Request) -> Response:
 | 
					    def mark_all_seen(self, request: Request) -> Response:
 | 
				
			||||||
        """Mark all the user's notifications as seen"""
 | 
					        """Mark all the user's notifications as seen"""
 | 
				
			||||||
        Notification.objects.filter(user=request.user, seen=False).update(seen=True)
 | 
					        Notification.objects.filter(user=request.user, seen=False).update(seen=True)
 | 
				
			||||||
 | 
				
			|||||||
@ -1,41 +0,0 @@
 | 
				
			|||||||
# Generated by Django 5.0.10 on 2025-01-13 18:05
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.db import migrations, models
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Migration(migrations.Migration):
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    dependencies = [
 | 
					 | 
				
			||||||
        ("authentik_events", "0007_event_authentik_e_action_9a9dd9_idx_and_more"),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    operations = [
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="event",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expires"], name="authentik_e_expires_8c73a8_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="event",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expiring"], name="authentik_e_expirin_b5cb5e_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="event",
 | 
					 | 
				
			||||||
            index=models.Index(
 | 
					 | 
				
			||||||
                fields=["expiring", "expires"], name="authentik_e_expirin_e37180_idx"
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="systemtask",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expires"], name="authentik_e_expires_4d3985_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="systemtask",
 | 
					 | 
				
			||||||
            index=models.Index(fields=["expiring"], name="authentik_e_expirin_81d649_idx"),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        migrations.AddIndex(
 | 
					 | 
				
			||||||
            model_name="systemtask",
 | 
					 | 
				
			||||||
            index=models.Index(
 | 
					 | 
				
			||||||
                fields=["expiring", "expires"], name="authentik_e_expirin_eb3598_idx"
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
    ]
 | 
					 | 
				
			||||||
@ -60,7 +60,7 @@ def default_event_duration():
 | 
				
			|||||||
    """Default duration an Event is saved.
 | 
					    """Default duration an Event is saved.
 | 
				
			||||||
    This is used as a fallback when no brand is available"""
 | 
					    This is used as a fallback when no brand is available"""
 | 
				
			||||||
    try:
 | 
					    try:
 | 
				
			||||||
        tenant = get_current_tenant(only=["event_retention"])
 | 
					        tenant = get_current_tenant()
 | 
				
			||||||
        return now() + timedelta_from_string(tenant.event_retention)
 | 
					        return now() + timedelta_from_string(tenant.event_retention)
 | 
				
			||||||
    except Tenant.DoesNotExist:
 | 
					    except Tenant.DoesNotExist:
 | 
				
			||||||
        return now() + timedelta(days=365)
 | 
					        return now() + timedelta(days=365)
 | 
				
			||||||
@ -306,7 +306,7 @@ class Event(SerializerModel, ExpiringModel):
 | 
				
			|||||||
    class Meta:
 | 
					    class Meta:
 | 
				
			||||||
        verbose_name = _("Event")
 | 
					        verbose_name = _("Event")
 | 
				
			||||||
        verbose_name_plural = _("Events")
 | 
					        verbose_name_plural = _("Events")
 | 
				
			||||||
        indexes = ExpiringModel.Meta.indexes + [
 | 
					        indexes = [
 | 
				
			||||||
            models.Index(fields=["action"]),
 | 
					            models.Index(fields=["action"]),
 | 
				
			||||||
            models.Index(fields=["user"]),
 | 
					            models.Index(fields=["user"]),
 | 
				
			||||||
            models.Index(fields=["app"]),
 | 
					            models.Index(fields=["app"]),
 | 
				
			||||||
@ -694,4 +694,3 @@ class SystemTask(SerializerModel, ExpiringModel):
 | 
				
			|||||||
        permissions = [("run_task", _("Run task"))]
 | 
					        permissions = [("run_task", _("Run task"))]
 | 
				
			||||||
        verbose_name = _("System Task")
 | 
					        verbose_name = _("System Task")
 | 
				
			||||||
        verbose_name_plural = _("System Tasks")
 | 
					        verbose_name_plural = _("System Tasks")
 | 
				
			||||||
        indexes = ExpiringModel.Meta.indexes
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -106,9 +106,9 @@ def on_invitation_used(sender, request: HttpRequest, invitation: Invitation, **_
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@receiver(password_changed)
 | 
					@receiver(password_changed)
 | 
				
			||||||
def on_password_changed(sender, user: User, password: str, request: HttpRequest | None, **_):
 | 
					def on_password_changed(sender, user: User, password: str, **_):
 | 
				
			||||||
    """Log password change"""
 | 
					    """Log password change"""
 | 
				
			||||||
    Event.new(EventAction.PASSWORD_SET).from_http(request, user=user)
 | 
					    Event.new(EventAction.PASSWORD_SET).from_http(None, user=user)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@receiver(post_save, sender=Event)
 | 
					@receiver(post_save, sender=Event)
 | 
				
			||||||
 | 
				
			|||||||
@ -138,6 +138,7 @@ def notification_cleanup(self: SystemTask):
 | 
				
			|||||||
    """Cleanup seen notifications and notifications whose event expired."""
 | 
					    """Cleanup seen notifications and notifications whose event expired."""
 | 
				
			||||||
    notifications = Notification.objects.filter(Q(event=None) | Q(seen=True))
 | 
					    notifications = Notification.objects.filter(Q(event=None) | Q(seen=True))
 | 
				
			||||||
    amount = notifications.count()
 | 
					    amount = notifications.count()
 | 
				
			||||||
    notifications.delete()
 | 
					    for notification in notifications:
 | 
				
			||||||
 | 
					        notification.delete()
 | 
				
			||||||
    LOGGER.debug("Expired notifications", amount=amount)
 | 
					    LOGGER.debug("Expired notifications", amount=amount)
 | 
				
			||||||
    self.set_status(TaskStatus.SUCCESSFUL, f"Expired {amount} Notifications")
 | 
					    self.set_status(TaskStatus.SUCCESSFUL, f"Expired {amount} Notifications")
 | 
				
			||||||
 | 
				
			|||||||
@ -1,7 +1,5 @@
 | 
				
			|||||||
"""Flow Stage API Views"""
 | 
					"""Flow Stage API Views"""
 | 
				
			||||||
 | 
					
 | 
				
			||||||
from uuid import uuid4
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
from django.urls.base import reverse
 | 
					from django.urls.base import reverse
 | 
				
			||||||
from drf_spectacular.utils import extend_schema
 | 
					from drf_spectacular.utils import extend_schema
 | 
				
			||||||
from rest_framework import mixins
 | 
					from rest_framework import mixins
 | 
				
			||||||
@ -29,11 +27,6 @@ class StageSerializer(ModelSerializer, MetaNameSerializer):
 | 
				
			|||||||
    component = SerializerMethodField()
 | 
					    component = SerializerMethodField()
 | 
				
			||||||
    flow_set = FlowSetSerializer(many=True, required=False)
 | 
					    flow_set = FlowSetSerializer(many=True, required=False)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    def to_representation(self, instance: Stage):
 | 
					 | 
				
			||||||
        if isinstance(instance, Stage) and instance.is_in_memory:
 | 
					 | 
				
			||||||
            instance.stage_uuid = uuid4()
 | 
					 | 
				
			||||||
        return super().to_representation(instance)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    def get_component(self, obj: Stage) -> str:
 | 
					    def get_component(self, obj: Stage) -> str:
 | 
				
			||||||
        """Get object type so that we know how to edit the object"""
 | 
					        """Get object type so that we know how to edit the object"""
 | 
				
			||||||
        if obj.__class__ == Stage:
 | 
					        if obj.__class__ == Stage:
 | 
				
			||||||
 | 
				
			|||||||
@ -97,9 +97,12 @@ class FlowErrorChallenge(Challenge):
 | 
				
			|||||||
        if not request or not error:
 | 
					        if not request or not error:
 | 
				
			||||||
            return
 | 
					            return
 | 
				
			||||||
        self.initial_data["request_id"] = request.request_id
 | 
					        self.initial_data["request_id"] = request.request_id
 | 
				
			||||||
 | 
					        from authentik.core.models import USER_ATTRIBUTE_DEBUG
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if request.user and request.user.is_authenticated:
 | 
					        if request.user and request.user.is_authenticated:
 | 
				
			||||||
            if request.user.has_perm("authentik_core.user_view_debug"):
 | 
					            if request.user.is_superuser or request.user.group_attributes(request).get(
 | 
				
			||||||
 | 
					                USER_ATTRIBUTE_DEBUG, False
 | 
				
			||||||
 | 
					            ):
 | 
				
			||||||
                self.initial_data["error"] = str(error)
 | 
					                self.initial_data["error"] = str(error)
 | 
				
			||||||
                self.initial_data["traceback"] = exception_to_string(error)
 | 
					                self.initial_data["traceback"] = exception_to_string(error)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user