Compare commits
	
		
			5 Commits
		
	
	
		
			version-te
			...
			blog-ent
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 8888f80642 | |||
| 0b7ca0abc9 | |||
| efaa61a8ff | |||
| 90b149cf0a | |||
| 818a03b3b1 | 
| @ -1,20 +1,12 @@ | |||||||
| [bumpversion] | [bumpversion] | ||||||
| current_version = 2024.2.1 | current_version = 2023.8.1 | ||||||
| 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+) | ||||||
| serialize =  | serialize = {major}.{minor}.{patch} | ||||||
| 	{major}.{minor}.{patch}-{rc_t}{rc_n} |  | ||||||
| 	{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] |  | ||||||
| values =  |  | ||||||
| 	rc |  | ||||||
| 	final |  | ||||||
| optional_value = final |  | ||||||
|  |  | ||||||
| [bumpversion:file:pyproject.toml] | [bumpversion:file:pyproject.toml] | ||||||
|  |  | ||||||
| [bumpversion:file:docker-compose.yml] | [bumpversion:file:docker-compose.yml] | ||||||
|  | |||||||
| @ -1,12 +1,10 @@ | |||||||
|  | env | ||||||
| htmlcov | htmlcov | ||||||
| *.env.yml | *.env.yml | ||||||
| **/node_modules | **/node_modules | ||||||
| dist/** | dist/** | ||||||
| build/** | build/** | ||||||
| build_docs/** | build_docs/** | ||||||
| *Dockerfile | Dockerfile | ||||||
|  | authentik/enterprise | ||||||
| blueprints/local | blueprints/local | ||||||
| .git |  | ||||||
| !gen-ts-api/node_modules |  | ||||||
| !gen-ts-api/dist/** |  | ||||||
| !gen-go-api/ |  | ||||||
|  | |||||||
| @ -9,6 +9,9 @@ inputs: | |||||||
| runs: | runs: | ||||||
|   using: "composite" |   using: "composite" | ||||||
|   steps: |   steps: | ||||||
|  |     - name: Generate config | ||||||
|  |       id: ev | ||||||
|  |       uses: ./.github/actions/docker-push-variables | ||||||
|     - name: Find Comment |     - name: Find Comment | ||||||
|       uses: peter-evans/find-comment@v2 |       uses: peter-evans/find-comment@v2 | ||||||
|       id: fc |       id: fc | ||||||
|  | |||||||
							
								
								
									
										75
									
								
								.github/actions/docker-push-variables/action.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										75
									
								
								.github/actions/docker-push-variables/action.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,43 +1,64 @@ | |||||||
| --- |  | ||||||
| name: "Prepare docker environment variables" | name: "Prepare docker environment variables" | ||||||
| description: "Prepare docker environment variables" | description: "Prepare docker environment variables" | ||||||
|  |  | ||||||
| inputs: |  | ||||||
|   image-name: |  | ||||||
|     required: true |  | ||||||
|     description: "Docker image prefix" |  | ||||||
|   image-arch: |  | ||||||
|     required: false |  | ||||||
|     description: "Docker image arch" |  | ||||||
|  |  | ||||||
| outputs: | outputs: | ||||||
|  |   shouldBuild: | ||||||
|  |     description: "Whether to build image or not" | ||||||
|  |     value: ${{ steps.ev.outputs.shouldBuild }} | ||||||
|  |   branchName: | ||||||
|  |     description: "Branch name" | ||||||
|  |     value: ${{ steps.ev.outputs.branchName }} | ||||||
|  |   branchNameContainer: | ||||||
|  |     description: "Branch name (for containers)" | ||||||
|  |     value: ${{ steps.ev.outputs.branchNameContainer }} | ||||||
|  |   timestamp: | ||||||
|  |     description: "Timestamp" | ||||||
|  |     value: ${{ steps.ev.outputs.timestamp }} | ||||||
|   sha: |   sha: | ||||||
|     description: "sha" |     description: "sha" | ||||||
|     value: ${{ steps.ev.outputs.sha }} |     value: ${{ steps.ev.outputs.sha }} | ||||||
|  |   shortHash: | ||||||
|  |     description: "shortHash" | ||||||
|  |     value: ${{ steps.ev.outputs.shortHash }} | ||||||
|   version: |   version: | ||||||
|     description: "Version" |     description: "version" | ||||||
|     value: ${{ steps.ev.outputs.version }} |     value: ${{ steps.ev.outputs.version }} | ||||||
|   prerelease: |   versionFamily: | ||||||
|     description: "Prerelease" |     description: "versionFamily" | ||||||
|     value: ${{ steps.ev.outputs.prerelease }} |     value: ${{ steps.ev.outputs.versionFamily }} | ||||||
|  |  | ||||||
|   imageTags: |  | ||||||
|     description: "Docker image tags" |  | ||||||
|     value: ${{ steps.ev.outputs.imageTags }} |  | ||||||
|   imageMainTag: |  | ||||||
|     description: "Docker image main tag" |  | ||||||
|     value: ${{ steps.ev.outputs.imageMainTag }} |  | ||||||
|  |  | ||||||
| runs: | runs: | ||||||
|   using: "composite" |   using: "composite" | ||||||
|   steps: |   steps: | ||||||
|     - name: Generate config |     - name: Generate config | ||||||
|       id: ev |       id: ev | ||||||
|       shell: bash |       shell: python | ||||||
|       env: |  | ||||||
|         IMAGE_NAME: ${{ inputs.image-name }} |  | ||||||
|         IMAGE_ARCH: ${{ inputs.image-arch }} |  | ||||||
|         PR_HEAD_SHA: ${{ github.event.pull_request.head.sha }} |  | ||||||
|       run: | |       run: | | ||||||
|         python3 ${{ github.action_path }}/push_vars.py |         """Helper script to get the actual branch name, docker safe""" | ||||||
|  |         import configparser | ||||||
|  |         import os | ||||||
|  |         from time import time | ||||||
|  |  | ||||||
|  |         parser = configparser.ConfigParser() | ||||||
|  |         parser.read(".bumpversion.cfg") | ||||||
|  |  | ||||||
|  |         branch_name = os.environ["GITHUB_REF"] | ||||||
|  |         if os.environ.get("GITHUB_HEAD_REF", "") != "": | ||||||
|  |             branch_name = os.environ["GITHUB_HEAD_REF"] | ||||||
|  |  | ||||||
|  |         should_build = str(os.environ.get("DOCKER_USERNAME", "") != "").lower() | ||||||
|  |         version = parser.get("bumpversion", "current_version") | ||||||
|  |         version_family = ".".join(version.split(".")[:-1]) | ||||||
|  |         safe_branch_name = branch_name.replace("refs/heads/", "").replace("/", "-") | ||||||
|  |  | ||||||
|  |         sha = os.environ["GITHUB_SHA"] if not "${{ github.event.pull_request.head.sha }}" else "${{ github.event.pull_request.head.sha }}" | ||||||
|  |  | ||||||
|  |         with open(os.environ["GITHUB_OUTPUT"], "a+", encoding="utf-8") as _output: | ||||||
|  |             print("branchName=%s" % branch_name, file=_output) | ||||||
|  |             print("branchNameContainer=%s" % safe_branch_name, file=_output) | ||||||
|  |             print("timestamp=%s" % int(time()), file=_output) | ||||||
|  |             print("sha=%s" % sha, file=_output) | ||||||
|  |             print("shortHash=%s" % sha[:7], file=_output) | ||||||
|  |             print("shouldBuild=%s" % should_build, file=_output) | ||||||
|  |             print("version=%s" % version, file=_output) | ||||||
|  |             print("versionFamily=%s" % version_family, file=_output) | ||||||
|  | |||||||
| @ -1,59 +0,0 @@ | |||||||
| """Helper script to get the actual branch name, docker safe""" |  | ||||||
|  |  | ||||||
| import configparser |  | ||||||
| import os |  | ||||||
| from time import time |  | ||||||
|  |  | ||||||
| parser = configparser.ConfigParser() |  | ||||||
| parser.read(".bumpversion.cfg") |  | ||||||
|  |  | ||||||
| branch_name = os.environ["GITHUB_REF"] |  | ||||||
| if os.environ.get("GITHUB_HEAD_REF", "") != "": |  | ||||||
|     branch_name = os.environ["GITHUB_HEAD_REF"] |  | ||||||
| safe_branch_name = branch_name.replace("refs/heads/", "").replace("/", "-") |  | ||||||
|  |  | ||||||
| image_names = os.getenv("IMAGE_NAME").split(",") |  | ||||||
| image_arch = os.getenv("IMAGE_ARCH") or None |  | ||||||
|  |  | ||||||
| is_pull_request = bool(os.getenv("PR_HEAD_SHA")) |  | ||||||
| is_release = "dev" not in image_names[0] |  | ||||||
|  |  | ||||||
| sha = os.environ["GITHUB_SHA"] if not is_pull_request else os.getenv("PR_HEAD_SHA") |  | ||||||
|  |  | ||||||
| # 2042.1.0 or 2042.1.0-rc1 |  | ||||||
| version = parser.get("bumpversion", "current_version") |  | ||||||
| # 2042.1 |  | ||||||
| version_family = ".".join(version.split("-", 1)[0].split(".")[:-1]) |  | ||||||
| prerelease = "-" in version |  | ||||||
|  |  | ||||||
| image_tags = [] |  | ||||||
| if is_release: |  | ||||||
|     for name in image_names: |  | ||||||
|         image_tags += [ |  | ||||||
|             f"{name}:{version}", |  | ||||||
|         ] |  | ||||||
|         if not prerelease: |  | ||||||
|             image_tags += [ |  | ||||||
|                 f"{name}:latest", |  | ||||||
|                 f"{name}:{version_family}", |  | ||||||
|             ] |  | ||||||
| else: |  | ||||||
|     suffix = "" |  | ||||||
|     if image_arch and image_arch != "amd64": |  | ||||||
|         suffix = f"-{image_arch}" |  | ||||||
|     for name in image_names: |  | ||||||
|         image_tags += [ |  | ||||||
|             f"{name}:gh-{sha}{suffix}",  # Used for ArgoCD and PR comments |  | ||||||
|             f"{name}:gh-{safe_branch_name}{suffix}",  # For convenience |  | ||||||
|             f"{name}:gh-{safe_branch_name}-{int(time())}-{sha[:7]}{suffix}",  # Use by FluxCD |  | ||||||
|         ] |  | ||||||
|  |  | ||||||
| image_main_tag = image_tags[0] |  | ||||||
| image_tags_rendered = ",".join(image_tags) |  | ||||||
|  |  | ||||||
| with open(os.environ["GITHUB_OUTPUT"], "a+", encoding="utf-8") as _output: |  | ||||||
|     print("sha=%s" % sha, file=_output) |  | ||||||
|     print("version=%s" % version, file=_output) |  | ||||||
|     print("prerelease=%s" % prerelease, file=_output) |  | ||||||
|     print("imageTags=%s" % image_tags_rendered, file=_output) |  | ||||||
|     print("imageMainTag=%s" % image_main_tag, file=_output) |  | ||||||
| @ -1,7 +0,0 @@ | |||||||
| #!/bin/bash -x |  | ||||||
| SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) |  | ||||||
| GITHUB_OUTPUT=/dev/stdout \ |  | ||||||
|     GITHUB_REF=ref \ |  | ||||||
|     GITHUB_SHA=sha \ |  | ||||||
|     IMAGE_NAME=ghcr.io/goauthentik/server,beryju/authentik \ |  | ||||||
|     python $SCRIPT_DIR/push_vars.py |  | ||||||
							
								
								
									
										23
									
								
								.github/actions/setup/action.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										23
									
								
								.github/actions/setup/action.yml
									
									
									
									
										vendored
									
									
								
							| @ -2,39 +2,36 @@ name: "Setup authentik testing environment" | |||||||
| description: "Setup authentik testing environment" | description: "Setup authentik testing environment" | ||||||
|  |  | ||||||
| inputs: | inputs: | ||||||
|   postgresql_version: |   postgresql_tag: | ||||||
|     description: "Optional postgresql image tag" |     description: "Optional postgresql image tag" | ||||||
|     default: "16" |     default: "12" | ||||||
|  |  | ||||||
| runs: | runs: | ||||||
|   using: "composite" |   using: "composite" | ||||||
|   steps: |   steps: | ||||||
|     - name: Install poetry & deps |     - name: Install poetry | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: | |       run: | | ||||||
|         pipx install poetry || true |         pipx install poetry || true | ||||||
|         sudo apt-get update |         sudo apt update | ||||||
|         sudo apt-get install --no-install-recommends -y libpq-dev openssl libxmlsec1-dev pkg-config gettext |         sudo apt install -y libpq-dev openssl libxmlsec1-dev pkg-config gettext | ||||||
|     - name: Setup python and restore poetry |     - name: Setup python and restore poetry | ||||||
|       uses: actions/setup-python@v4 |       uses: actions/setup-python@v3 | ||||||
|       with: |       with: | ||||||
|         python-version-file: "pyproject.toml" |         python-version: "3.11" | ||||||
|         cache: "poetry" |         cache: "poetry" | ||||||
|     - name: Setup node |     - name: Setup node | ||||||
|       uses: actions/setup-node@v3 |       uses: actions/setup-node@v3 | ||||||
|       with: |       with: | ||||||
|         node-version-file: web/package.json |         node-version: "20" | ||||||
|         cache: "npm" |         cache: "npm" | ||||||
|         cache-dependency-path: web/package-lock.json |         cache-dependency-path: web/package-lock.json | ||||||
|     - name: Setup go |  | ||||||
|       uses: actions/setup-go@v4 |  | ||||||
|       with: |  | ||||||
|         go-version-file: "go.mod" |  | ||||||
|     - name: Setup dependencies |     - name: Setup dependencies | ||||||
|       shell: bash |       shell: bash | ||||||
|       run: | |       run: | | ||||||
|         export PSQL_TAG=${{ inputs.postgresql_version }} |         export PSQL_TAG=${{ inputs.postgresql_tag }} | ||||||
|         docker-compose -f .github/actions/setup/docker-compose.yml up -d |         docker-compose -f .github/actions/setup/docker-compose.yml up -d | ||||||
|  |         poetry env use python3.11 | ||||||
|         poetry install |         poetry install | ||||||
|         cd web && npm ci |         cd web && npm ci | ||||||
|     - name: Generate config |     - name: Generate config | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.github/actions/setup/docker-compose.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/actions/setup/docker-compose.yml
									
									
									
									
										vendored
									
									
								
							| @ -2,7 +2,7 @@ version: "3.7" | |||||||
|  |  | ||||||
| services: | services: | ||||||
|   postgresql: |   postgresql: | ||||||
|     image: docker.io/library/postgres:${PSQL_TAG:-16} |     image: docker.io/library/postgres:${PSQL_TAG:-12} | ||||||
|     volumes: |     volumes: | ||||||
|       - db-data:/var/lib/postgresql/data |       - db-data:/var/lib/postgresql/data | ||||||
|     environment: |     environment: | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.github/codecov.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/codecov.yml
									
									
									
									
										vendored
									
									
								
							| @ -6,5 +6,5 @@ coverage: | |||||||
|         # adjust accordingly based on how flaky your tests are |         # adjust accordingly based on how flaky your tests are | ||||||
|         # this allows a 1% drop from the previous base commit coverage |         # this allows a 1% drop from the previous base commit coverage | ||||||
|         threshold: 1% |         threshold: 1% | ||||||
| comment: |   notify: | ||||||
|     after_n_builds: 3 |     after_n_builds: 3 | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.github/codespell-words.txt
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/codespell-words.txt
									
									
									
									
										vendored
									
									
								
							| @ -2,5 +2,3 @@ keypair | |||||||
| keypairs | keypairs | ||||||
| hass | hass | ||||||
| warmup | warmup | ||||||
| ontext |  | ||||||
| singed |  | ||||||
|  | |||||||
							
								
								
									
										39
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										39
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @ -30,19 +30,17 @@ updates: | |||||||
|     open-pull-requests-limit: 10 |     open-pull-requests-limit: 10 | ||||||
|     commit-message: |     commit-message: | ||||||
|       prefix: "web:" |       prefix: "web:" | ||||||
|     # TODO: deduplicate these groups |  | ||||||
|     groups: |     groups: | ||||||
|       sentry: |       sentry: | ||||||
|         patterns: |         patterns: | ||||||
|           - "@sentry/*" |           - "@sentry/*" | ||||||
|           - "@spotlightjs/*" |  | ||||||
|       babel: |       babel: | ||||||
|         patterns: |         patterns: | ||||||
|           - "@babel/*" |           - "@babel/*" | ||||||
|           - "babel-*" |           - "babel-*" | ||||||
|       eslint: |       eslint: | ||||||
|         patterns: |         patterns: | ||||||
|           - "@typescript-eslint/*" |           - "@typescript-eslint/eslint-*" | ||||||
|           - "eslint" |           - "eslint" | ||||||
|           - "eslint-*" |           - "eslint-*" | ||||||
|       storybook: |       storybook: | ||||||
| @ -52,41 +50,6 @@ updates: | |||||||
|       esbuild: |       esbuild: | ||||||
|         patterns: |         patterns: | ||||||
|           - "@esbuild/*" |           - "@esbuild/*" | ||||||
|   - package-ecosystem: npm |  | ||||||
|     directory: "/tests/wdio" |  | ||||||
|     schedule: |  | ||||||
|       interval: daily |  | ||||||
|       time: "04:00" |  | ||||||
|     labels: |  | ||||||
|       - dependencies |  | ||||||
|     open-pull-requests-limit: 10 |  | ||||||
|     commit-message: |  | ||||||
|       prefix: "web:" |  | ||||||
|     # TODO: deduplicate these groups |  | ||||||
|     groups: |  | ||||||
|       sentry: |  | ||||||
|         patterns: |  | ||||||
|           - "@sentry/*" |  | ||||||
|           - "@spotlightjs/*" |  | ||||||
|       babel: |  | ||||||
|         patterns: |  | ||||||
|           - "@babel/*" |  | ||||||
|           - "babel-*" |  | ||||||
|       eslint: |  | ||||||
|         patterns: |  | ||||||
|           - "@typescript-eslint/*" |  | ||||||
|           - "eslint" |  | ||||||
|           - "eslint-*" |  | ||||||
|       storybook: |  | ||||||
|         patterns: |  | ||||||
|           - "@storybook/*" |  | ||||||
|           - "*storybook*" |  | ||||||
|       esbuild: |  | ||||||
|         patterns: |  | ||||||
|           - "@esbuild/*" |  | ||||||
|       wdio: |  | ||||||
|         patterns: |  | ||||||
|           - "@wdio/*" |  | ||||||
|   - package-ecosystem: npm |   - package-ecosystem: npm | ||||||
|     directory: "/website" |     directory: "/website" | ||||||
|     schedule: |     schedule: | ||||||
|  | |||||||
							
								
								
									
										1
									
								
								.github/pull_request_template.md
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.github/pull_request_template.md
									
									
									
									
										vendored
									
									
								
							| @ -27,6 +27,7 @@ If an API change has been made | |||||||
| If changes to the frontend have been made | If changes to the frontend have been made | ||||||
|  |  | ||||||
| -   [ ] The code has been formatted (`make web`) | -   [ ] The code has been formatted (`make web`) | ||||||
|  | -   [ ] The translation files have been updated (`make i18n-extract`) | ||||||
|  |  | ||||||
| If applicable | If applicable | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										157
									
								
								.github/workflows/ci-main.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										157
									
								
								.github/workflows/ci-main.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +1,3 @@ | |||||||
| --- |  | ||||||
| name: authentik-ci-main | name: authentik-ci-main | ||||||
|  |  | ||||||
| on: | on: | ||||||
| @ -8,11 +7,10 @@ on: | |||||||
|       - next |       - next | ||||||
|       - version-* |       - version-* | ||||||
|     paths-ignore: |     paths-ignore: | ||||||
|       - website/** |       - website | ||||||
|   pull_request: |   pull_request: | ||||||
|     branches: |     branches: | ||||||
|       - main |       - main | ||||||
|       - version-* |  | ||||||
|  |  | ||||||
| env: | env: | ||||||
|   POSTGRES_DB: authentik |   POSTGRES_DB: authentik | ||||||
| @ -28,11 +26,14 @@ jobs: | |||||||
|           - bandit |           - bandit | ||||||
|           - black |           - black | ||||||
|           - codespell |           - codespell | ||||||
|  |           - isort | ||||||
|           - pending-migrations |           - pending-migrations | ||||||
|  |           - pylint | ||||||
|  |           - pyright | ||||||
|           - ruff |           - ruff | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Setup authentik env |       - name: Setup authentik env | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|       - name: run job |       - name: run job | ||||||
| @ -40,40 +41,31 @@ jobs: | |||||||
|   test-migrations: |   test-migrations: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Setup authentik env |       - name: Setup authentik env | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|       - name: run migrations |       - name: run migrations | ||||||
|         run: poetry run python -m lifecycle.migrate |         run: poetry run python -m lifecycle.migrate | ||||||
|   test-migrations-from-stable: |   test-migrations-from-stable: | ||||||
|     name: test-migrations-from-stable - PostgreSQL ${{ matrix.psql }} |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     strategy: |     continue-on-error: true | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |  | ||||||
|         psql: |  | ||||||
|           - 12-alpine |  | ||||||
|           - 15-alpine |  | ||||||
|           - 16-alpine |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         with: |         with: | ||||||
|           fetch-depth: 0 |           fetch-depth: 0 | ||||||
|  |       - name: Setup authentik env | ||||||
|  |         uses: ./.github/actions/setup | ||||||
|       - name: checkout stable |       - name: checkout stable | ||||||
|         run: | |         run: | | ||||||
|           # Delete all poetry envs |  | ||||||
|           rm -rf /home/runner/.cache/pypoetry |  | ||||||
|           # Copy current, latest config to local |           # Copy current, latest config to local | ||||||
|           cp authentik/lib/default.yml local.env.yml |           cp authentik/lib/default.yml local.env.yml | ||||||
|           cp -R .github .. |           cp -R .github .. | ||||||
|           cp -R scripts .. |           cp -R scripts .. | ||||||
|           git checkout $(git tag --sort=version:refname | grep '^version/' | grep -vE -- '-rc[0-9]+$' | tail -n1) |           git checkout $(git describe --tags $(git rev-list --tags --max-count=1)) | ||||||
|           rm -rf .github/ scripts/ |           rm -rf .github/ scripts/ | ||||||
|           mv ../.github ../scripts . |           mv ../.github ../scripts . | ||||||
|       - name: Setup authentik env (stable) |       - name: Setup authentik env (ensure stable deps are installed) | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|         with: |  | ||||||
|           postgresql_version: ${{ matrix.psql }} |  | ||||||
|       - name: run migrations to stable |       - name: run migrations to stable | ||||||
|         run: poetry run python -m lifecycle.migrate |         run: poetry run python -m lifecycle.migrate | ||||||
|       - name: checkout current code |       - name: checkout current code | ||||||
| @ -83,21 +75,11 @@ jobs: | |||||||
|           git reset --hard HEAD |           git reset --hard HEAD | ||||||
|           git clean -d -fx . |           git clean -d -fx . | ||||||
|           git checkout $GITHUB_SHA |           git checkout $GITHUB_SHA | ||||||
|           # Delete previous poetry env |           poetry install | ||||||
|           rm -rf /home/runner/.cache/pypoetry/virtualenvs/* |  | ||||||
|       - name: Setup authentik env (ensure latest deps are installed) |       - name: Setup authentik env (ensure latest deps are installed) | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|         with: |  | ||||||
|           postgresql_version: ${{ matrix.psql }} |  | ||||||
|       - name: migrate to latest |       - name: migrate to latest | ||||||
|         run: | |         run: poetry run python -m lifecycle.migrate | ||||||
|           poetry run python -m lifecycle.migrate |  | ||||||
|       - name: run tests |  | ||||||
|         env: |  | ||||||
|           # Test in the main database that we just migrated from the previous stable version |  | ||||||
|           AUTHENTIK_POSTGRESQL__TEST__NAME: authentik |  | ||||||
|         run: | |  | ||||||
|           poetry run make test |  | ||||||
|   test-unittest: |   test-unittest: | ||||||
|     name: test-unittest - PostgreSQL ${{ matrix.psql }} |     name: test-unittest - PostgreSQL ${{ matrix.psql }} | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
| @ -108,40 +90,37 @@ jobs: | |||||||
|         psql: |         psql: | ||||||
|           - 12-alpine |           - 12-alpine | ||||||
|           - 15-alpine |           - 15-alpine | ||||||
|           - 16-alpine |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Setup authentik env |       - name: Setup authentik env | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|         with: |         with: | ||||||
|           postgresql_version: ${{ matrix.psql }} |           postgresql_tag: ${{ matrix.psql }} | ||||||
|       - name: run unittest |       - name: run unittest | ||||||
|         run: | |         run: | | ||||||
|           poetry run make test |           poetry run make test | ||||||
|           poetry run coverage xml |           poetry run coverage xml | ||||||
|       - if: ${{ always() }} |       - if: ${{ always() }} | ||||||
|         uses: codecov/codecov-action@v4 |         uses: codecov/codecov-action@v3 | ||||||
|         with: |         with: | ||||||
|           flags: unit |           flags: unit | ||||||
|           token: ${{ secrets.CODECOV_TOKEN }} |  | ||||||
|   test-integration: |   test-integration: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     timeout-minutes: 30 |     timeout-minutes: 30 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - 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.9.0 |         uses: helm/kind-action@v1.8.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@v4 |         uses: codecov/codecov-action@v3 | ||||||
|         with: |         with: | ||||||
|           flags: integration |           flags: integration | ||||||
|           token: ${{ secrets.CODECOV_TOKEN }} |  | ||||||
|   test-e2e: |   test-e2e: | ||||||
|     name: test-e2e (${{ matrix.job.name }}) |     name: test-e2e (${{ matrix.job.name }}) | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
| @ -165,14 +144,14 @@ jobs: | |||||||
|           - name: flows |           - name: flows | ||||||
|             glob: tests/e2e/test_flows* |             glob: tests/e2e/test_flows* | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Setup authentik env |       - name: Setup authentik env | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|       - name: Setup e2e env (chrome, etc) |       - name: Setup e2e env (chrome, etc) | ||||||
|         run: | |         run: | | ||||||
|           docker-compose -f tests/e2e/docker-compose.yml up -d |           docker-compose -f tests/e2e/docker-compose.yml up -d | ||||||
|       - id: cache-web |       - id: cache-web | ||||||
|         uses: actions/cache@v4 |         uses: actions/cache@v3 | ||||||
|         with: |         with: | ||||||
|           path: web/dist |           path: web/dist | ||||||
|           key: ${{ runner.os }}-web-${{ hashFiles('web/package-lock.json', 'web/src/**') }} |           key: ${{ runner.os }}-web-${{ hashFiles('web/package-lock.json', 'web/src/**') }} | ||||||
| @ -188,10 +167,9 @@ 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@v4 |         uses: codecov/codecov-action@v3 | ||||||
|         with: |         with: | ||||||
|           flags: e2e |           flags: e2e | ||||||
|           token: ${{ secrets.CODECOV_TOKEN }} |  | ||||||
|   ci-core-mark: |   ci-core-mark: | ||||||
|     needs: |     needs: | ||||||
|       - lint |       - lint | ||||||
| @ -204,74 +182,87 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - run: echo mark |       - run: echo mark | ||||||
|   build: |   build: | ||||||
|     strategy: |  | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |  | ||||||
|         arch: |  | ||||||
|           - amd64 |  | ||||||
|           - arm64 |  | ||||||
|     needs: ci-core-mark |     needs: ci-core-mark | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     permissions: |  | ||||||
|       # Needed to upload contianer images to ghcr.io |  | ||||||
|       packages: write |  | ||||||
|     timeout-minutes: 120 |     timeout-minutes: 120 | ||||||
|     if: "github.repository == 'goauthentik/authentik'" |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         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.0.0 |         uses: docker/setup-qemu-action@v2.2.0 | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v3 |         uses: docker/setup-buildx-action@v2 | ||||||
|       - name: prepare variables |       - name: prepare variables | ||||||
|         uses: ./.github/actions/docker-push-variables |         uses: ./.github/actions/docker-push-variables | ||||||
|         id: ev |         id: ev | ||||||
|         with: |         env: | ||||||
|           image-name: ghcr.io/goauthentik/dev-server |           DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }} | ||||||
|           image-arch: ${{ matrix.arch }} |  | ||||||
|       - name: Login to Container Registry |       - name: Login to Container Registry | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v2 | ||||||
|  |         if: ${{ steps.ev.outputs.shouldBuild == 'true' }} | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: generate ts client |  | ||||||
|         run: make gen-client-ts |  | ||||||
|       - name: Build Docker Image |       - name: Build Docker Image | ||||||
|         uses: docker/build-push-action@v5 |         uses: docker/build-push-action@v4 | ||||||
|         with: |         with: | ||||||
|           context: . |  | ||||||
|           secrets: | |           secrets: | | ||||||
|             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 }} |           push: ${{ steps.ev.outputs.shouldBuild == 'true' }} | ||||||
|           push: true |           tags: | | ||||||
|  |             ghcr.io/goauthentik/dev-server:gh-${{ steps.ev.outputs.branchNameContainer }} | ||||||
|  |             ghcr.io/goauthentik/dev-server:gh-${{ steps.ev.outputs.sha }} | ||||||
|  |             ghcr.io/goauthentik/dev-server:gh-${{ steps.ev.outputs.branchNameContainer }}-${{ steps.ev.outputs.timestamp }}-${{ steps.ev.outputs.shortHash }} | ||||||
|           build-args: | |           build-args: | | ||||||
|             GIT_BUILD_HASH=${{ steps.ev.outputs.sha }} |             GIT_BUILD_HASH=${{ steps.ev.outputs.sha }} | ||||||
|           cache-from: type=gha |             VERSION=${{ steps.ev.outputs.version }} | ||||||
|           cache-to: type=gha,mode=max |             VERSION_FAMILY=${{ steps.ev.outputs.versionFamily }} | ||||||
|           platforms: linux/${{ matrix.arch }} |       - name: Comment on PR | ||||||
|   pr-comment: |         if: github.event_name == 'pull_request' | ||||||
|     needs: |         continue-on-error: true | ||||||
|       - build |         uses: ./.github/actions/comment-pr-instructions | ||||||
|  |         with: | ||||||
|  |           tag: gh-${{ steps.ev.outputs.branchNameContainer }}-${{ steps.ev.outputs.timestamp }}-${{ steps.ev.outputs.shortHash }} | ||||||
|  |   build-arm64: | ||||||
|  |     needs: ci-core-mark | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: ${{ github.event_name == 'pull_request' }} |  | ||||||
|     permissions: |  | ||||||
|       # Needed to write comments on PRs |  | ||||||
|       pull-requests: write |  | ||||||
|     timeout-minutes: 120 |     timeout-minutes: 120 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         with: |         with: | ||||||
|           ref: ${{ github.event.pull_request.head.sha }} |           ref: ${{ github.event.pull_request.head.sha }} | ||||||
|  |       - name: Set up QEMU | ||||||
|  |         uses: docker/setup-qemu-action@v2.2.0 | ||||||
|  |       - name: Set up Docker Buildx | ||||||
|  |         uses: docker/setup-buildx-action@v2 | ||||||
|       - name: prepare variables |       - name: prepare variables | ||||||
|         uses: ./.github/actions/docker-push-variables |         uses: ./.github/actions/docker-push-variables | ||||||
|         id: ev |         id: ev | ||||||
|  |         env: | ||||||
|  |           DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }} | ||||||
|  |       - name: Login to Container Registry | ||||||
|  |         uses: docker/login-action@v2 | ||||||
|  |         if: ${{ steps.ev.outputs.shouldBuild == 'true' }} | ||||||
|         with: |         with: | ||||||
|           image-name: ghcr.io/goauthentik/dev-server |           registry: ghcr.io | ||||||
|       - name: Comment on PR |           username: ${{ github.repository_owner }} | ||||||
|         uses: ./.github/actions/comment-pr-instructions |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|  |       - name: Build Docker Image | ||||||
|  |         uses: docker/build-push-action@v4 | ||||||
|         with: |         with: | ||||||
|           tag: gh-${{ steps.ev.outputs.imageMainTag }} |           secrets: | | ||||||
|  |             GEOIPUPDATE_ACCOUNT_ID=${{ secrets.GEOIPUPDATE_ACCOUNT_ID }} | ||||||
|  |             GEOIPUPDATE_LICENSE_KEY=${{ secrets.GEOIPUPDATE_LICENSE_KEY }} | ||||||
|  |           push: ${{ steps.ev.outputs.shouldBuild == 'true' }} | ||||||
|  |           tags: | | ||||||
|  |             ghcr.io/goauthentik/dev-server:gh-${{ steps.ev.outputs.branchNameContainer }}-arm64 | ||||||
|  |             ghcr.io/goauthentik/dev-server:gh-${{ steps.ev.outputs.sha }}-arm64 | ||||||
|  |             ghcr.io/goauthentik/dev-server:gh-${{ steps.ev.outputs.branchNameContainer }}-${{ steps.ev.outputs.timestamp }}-${{ steps.ev.outputs.shortHash }}-arm64 | ||||||
|  |           build-args: | | ||||||
|  |             GIT_BUILD_HASH=${{ steps.ev.outputs.sha }} | ||||||
|  |             VERSION=${{ steps.ev.outputs.version }} | ||||||
|  |             VERSION_FAMILY=${{ steps.ev.outputs.versionFamily }} | ||||||
|  |           platforms: linux/arm64 | ||||||
|  | |||||||
							
								
								
									
										57
									
								
								.github/workflows/ci-outpost.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										57
									
								
								.github/workflows/ci-outpost.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +1,3 @@ | |||||||
| --- |  | ||||||
| name: authentik-ci-outpost | name: authentik-ci-outpost | ||||||
|  |  | ||||||
| on: | on: | ||||||
| @ -10,14 +9,13 @@ on: | |||||||
|   pull_request: |   pull_request: | ||||||
|     branches: |     branches: | ||||||
|       - main |       - main | ||||||
|       - version-* |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   lint-golint: |   lint-golint: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-go@v5 |       - uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version-file: "go.mod" |           go-version-file: "go.mod" | ||||||
|       - name: Prepare and generate API |       - name: Prepare and generate API | ||||||
| @ -29,20 +27,18 @@ jobs: | |||||||
|       - name: Generate API |       - name: Generate API | ||||||
|         run: make gen-client-go |         run: make gen-client-go | ||||||
|       - name: golangci-lint |       - name: golangci-lint | ||||||
|         uses: golangci/golangci-lint-action@v4 |         uses: golangci/golangci-lint-action@v3 | ||||||
|         with: |         with: | ||||||
|           version: v1.54.2 |           version: v1.52.2 | ||||||
|           args: --timeout 5000s --verbose |           args: --timeout 5000s --verbose | ||||||
|           skip-cache: true |           skip-pkg-cache: true | ||||||
|   test-unittest: |   test-unittest: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-go@v5 |       - uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version-file: "go.mod" |           go-version-file: "go.mod" | ||||||
|       - name: Setup authentik env |  | ||||||
|         uses: ./.github/actions/setup |  | ||||||
|       - name: Generate API |       - name: Generate API | ||||||
|         run: make gen-client-go |         run: make gen-client-go | ||||||
|       - name: Go unittests |       - name: Go unittests | ||||||
| @ -66,27 +62,23 @@ jobs: | |||||||
|           - proxy |           - proxy | ||||||
|           - ldap |           - ldap | ||||||
|           - radius |           - radius | ||||||
|           - rac |  | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     permissions: |  | ||||||
|       # Needed to upload contianer images to ghcr.io |  | ||||||
|       packages: write |  | ||||||
|     if: "github.repository == 'goauthentik/authentik'" |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         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.0.0 |         uses: docker/setup-qemu-action@v2.2.0 | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v3 |         uses: docker/setup-buildx-action@v2 | ||||||
|       - name: prepare variables |       - name: prepare variables | ||||||
|         uses: ./.github/actions/docker-push-variables |         uses: ./.github/actions/docker-push-variables | ||||||
|         id: ev |         id: ev | ||||||
|         with: |         env: | ||||||
|           image-name: ghcr.io/goauthentik/dev-${{ matrix.type }} |           DOCKER_USERNAME: ${{ secrets.DOCKER_USERNAME }} | ||||||
|       - name: Login to Container Registry |       - name: Login to Container Registry | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v2 | ||||||
|  |         if: ${{ steps.ev.outputs.shouldBuild == 'true' }} | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
| @ -94,17 +86,19 @@ jobs: | |||||||
|       - name: Generate API |       - name: Generate API | ||||||
|         run: make gen-client-go |         run: make gen-client-go | ||||||
|       - name: Build Docker Image |       - name: Build Docker Image | ||||||
|         uses: docker/build-push-action@v5 |         uses: docker/build-push-action@v4 | ||||||
|         with: |         with: | ||||||
|           tags: ${{ steps.ev.outputs.imageTags }} |           push: ${{ steps.ev.outputs.shouldBuild == 'true' }} | ||||||
|  |           tags: | | ||||||
|  |             ghcr.io/goauthentik/dev-${{ matrix.type }}:gh-${{ steps.ev.outputs.branchNameContainer }} | ||||||
|  |             ghcr.io/goauthentik/dev-${{ matrix.type }}:gh-${{ steps.ev.outputs.sha }} | ||||||
|           file: ${{ matrix.type }}.Dockerfile |           file: ${{ matrix.type }}.Dockerfile | ||||||
|           push: true |  | ||||||
|           build-args: | |           build-args: | | ||||||
|             GIT_BUILD_HASH=${{ steps.ev.outputs.sha }} |             GIT_BUILD_HASH=${{ steps.ev.outputs.sha }} | ||||||
|  |             VERSION=${{ steps.ev.outputs.version }} | ||||||
|  |             VERSION_FAMILY=${{ steps.ev.outputs.versionFamily }} | ||||||
|           platforms: linux/amd64,linux/arm64 |           platforms: linux/amd64,linux/arm64 | ||||||
|           context: . |           context: . | ||||||
|           cache-from: type=gha |  | ||||||
|           cache-to: type=gha,mode=max |  | ||||||
|   build-binary: |   build-binary: | ||||||
|     timeout-minutes: 120 |     timeout-minutes: 120 | ||||||
|     needs: |     needs: | ||||||
| @ -117,19 +111,18 @@ jobs: | |||||||
|           - proxy |           - proxy | ||||||
|           - ldap |           - ldap | ||||||
|           - radius |           - radius | ||||||
|           - rac |  | ||||||
|         goos: [linux] |         goos: [linux] | ||||||
|         goarch: [amd64, arm64] |         goarch: [amd64, arm64] | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         with: |         with: | ||||||
|           ref: ${{ github.event.pull_request.head.sha }} |           ref: ${{ github.event.pull_request.head.sha }} | ||||||
|       - uses: actions/setup-go@v5 |       - uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version-file: "go.mod" |           go-version-file: "go.mod" | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: web/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: web/package-lock.json |           cache-dependency-path: web/package-lock.json | ||||||
|       - name: Generate API |       - name: Generate API | ||||||
|  | |||||||
							
								
								
									
										55
									
								
								.github/workflows/ci-web.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										55
									
								
								.github/workflows/ci-web.yml
									
									
									
									
										vendored
									
									
								
							| @ -9,38 +9,31 @@ on: | |||||||
|   pull_request: |   pull_request: | ||||||
|     branches: |     branches: | ||||||
|       - main |       - main | ||||||
|       - version-* |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   lint-eslint: |   lint-eslint: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     strategy: |  | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |  | ||||||
|         project: |  | ||||||
|           - web |  | ||||||
|           - tests/wdio |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ${{ matrix.project }}/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: ${{ matrix.project }}/package-lock.json |           cache-dependency-path: web/package-lock.json | ||||||
|       - working-directory: ${{ matrix.project }}/ |       - working-directory: web/ | ||||||
|         run: npm ci |         run: npm ci | ||||||
|       - name: Generate API |       - name: Generate API | ||||||
|         run: make gen-client-ts |         run: make gen-client-ts | ||||||
|       - name: Eslint |       - name: Eslint | ||||||
|         working-directory: ${{ matrix.project }}/ |         working-directory: web/ | ||||||
|         run: npm run lint |         run: npm run lint | ||||||
|   lint-build: |   lint-build: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: web/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: web/package-lock.json |           cache-dependency-path: web/package-lock.json | ||||||
|       - working-directory: web/ |       - working-directory: web/ | ||||||
| @ -52,33 +45,27 @@ jobs: | |||||||
|         run: npm run tsc |         run: npm run tsc | ||||||
|   lint-prettier: |   lint-prettier: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     strategy: |  | ||||||
|       fail-fast: false |  | ||||||
|       matrix: |  | ||||||
|         project: |  | ||||||
|           - web |  | ||||||
|           - tests/wdio |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: ${{ matrix.project }}/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: ${{ matrix.project }}/package-lock.json |           cache-dependency-path: web/package-lock.json | ||||||
|       - working-directory: ${{ matrix.project }}/ |       - working-directory: web/ | ||||||
|         run: npm ci |         run: npm ci | ||||||
|       - name: Generate API |       - name: Generate API | ||||||
|         run: make gen-client-ts |         run: make gen-client-ts | ||||||
|       - name: prettier |       - name: prettier | ||||||
|         working-directory: ${{ matrix.project }}/ |         working-directory: web/ | ||||||
|         run: npm run prettier-check |         run: npm run prettier-check | ||||||
|   lint-lit-analyse: |   lint-lit-analyse: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: web/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: web/package-lock.json |           cache-dependency-path: web/package-lock.json | ||||||
|       - working-directory: web/ |       - working-directory: web/ | ||||||
| @ -107,10 +94,10 @@ jobs: | |||||||
|       - ci-web-mark |       - ci-web-mark | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: web/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: web/package-lock.json |           cache-dependency-path: web/package-lock.json | ||||||
|       - working-directory: web/ |       - working-directory: web/ | ||||||
|  | |||||||
							
								
								
									
										19
									
								
								.github/workflows/ci-website.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										19
									
								
								.github/workflows/ci-website.yml
									
									
									
									
										vendored
									
									
								
							| @ -9,16 +9,15 @@ on: | |||||||
|   pull_request: |   pull_request: | ||||||
|     branches: |     branches: | ||||||
|       - main |       - main | ||||||
|       - version-* |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   lint-prettier: |   lint-prettier: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: website/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: website/package-lock.json |           cache-dependency-path: website/package-lock.json | ||||||
|       - working-directory: website/ |       - working-directory: website/ | ||||||
| @ -29,10 +28,10 @@ jobs: | |||||||
|   test: |   test: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: website/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: website/package-lock.json |           cache-dependency-path: website/package-lock.json | ||||||
|       - working-directory: website/ |       - working-directory: website/ | ||||||
| @ -50,10 +49,10 @@ jobs: | |||||||
|           - build |           - build | ||||||
|           - build-docs-only |           - build-docs-only | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: website/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: website/package-lock.json |           cache-dependency-path: website/package-lock.json | ||||||
|       - working-directory: website/ |       - working-directory: website/ | ||||||
|  | |||||||
							
								
								
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										8
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @ -23,14 +23,14 @@ jobs: | |||||||
|         language: ["go", "javascript", "python"] |         language: ["go", "javascript", "python"] | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout repository |       - name: Checkout repository | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v3 | ||||||
|       - name: Setup authentik env |       - name: Setup authentik env | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|       - name: Initialize CodeQL |       - name: Initialize CodeQL | ||||||
|         uses: github/codeql-action/init@v3 |         uses: github/codeql-action/init@v2 | ||||||
|         with: |         with: | ||||||
|           languages: ${{ matrix.language }} |           languages: ${{ matrix.language }} | ||||||
|       - name: Autobuild |       - name: Autobuild | ||||||
|         uses: github/codeql-action/autobuild@v3 |         uses: github/codeql-action/autobuild@v2 | ||||||
|       - name: Perform CodeQL Analysis |       - name: Perform CodeQL Analysis | ||||||
|         uses: github/codeql-action/analyze@v3 |         uses: github/codeql-action/analyze@v2 | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								.github/workflows/gha-cache-cleanup.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/gha-cache-cleanup.yml
									
									
									
									
										vendored
									
									
								
							| @ -6,16 +6,12 @@ on: | |||||||
|     types: |     types: | ||||||
|       - closed |       - closed | ||||||
|  |  | ||||||
| permissions: |  | ||||||
|   # Permission to delete cache |  | ||||||
|   actions: write |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   cleanup: |   cleanup: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Check out code |       - name: Check out code | ||||||
|         uses: actions/checkout@v4 |         uses: actions/checkout@v3 | ||||||
|  |  | ||||||
|       - name: Cleanup |       - name: Cleanup | ||||||
|         run: | |         run: | | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								.github/workflows/ghcr-retention.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/ghcr-retention.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,8 +1,8 @@ | |||||||
| name: ghcr-retention | name: ghcr-retention | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   # schedule: |   schedule: | ||||||
|   #   - cron: "0 0 * * *" # every day at midnight |     - cron: "0 0 * * *" # every day at midnight | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
| @ -11,7 +11,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - id: generate_token |       - id: generate_token | ||||||
|         uses: tibdex/github-app-token@v2 |         uses: tibdex/github-app-token@v1 | ||||||
|         with: |         with: | ||||||
|           app_id: ${{ secrets.GH_APP_ID }} |           app_id: ${{ secrets.GH_APP_ID }} | ||||||
|           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} |           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								.github/workflows/image-compress.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/image-compress.yml
									
									
									
									
										vendored
									
									
								
							| @ -29,11 +29,11 @@ jobs: | |||||||
|        github.event.pull_request.head.repo.full_name == github.repository) |        github.event.pull_request.head.repo.full_name == github.repository) | ||||||
|     steps: |     steps: | ||||||
|       - id: generate_token |       - id: generate_token | ||||||
|         uses: tibdex/github-app-token@v2 |         uses: tibdex/github-app-token@v1 | ||||||
|         with: |         with: | ||||||
|           app_id: ${{ secrets.GH_APP_ID }} |           app_id: ${{ secrets.GH_APP_ID }} | ||||||
|           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} |           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         with: |         with: | ||||||
|           token: ${{ steps.generate_token.outputs.token }} |           token: ${{ steps.generate_token.outputs.token }} | ||||||
|       - name: Compress images |       - name: Compress images | ||||||
| @ -42,7 +42,7 @@ jobs: | |||||||
|         with: |         with: | ||||||
|           githubToken: ${{ steps.generate_token.outputs.token }} |           githubToken: ${{ steps.generate_token.outputs.token }} | ||||||
|           compressOnly: ${{ github.event_name != 'pull_request' }} |           compressOnly: ${{ github.event_name != 'pull_request' }} | ||||||
|       - uses: peter-evans/create-pull-request@v6 |       - uses: peter-evans/create-pull-request@v5 | ||||||
|         if: "${{ github.event_name != 'pull_request' && steps.compress.outputs.markdown != '' }}" |         if: "${{ github.event_name != 'pull_request' && steps.compress.outputs.markdown != '' }}" | ||||||
|         id: cpr |         id: cpr | ||||||
|         with: |         with: | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.github/workflows/publish-source-docs.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/publish-source-docs.yml
									
									
									
									
										vendored
									
									
								
							| @ -15,7 +15,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     timeout-minutes: 120 |     timeout-minutes: 120 | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Setup authentik env |       - name: Setup authentik env | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|       - name: generate docs |       - name: generate docs | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								.github/workflows/release-next-branch.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/workflows/release-next-branch.yml
									
									
									
									
										vendored
									
									
								
							| @ -6,7 +6,6 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| permissions: | permissions: | ||||||
|   # Needed to be able to push to the next branch |  | ||||||
|   contents: write |   contents: write | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
| @ -14,7 +13,7 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     environment: internal-production |     environment: internal-production | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         with: |         with: | ||||||
|           ref: main |           ref: main | ||||||
|       - run: | |       - run: | | ||||||
|  | |||||||
							
								
								
									
										96
									
								
								.github/workflows/release-publish.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										96
									
								
								.github/workflows/release-publish.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +1,3 @@ | |||||||
| --- |  | ||||||
| name: authentik-on-release | name: authentik-on-release | ||||||
|  |  | ||||||
| on: | on: | ||||||
| @ -8,50 +7,46 @@ on: | |||||||
| jobs: | jobs: | ||||||
|   build-server: |   build-server: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     permissions: |  | ||||||
|       # Needed to upload contianer images to ghcr.io |  | ||||||
|       packages: write |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Set up QEMU |       - name: Set up QEMU | ||||||
|         uses: docker/setup-qemu-action@v3.0.0 |         uses: docker/setup-qemu-action@v2.2.0 | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v3 |         uses: docker/setup-buildx-action@v2 | ||||||
|       - name: prepare variables |       - name: prepare variables | ||||||
|         uses: ./.github/actions/docker-push-variables |         uses: ./.github/actions/docker-push-variables | ||||||
|         id: ev |         id: ev | ||||||
|         with: |  | ||||||
|           image-name: ghcr.io/goauthentik/server,beryju/authentik |  | ||||||
|       - name: Docker Login Registry |       - name: Docker Login Registry | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v2 | ||||||
|         with: |         with: | ||||||
|           username: ${{ secrets.DOCKER_USERNAME }} |           username: ${{ secrets.DOCKER_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKER_PASSWORD }} |           password: ${{ secrets.DOCKER_PASSWORD }} | ||||||
|       - name: Login to GitHub Container Registry |       - name: Login to GitHub Container Registry | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v2 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: make empty clients |  | ||||||
|         run: | |  | ||||||
|           mkdir -p ./gen-ts-api |  | ||||||
|           mkdir -p ./gen-go-api |  | ||||||
|       - name: Build Docker Image |       - name: Build Docker Image | ||||||
|         uses: docker/build-push-action@v5 |         uses: docker/build-push-action@v4 | ||||||
|         with: |         with: | ||||||
|           context: . |           push: ${{ github.event_name == 'release' }} | ||||||
|           push: true |  | ||||||
|           secrets: | |           secrets: | | ||||||
|             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: | | ||||||
|  |             beryju/authentik:${{ steps.ev.outputs.version }}, | ||||||
|  |             beryju/authentik:${{ steps.ev.outputs.versionFamily }}, | ||||||
|  |             beryju/authentik:latest, | ||||||
|  |             ghcr.io/goauthentik/server:${{ steps.ev.outputs.version }}, | ||||||
|  |             ghcr.io/goauthentik/server:${{ steps.ev.outputs.versionFamily }}, | ||||||
|  |             ghcr.io/goauthentik/server:latest | ||||||
|           platforms: linux/amd64,linux/arm64 |           platforms: linux/amd64,linux/arm64 | ||||||
|  |           build-args: | | ||||||
|  |             VERSION=${{ steps.ev.outputs.version }} | ||||||
|  |             VERSION_FAMILY=${{ steps.ev.outputs.versionFamily }} | ||||||
|   build-outpost: |   build-outpost: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     permissions: |  | ||||||
|       # Needed to upload contianer images to ghcr.io |  | ||||||
|       packages: write |  | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
| @ -59,50 +54,48 @@ jobs: | |||||||
|           - proxy |           - proxy | ||||||
|           - ldap |           - ldap | ||||||
|           - radius |           - radius | ||||||
|           - rac |  | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-go@v5 |       - uses: actions/setup-go@v4 | ||||||
|         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.0.0 |         uses: docker/setup-qemu-action@v2.2.0 | ||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v3 |         uses: docker/setup-buildx-action@v2 | ||||||
|       - name: prepare variables |       - name: prepare variables | ||||||
|         uses: ./.github/actions/docker-push-variables |         uses: ./.github/actions/docker-push-variables | ||||||
|         id: ev |         id: ev | ||||||
|         with: |  | ||||||
|           image-name: ghcr.io/goauthentik/${{ matrix.type }},beryju/authentik-${{ matrix.type }} |  | ||||||
|       - name: make empty clients |  | ||||||
|         run: | |  | ||||||
|           mkdir -p ./gen-ts-api |  | ||||||
|           mkdir -p ./gen-go-api |  | ||||||
|       - name: Docker Login Registry |       - name: Docker Login Registry | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v2 | ||||||
|         with: |         with: | ||||||
|           username: ${{ secrets.DOCKER_USERNAME }} |           username: ${{ secrets.DOCKER_USERNAME }} | ||||||
|           password: ${{ secrets.DOCKER_PASSWORD }} |           password: ${{ secrets.DOCKER_PASSWORD }} | ||||||
|       - name: Login to GitHub Container Registry |       - name: Login to GitHub Container Registry | ||||||
|         uses: docker/login-action@v3 |         uses: docker/login-action@v2 | ||||||
|         with: |         with: | ||||||
|           registry: ghcr.io |           registry: ghcr.io | ||||||
|           username: ${{ github.repository_owner }} |           username: ${{ github.repository_owner }} | ||||||
|           password: ${{ secrets.GITHUB_TOKEN }} |           password: ${{ secrets.GITHUB_TOKEN }} | ||||||
|       - name: Build Docker Image |       - name: Build Docker Image | ||||||
|         uses: docker/build-push-action@v5 |         uses: docker/build-push-action@v4 | ||||||
|         with: |         with: | ||||||
|           push: true |           push: ${{ github.event_name == 'release' }} | ||||||
|           tags: ${{ steps.ev.outputs.imageTags }} |           tags: | | ||||||
|  |             beryju/authentik-${{ matrix.type }}:${{ steps.ev.outputs.version }}, | ||||||
|  |             beryju/authentik-${{ matrix.type }}:${{ steps.ev.outputs.versionFamily }}, | ||||||
|  |             beryju/authentik-${{ matrix.type }}:latest, | ||||||
|  |             ghcr.io/goauthentik/${{ matrix.type }}:${{ steps.ev.outputs.version }}, | ||||||
|  |             ghcr.io/goauthentik/${{ matrix.type }}:${{ steps.ev.outputs.versionFamily }}, | ||||||
|  |             ghcr.io/goauthentik/${{ matrix.type }}:latest | ||||||
|           file: ${{ matrix.type }}.Dockerfile |           file: ${{ matrix.type }}.Dockerfile | ||||||
|           platforms: linux/amd64,linux/arm64 |           platforms: linux/amd64,linux/arm64 | ||||||
|           context: . |           build-args: | | ||||||
|  |             VERSION=${{ steps.ev.outputs.version }} | ||||||
|  |             VERSION_FAMILY=${{ steps.ev.outputs.versionFamily }} | ||||||
|   build-outpost-binary: |   build-outpost-binary: | ||||||
|     timeout-minutes: 120 |     timeout-minutes: 120 | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     permissions: |  | ||||||
|       # Needed to upload binaries to the release |  | ||||||
|       contents: write |  | ||||||
|     strategy: |     strategy: | ||||||
|       fail-fast: false |       fail-fast: false | ||||||
|       matrix: |       matrix: | ||||||
| @ -113,13 +106,13 @@ jobs: | |||||||
|         goos: [linux, darwin] |         goos: [linux, darwin] | ||||||
|         goarch: [amd64, arm64] |         goarch: [amd64, arm64] | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - uses: actions/setup-go@v5 |       - uses: actions/setup-go@v4 | ||||||
|         with: |         with: | ||||||
|           go-version-file: "go.mod" |           go-version-file: "go.mod" | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: web/package.json |           node-version: "20" | ||||||
|           cache: "npm" |           cache: "npm" | ||||||
|           cache-dependency-path: web/package-lock.json |           cache-dependency-path: web/package-lock.json | ||||||
|       - name: Build web |       - name: Build web | ||||||
| @ -148,7 +141,7 @@ jobs: | |||||||
|       - build-outpost-binary |       - build-outpost-binary | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Run test suite in final docker images |       - name: Run test suite in final docker images | ||||||
|         run: | |         run: | | ||||||
|           echo "PG_PASS=$(openssl rand -base64 32)" >> .env |           echo "PG_PASS=$(openssl rand -base64 32)" >> .env | ||||||
| @ -164,20 +157,19 @@ jobs: | |||||||
|       - build-outpost-binary |       - build-outpost-binary | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: prepare variables |       - name: prepare variables | ||||||
|         uses: ./.github/actions/docker-push-variables |         uses: ./.github/actions/docker-push-variables | ||||||
|         id: ev |         id: ev | ||||||
|         with: |  | ||||||
|           image-name: ghcr.io/goauthentik/server |  | ||||||
|       - name: Get static files from docker image |       - name: Get static files from docker image | ||||||
|         run: | |         run: | | ||||||
|           docker pull ${{ steps.ev.outputs.imageMainTag }} |           docker pull ghcr.io/goauthentik/server:latest | ||||||
|           container=$(docker container create ${{ steps.ev.outputs.imageMainTag }}) |           container=$(docker container create ghcr.io/goauthentik/server:latest) | ||||||
|           docker cp ${container}:web/ . |           docker cp ${container}:web/ . | ||||||
|       - name: Create a Sentry.io release |       - name: Create a Sentry.io release | ||||||
|         uses: getsentry/action-release@v1 |         uses: getsentry/action-release@v1 | ||||||
|         continue-on-error: true |         continue-on-error: true | ||||||
|  |         if: ${{ github.event_name == 'release' }} | ||||||
|         env: |         env: | ||||||
|           SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }} |           SENTRY_AUTH_TOKEN: ${{ secrets.SENTRY_AUTH_TOKEN }} | ||||||
|           SENTRY_ORG: authentik-security-inc |           SENTRY_ORG: authentik-security-inc | ||||||
|  | |||||||
							
								
								
									
										20
									
								
								.github/workflows/release-tag.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										20
									
								
								.github/workflows/release-tag.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,4 +1,3 @@ | |||||||
| --- |  | ||||||
| name: authentik-on-tag | name: authentik-on-tag | ||||||
|  |  | ||||||
| on: | on: | ||||||
| @ -11,13 +10,12 @@ jobs: | |||||||
|     name: Create Release from Tag |     name: Create Release from Tag | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|       - name: Pre-release test |       - name: Pre-release test | ||||||
|         run: | |         run: | | ||||||
|           echo "PG_PASS=$(openssl rand -base64 32)" >> .env |           echo "PG_PASS=$(openssl rand -base64 32)" >> .env | ||||||
|           echo "AUTHENTIK_SECRET_KEY=$(openssl rand -base64 32)" >> .env |           echo "AUTHENTIK_SECRET_KEY=$(openssl rand -base64 32)" >> .env | ||||||
|           docker buildx install |           docker buildx install | ||||||
|           mkdir -p ./gen-ts-api |  | ||||||
|           docker build -t testing:latest . |           docker build -t testing:latest . | ||||||
|           echo "AUTHENTIK_IMAGE=testing" >> .env |           echo "AUTHENTIK_IMAGE=testing" >> .env | ||||||
|           echo "AUTHENTIK_TAG=latest" >> .env |           echo "AUTHENTIK_TAG=latest" >> .env | ||||||
| @ -25,15 +23,17 @@ jobs: | |||||||
|           docker-compose start postgresql redis |           docker-compose start postgresql redis | ||||||
|           docker-compose run -u root server test-all |           docker-compose run -u root server test-all | ||||||
|       - id: generate_token |       - id: generate_token | ||||||
|         uses: tibdex/github-app-token@v2 |         uses: tibdex/github-app-token@v1 | ||||||
|         with: |         with: | ||||||
|           app_id: ${{ secrets.GH_APP_ID }} |           app_id: ${{ secrets.GH_APP_ID }} | ||||||
|           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} |           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} | ||||||
|       - name: prepare variables |       - name: Extract version number | ||||||
|         uses: ./.github/actions/docker-push-variables |         id: get_version | ||||||
|         id: ev |         uses: actions/github-script@v6 | ||||||
|         with: |         with: | ||||||
|           image-name: ghcr.io/goauthentik/server |           github-token: ${{ steps.generate_token.outputs.token }} | ||||||
|  |           script: | | ||||||
|  |             return context.payload.ref.replace(/\/refs\/tags\/version\//, ''); | ||||||
|       - name: Create Release |       - name: Create Release | ||||||
|         id: create_release |         id: create_release | ||||||
|         uses: actions/create-release@v1.1.4 |         uses: actions/create-release@v1.1.4 | ||||||
| @ -41,6 +41,6 @@ jobs: | |||||||
|           GITHUB_TOKEN: ${{ steps.generate_token.outputs.token }} |           GITHUB_TOKEN: ${{ steps.generate_token.outputs.token }} | ||||||
|         with: |         with: | ||||||
|           tag_name: ${{ github.ref }} |           tag_name: ${{ github.ref }} | ||||||
|           release_name: Release ${{ steps.ev.outputs.version }} |           release_name: Release ${{ steps.get_version.outputs.result }} | ||||||
|           draft: true |           draft: true | ||||||
|           prerelease: ${{ steps.ev.outputs.prerelease == 'true' }} |           prerelease: false | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								.github/workflows/repo-stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/repo-stale.yml
									
									
									
									
										vendored
									
									
								
							| @ -6,19 +6,19 @@ on: | |||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
|  |  | ||||||
| permissions: | permissions: | ||||||
|   # Needed to update issues and PRs |  | ||||||
|   issues: write |   issues: write | ||||||
|  |   pull-requests: write | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   stale: |   stale: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - id: generate_token |       - id: generate_token | ||||||
|         uses: tibdex/github-app-token@v2 |         uses: tibdex/github-app-token@v1 | ||||||
|         with: |         with: | ||||||
|           app_id: ${{ secrets.GH_APP_ID }} |           app_id: ${{ secrets.GH_APP_ID }} | ||||||
|           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} |           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} | ||||||
|       - uses: actions/stale@v9 |       - uses: actions/stale@v8 | ||||||
|         with: |         with: | ||||||
|           repo-token: ${{ steps.generate_token.outputs.token }} |           repo-token: ${{ steps.generate_token.outputs.token }} | ||||||
|           days-before-stale: 60 |           days-before-stale: 60 | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								.github/workflows/translation-advice.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								.github/workflows/translation-advice.yml
									
									
									
									
										vendored
									
									
								
							| @ -7,26 +7,21 @@ on: | |||||||
|     paths: |     paths: | ||||||
|       - "!**" |       - "!**" | ||||||
|       - "locale/**" |       - "locale/**" | ||||||
|       - "!locale/en/**" |       - "web/src/locales/**" | ||||||
|       - "web/xliff/**" |  | ||||||
|  |  | ||||||
| permissions: |  | ||||||
|   # Permission to write comment |  | ||||||
|   pull-requests: write |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   post-comment: |   post-comment: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - name: Find Comment |       - name: Find Comment | ||||||
|         uses: peter-evans/find-comment@v3 |         uses: peter-evans/find-comment@v2 | ||||||
|         id: fc |         id: fc | ||||||
|         with: |         with: | ||||||
|           issue-number: ${{ github.event.pull_request.number }} |           issue-number: ${{ github.event.pull_request.number }} | ||||||
|           comment-author: "github-actions[bot]" |           comment-author: "github-actions[bot]" | ||||||
|           body-includes: authentik translations instructions |           body-includes: authentik translations instructions | ||||||
|       - name: Create or update comment |       - name: Create or update comment | ||||||
|         uses: peter-evans/create-or-update-comment@v4 |         uses: peter-evans/create-or-update-comment@v3 | ||||||
|         with: |         with: | ||||||
|           comment-id: ${{ steps.fc.outputs.comment-id }} |           comment-id: ${{ steps.fc.outputs.comment-id }} | ||||||
|           issue-number: ${{ github.event.pull_request.number }} |           issue-number: ${{ github.event.pull_request.number }} | ||||||
|  | |||||||
| @ -1,8 +1,9 @@ | |||||||
| --- | name: authentik-backend-translate-compile | ||||||
| name: authentik-backend-translate-extract-compile |  | ||||||
| on: | on: | ||||||
|   schedule: |   push: | ||||||
|     - cron: "0 0 * * *" # every day at midnight |     branches: [main] | ||||||
|  |     paths: | ||||||
|  |       - "locale/**" | ||||||
|   workflow_dispatch: |   workflow_dispatch: | ||||||
| 
 | 
 | ||||||
| env: | env: | ||||||
| @ -15,29 +16,25 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - id: generate_token |       - id: generate_token | ||||||
|         uses: tibdex/github-app-token@v2 |         uses: tibdex/github-app-token@v1 | ||||||
|         with: |         with: | ||||||
|           app_id: ${{ secrets.GH_APP_ID }} |           app_id: ${{ secrets.GH_APP_ID }} | ||||||
|           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} |           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         with: |         with: | ||||||
|           token: ${{ steps.generate_token.outputs.token }} |           token: ${{ steps.generate_token.outputs.token }} | ||||||
|       - name: Setup authentik env |       - name: Setup authentik env | ||||||
|         uses: ./.github/actions/setup |         uses: ./.github/actions/setup | ||||||
|       - name: run extract |  | ||||||
|         run: | |  | ||||||
|           poetry run make i18n-extract |  | ||||||
|       - name: run compile |       - name: run compile | ||||||
|         run: | |         run: poetry run ak compilemessages | ||||||
|           poetry run ak compilemessages |  | ||||||
|           make web-check-compile |  | ||||||
|       - name: Create Pull Request |       - name: Create Pull Request | ||||||
|         uses: peter-evans/create-pull-request@v6 |         uses: peter-evans/create-pull-request@v5 | ||||||
|  |         id: cpr | ||||||
|         with: |         with: | ||||||
|           token: ${{ steps.generate_token.outputs.token }} |           token: ${{ steps.generate_token.outputs.token }} | ||||||
|           branch: extract-compile-backend-translation |           branch: compile-backend-translation | ||||||
|           commit-message: "core, web: update translations" |           commit-message: "core: compile backend translations" | ||||||
|           title: "core, web: update translations" |           title: "core: compile backend translations" | ||||||
|           body: "core, web: update translations" |           body: "core: compile backend translations" | ||||||
|           delete-branch: true |           delete-branch: true | ||||||
|           signoff: true |           signoff: true | ||||||
							
								
								
									
										6
									
								
								.github/workflows/translation-rename.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/workflows/translation-rename.yml
									
									
									
									
										vendored
									
									
								
							| @ -6,17 +6,13 @@ on: | |||||||
|   pull_request: |   pull_request: | ||||||
|     types: [opened, reopened] |     types: [opened, reopened] | ||||||
|  |  | ||||||
| permissions: |  | ||||||
|   # Permission to rename PR |  | ||||||
|   pull-requests: write |  | ||||||
|  |  | ||||||
| jobs: | jobs: | ||||||
|   rename_pr: |   rename_pr: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     if: ${{ github.event.pull_request.user.login == 'transifex-integration[bot]'}} |     if: ${{ github.event.pull_request.user.login == 'transifex-integration[bot]'}} | ||||||
|     steps: |     steps: | ||||||
|       - id: generate_token |       - id: generate_token | ||||||
|         uses: tibdex/github-app-token@v2 |         uses: tibdex/github-app-token@v1 | ||||||
|         with: |         with: | ||||||
|           app_id: ${{ secrets.GH_APP_ID }} |           app_id: ${{ secrets.GH_APP_ID }} | ||||||
|           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} |           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} | ||||||
|  | |||||||
							
								
								
									
										10
									
								
								.github/workflows/web-api-publish.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										10
									
								
								.github/workflows/web-api-publish.yml
									
									
									
									
										vendored
									
									
								
							| @ -10,16 +10,16 @@ jobs: | |||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     steps: |     steps: | ||||||
|       - id: generate_token |       - id: generate_token | ||||||
|         uses: tibdex/github-app-token@v2 |         uses: tibdex/github-app-token@v1 | ||||||
|         with: |         with: | ||||||
|           app_id: ${{ secrets.GH_APP_ID }} |           app_id: ${{ secrets.GH_APP_ID }} | ||||||
|           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} |           private_key: ${{ secrets.GH_APP_PRIVATE_KEY }} | ||||||
|       - uses: actions/checkout@v4 |       - uses: actions/checkout@v3 | ||||||
|         with: |         with: | ||||||
|           token: ${{ steps.generate_token.outputs.token }} |           token: ${{ steps.generate_token.outputs.token }} | ||||||
|       - uses: actions/setup-node@v4 |       - uses: actions/setup-node@v3.8.1 | ||||||
|         with: |         with: | ||||||
|           node-version-file: web/package.json |           node-version: "20" | ||||||
|           registry-url: "https://registry.npmjs.org" |           registry-url: "https://registry.npmjs.org" | ||||||
|       - name: Generate API Client |       - name: Generate API Client | ||||||
|         run: make gen-client-ts |         run: make gen-client-ts | ||||||
| @ -35,7 +35,7 @@ jobs: | |||||||
|         run: | |         run: | | ||||||
|           export VERSION=`node -e 'console.log(require("../gen-ts-api/package.json").version)'` |           export VERSION=`node -e 'console.log(require("../gen-ts-api/package.json").version)'` | ||||||
|           npm i @goauthentik/api@$VERSION |           npm i @goauthentik/api@$VERSION | ||||||
|       - uses: peter-evans/create-pull-request@v6 |       - uses: peter-evans/create-pull-request@v5 | ||||||
|         id: cpr |         id: cpr | ||||||
|         with: |         with: | ||||||
|           token: ${{ steps.generate_token.outputs.token }} |           token: ${{ steps.generate_token.outputs.token }} | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -206,6 +206,3 @@ data/ | |||||||
| .netlify | .netlify | ||||||
| .ruff_cache | .ruff_cache | ||||||
| source_docs/ | source_docs/ | ||||||
|  |  | ||||||
| ### Golang ### |  | ||||||
| /vendor/ |  | ||||||
|  | |||||||
							
								
								
									
										4
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
								
							| @ -10,10 +10,10 @@ | |||||||
|         "Gruntfuggly.todo-tree", |         "Gruntfuggly.todo-tree", | ||||||
|         "mechatroner.rainbow-csv", |         "mechatroner.rainbow-csv", | ||||||
|         "ms-python.black-formatter", |         "ms-python.black-formatter", | ||||||
|         "charliermarsh.ruff", |         "ms-python.isort", | ||||||
|  |         "ms-python.pylint", | ||||||
|         "ms-python.python", |         "ms-python.python", | ||||||
|         "ms-python.vscode-pylance", |         "ms-python.vscode-pylance", | ||||||
|         "ms-python.black-formatter", |  | ||||||
|         "redhat.vscode-yaml", |         "redhat.vscode-yaml", | ||||||
|         "Tobermory.es6-string-html", |         "Tobermory.es6-string-html", | ||||||
|         "unifiedjs.vscode-mdx", |         "unifiedjs.vscode-mdx", | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @ -19,8 +19,10 @@ | |||||||
|         "slo", |         "slo", | ||||||
|         "scim", |         "scim", | ||||||
|     ], |     ], | ||||||
|  |     "python.linting.pylintEnabled": true, | ||||||
|     "todo-tree.tree.showCountsInTree": true, |     "todo-tree.tree.showCountsInTree": true, | ||||||
|     "todo-tree.tree.showBadges": true, |     "todo-tree.tree.showBadges": true, | ||||||
|  |     "python.formatting.provider": "black", | ||||||
|     "yaml.customTags": [ |     "yaml.customTags": [ | ||||||
|         "!Find sequence", |         "!Find sequence", | ||||||
|         "!KeyOf scalar", |         "!KeyOf scalar", | ||||||
|  | |||||||
							
								
								
									
										30
									
								
								CODEOWNERS
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								CODEOWNERS
									
									
									
									
									
								
							| @ -1,28 +1,2 @@ | |||||||
| # Fallback | *   @goauthentik/core | ||||||
| *                               @goauthentik/backend @goauthentik/frontend | website/docs/security/**    @goauthentik/security | ||||||
| # Backend |  | ||||||
| authentik/                      @goauthentik/backend |  | ||||||
| blueprints/                     @goauthentik/backend |  | ||||||
| cmd/                            @goauthentik/backend |  | ||||||
| internal/                       @goauthentik/backend |  | ||||||
| lifecycle/                      @goauthentik/backend |  | ||||||
| schemas/                        @goauthentik/backend |  | ||||||
| scripts/                        @goauthentik/backend |  | ||||||
| tests/                          @goauthentik/backend |  | ||||||
| pyproject.toml                  @goauthentik/backend |  | ||||||
| poetry.lock                     @goauthentik/backend |  | ||||||
| go.mod                          @goauthentik/backend |  | ||||||
| go.sum                          @goauthentik/backend |  | ||||||
| # Infrastructure |  | ||||||
| .github/                        @goauthentik/infrastructure |  | ||||||
| Dockerfile                      @goauthentik/infrastructure |  | ||||||
| *Dockerfile                     @goauthentik/infrastructure |  | ||||||
| .dockerignore                   @goauthentik/infrastructure |  | ||||||
| docker-compose.yml              @goauthentik/infrastructure |  | ||||||
| # Web |  | ||||||
| web/                            @goauthentik/frontend |  | ||||||
| tests/wdio/                     @goauthentik/frontend |  | ||||||
| # Docs & Website |  | ||||||
| website/                        @goauthentik/docs |  | ||||||
| # Security |  | ||||||
| website/docs/security/          @goauthentik/security |  | ||||||
|  | |||||||
							
								
								
									
										143
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										143
									
								
								Dockerfile
									
									
									
									
									
								
							| @ -1,77 +1,56 @@ | |||||||
| # syntax=docker/dockerfile:1 |  | ||||||
|  |  | ||||||
| # Stage 1: Build website | # Stage 1: Build website | ||||||
| FROM --platform=${BUILDPLATFORM} docker.io/node:21 as website-builder | FROM --platform=${BUILDPLATFORM} docker.io/node:20 as website-builder | ||||||
|  |  | ||||||
| ENV NODE_ENV=production |  | ||||||
|  |  | ||||||
| WORKDIR /work/website |  | ||||||
|  |  | ||||||
| RUN --mount=type=bind,target=/work/website/package.json,src=./website/package.json \ |  | ||||||
|     --mount=type=bind,target=/work/website/package-lock.json,src=./website/package-lock.json \ |  | ||||||
|     --mount=type=cache,id=npm-website,sharing=shared,target=/root/.npm \ |  | ||||||
|     npm ci --include=dev |  | ||||||
|  |  | ||||||
| COPY ./website /work/website/ | COPY ./website /work/website/ | ||||||
| COPY ./blueprints /work/blueprints/ | COPY ./blueprints /work/blueprints/ | ||||||
| COPY ./SECURITY.md /work/ | COPY ./SECURITY.md /work/ | ||||||
|  |  | ||||||
| RUN npm run build-docs-only | ENV NODE_ENV=production | ||||||
|  | WORKDIR /work/website | ||||||
|  | RUN npm ci --include=dev && npm run build-docs-only | ||||||
|  |  | ||||||
| # Stage 2: Build webui | # Stage 2: Build webui | ||||||
| FROM --platform=${BUILDPLATFORM} docker.io/node:21 as web-builder | FROM --platform=${BUILDPLATFORM} docker.io/node:20 as web-builder | ||||||
|  |  | ||||||
| ENV NODE_ENV=production |  | ||||||
|  |  | ||||||
| WORKDIR /work/web |  | ||||||
|  |  | ||||||
| RUN --mount=type=bind,target=/work/web/package.json,src=./web/package.json \ |  | ||||||
|     --mount=type=bind,target=/work/web/package-lock.json,src=./web/package-lock.json \ |  | ||||||
|     --mount=type=cache,id=npm-web,sharing=shared,target=/root/.npm \ |  | ||||||
|     npm ci --include=dev |  | ||||||
|  |  | ||||||
| COPY ./web /work/web/ | COPY ./web /work/web/ | ||||||
| COPY ./website /work/website/ | COPY ./website /work/website/ | ||||||
| COPY ./gen-ts-api /work/web/node_modules/@goauthentik/api |  | ||||||
|  |  | ||||||
| RUN npm run build | ENV NODE_ENV=production | ||||||
|  | WORKDIR /work/web | ||||||
|  | RUN npm ci --include=dev && npm run build | ||||||
|  |  | ||||||
| # Stage 3: Build go proxy | # Stage 3: Poetry to requirements.txt export | ||||||
| FROM --platform=${BUILDPLATFORM} docker.io/golang:1.22.0-bookworm AS go-builder | FROM docker.io/python:3.11.5-slim-bookworm AS poetry-locker | ||||||
|  |  | ||||||
| ARG TARGETOS | WORKDIR /work | ||||||
| ARG TARGETARCH | COPY ./pyproject.toml /work | ||||||
| ARG TARGETVARIANT | COPY ./poetry.lock /work | ||||||
|  |  | ||||||
| ARG GOOS=$TARGETOS | RUN pip install --no-cache-dir poetry && \ | ||||||
| ARG GOARCH=$TARGETARCH |     poetry export -f requirements.txt --output requirements.txt && \ | ||||||
|  |     poetry export -f requirements.txt --dev --output requirements-dev.txt | ||||||
|  |  | ||||||
| WORKDIR /go/src/goauthentik.io | # Stage 4: Build go proxy | ||||||
|  | FROM docker.io/golang:1.21.0-bookworm AS go-builder | ||||||
|  |  | ||||||
| RUN --mount=type=bind,target=/go/src/goauthentik.io/go.mod,src=./go.mod \ | WORKDIR /work | ||||||
|     --mount=type=bind,target=/go/src/goauthentik.io/go.sum,src=./go.sum \ |  | ||||||
|     --mount=type=cache,target=/go/pkg/mod \ |  | ||||||
|     go mod download |  | ||||||
|  |  | ||||||
| COPY ./cmd /go/src/goauthentik.io/cmd | COPY --from=web-builder /work/web/robots.txt /work/web/robots.txt | ||||||
| COPY ./authentik/lib /go/src/goauthentik.io/authentik/lib | COPY --from=web-builder /work/web/security.txt /work/web/security.txt | ||||||
| COPY ./web/static.go /go/src/goauthentik.io/web/static.go |  | ||||||
| COPY --from=web-builder /work/web/robots.txt /go/src/goauthentik.io/web/robots.txt |  | ||||||
| COPY --from=web-builder /work/web/security.txt /go/src/goauthentik.io/web/security.txt |  | ||||||
| COPY ./internal /go/src/goauthentik.io/internal |  | ||||||
| COPY ./go.mod /go/src/goauthentik.io/go.mod |  | ||||||
| COPY ./go.sum /go/src/goauthentik.io/go.sum |  | ||||||
|  |  | ||||||
| ENV CGO_ENABLED=0 | COPY ./cmd /work/cmd | ||||||
|  | COPY ./authentik/lib /work/authentik/lib | ||||||
|  | COPY ./web/static.go /work/web/static.go | ||||||
|  | COPY ./internal /work/internal | ||||||
|  | COPY ./go.mod /work/go.mod | ||||||
|  | COPY ./go.sum /work/go.sum | ||||||
|  |  | ||||||
| RUN --mount=type=cache,sharing=locked,target=/go/pkg/mod \ | RUN go build -o /work/bin/authentik ./cmd/server/ | ||||||
|     --mount=type=cache,id=go-build-$TARGETARCH$TARGETVARIANT,sharing=locked,target=/root/.cache/go-build \ |  | ||||||
|     GOARM="${TARGETVARIANT#v}" go build -o /go/authentik ./cmd/server |  | ||||||
|  |  | ||||||
| # Stage 4: MaxMind GeoIP | # Stage 5: MaxMind GeoIP | ||||||
| FROM --platform=${BUILDPLATFORM} ghcr.io/maxmind/geoipupdate:v6.1 as geoip | FROM ghcr.io/maxmind/geoipupdate:v6.0 as geoip | ||||||
|  |  | ||||||
| ENV GEOIPUPDATE_EDITION_IDS="GeoLite2-City GeoLite2-ASN" | ENV GEOIPUPDATE_EDITION_IDS="GeoLite2-City" | ||||||
| ENV GEOIPUPDATE_VERBOSE="true" | ENV GEOIPUPDATE_VERBOSE="true" | ||||||
| ENV GEOIPUPDATE_ACCOUNT_ID_FILE="/run/secrets/GEOIPUPDATE_ACCOUNT_ID" | ENV GEOIPUPDATE_ACCOUNT_ID_FILE="/run/secrets/GEOIPUPDATE_ACCOUNT_ID" | ||||||
| ENV GEOIPUPDATE_LICENSE_KEY_FILE="/run/secrets/GEOIPUPDATE_LICENSE_KEY" | ENV GEOIPUPDATE_LICENSE_KEY_FILE="/run/secrets/GEOIPUPDATE_LICENSE_KEY" | ||||||
| @ -82,33 +61,8 @@ RUN --mount=type=secret,id=GEOIPUPDATE_ACCOUNT_ID \ | |||||||
|     mkdir -p /usr/share/GeoIP && \ |     mkdir -p /usr/share/GeoIP && \ | ||||||
|     /bin/sh -c "/usr/bin/entry.sh || echo 'Failed to get GeoIP database, disabling'; exit 0" |     /bin/sh -c "/usr/bin/entry.sh || echo 'Failed to get GeoIP database, disabling'; exit 0" | ||||||
|  |  | ||||||
| # Stage 5: Python dependencies |  | ||||||
| FROM docker.io/python:3.12.2-slim-bookworm AS python-deps |  | ||||||
|  |  | ||||||
| WORKDIR /ak-root/poetry |  | ||||||
|  |  | ||||||
| ENV VENV_PATH="/ak-root/venv" \ |  | ||||||
|     POETRY_VIRTUALENVS_CREATE=false \ |  | ||||||
|     PATH="/ak-root/venv/bin:$PATH" |  | ||||||
|  |  | ||||||
| RUN rm -f /etc/apt/apt.conf.d/docker-clean; echo 'Binary::apt::APT::Keep-Downloaded-Packages "true";' > /etc/apt/apt.conf.d/keep-cache |  | ||||||
|  |  | ||||||
| RUN --mount=type=cache,id=apt-$TARGETARCH$TARGETVARIANT,sharing=locked,target=/var/cache/apt \ |  | ||||||
|     apt-get update && \ |  | ||||||
|     # Required for installing pip packages |  | ||||||
|     apt-get install -y --no-install-recommends build-essential pkg-config libxmlsec1-dev zlib1g-dev libpq-dev |  | ||||||
|  |  | ||||||
| RUN --mount=type=bind,target=./pyproject.toml,src=./pyproject.toml \ |  | ||||||
|     --mount=type=bind,target=./poetry.lock,src=./poetry.lock \ |  | ||||||
|     --mount=type=cache,target=/root/.cache/pip \ |  | ||||||
|     --mount=type=cache,target=/root/.cache/pypoetry \ |  | ||||||
|     python -m venv /ak-root/venv/ && \ |  | ||||||
|     pip3 install --upgrade pip && \ |  | ||||||
|     pip3 install poetry && \ |  | ||||||
|     poetry install --only=main --no-ansi --no-interaction |  | ||||||
|  |  | ||||||
| # Stage 6: Run | # Stage 6: Run | ||||||
| FROM docker.io/python:3.12.2-slim-bookworm AS final-image | FROM docker.io/python:3.11.5-slim-bookworm AS final-image | ||||||
|  |  | ||||||
| ARG GIT_BUILD_HASH | ARG GIT_BUILD_HASH | ||||||
| ARG VERSION | ARG VERSION | ||||||
| @ -122,45 +76,46 @@ LABEL org.opencontainers.image.revision ${GIT_BUILD_HASH} | |||||||
|  |  | ||||||
| WORKDIR / | WORKDIR / | ||||||
|  |  | ||||||
| # We cannot cache this layer otherwise we'll end up with a bigger image | COPY --from=poetry-locker /work/requirements.txt / | ||||||
|  | COPY --from=poetry-locker /work/requirements-dev.txt / | ||||||
|  | COPY --from=geoip /usr/share/GeoIP /geoip | ||||||
|  |  | ||||||
| RUN apt-get update && \ | RUN apt-get update && \ | ||||||
|  |     # Required for installing pip packages | ||||||
|  |     apt-get install -y --no-install-recommends build-essential pkg-config libxmlsec1-dev zlib1g-dev libpq-dev python3-dev && \ | ||||||
|     # Required for runtime |     # Required for runtime | ||||||
|     apt-get install -y --no-install-recommends libpq5 openssl libxmlsec1-openssl libmaxminddb0 ca-certificates && \ |     apt-get install -y --no-install-recommends libpq5 openssl libxmlsec1-openssl libmaxminddb0 && \ | ||||||
|     # Required for bootstrap & healtcheck |     # Required for bootstrap & healtcheck | ||||||
|     apt-get install -y --no-install-recommends runit && \ |     apt-get install -y --no-install-recommends runit && \ | ||||||
|  |     pip install --no-cache-dir -r /requirements.txt && \ | ||||||
|  |     apt-get remove --purge -y build-essential pkg-config libxmlsec1-dev libpq-dev python3-dev && \ | ||||||
|  |     apt-get autoremove --purge -y && \ | ||||||
|     apt-get clean && \ |     apt-get clean && \ | ||||||
|     rm -rf /tmp/* /var/lib/apt/lists/* /var/tmp/ && \ |     rm -rf /tmp/* /var/lib/apt/lists/* /var/tmp/ && \ | ||||||
|     adduser --system --no-create-home --uid 1000 --group --home /authentik authentik && \ |     adduser --system --no-create-home --uid 1000 --group --home /authentik authentik && \ | ||||||
|     mkdir -p /certs /media /blueprints && \ |     mkdir -p /certs /media /blueprints && \ | ||||||
|     mkdir -p /authentik/.ssh && \ |     mkdir -p /authentik/.ssh && \ | ||||||
|     mkdir -p /ak-root && \ |     chown authentik:authentik /certs /media /authentik/.ssh | ||||||
|     chown authentik:authentik /certs /media /authentik/.ssh /ak-root |  | ||||||
|  |  | ||||||
| COPY ./authentik/ /authentik | COPY ./authentik/ /authentik | ||||||
| COPY ./pyproject.toml / | COPY ./pyproject.toml / | ||||||
| COPY ./poetry.lock / |  | ||||||
| COPY ./schemas /schemas | COPY ./schemas /schemas | ||||||
| COPY ./locale /locale | COPY ./locale /locale | ||||||
| COPY ./tests /tests | COPY ./tests /tests | ||||||
| COPY ./manage.py / | COPY ./manage.py / | ||||||
| COPY ./blueprints /blueprints | COPY ./blueprints /blueprints | ||||||
| COPY ./lifecycle/ /lifecycle | COPY ./lifecycle/ /lifecycle | ||||||
| COPY --from=go-builder /go/authentik /bin/authentik | COPY --from=go-builder /work/bin/authentik /bin/authentik | ||||||
| COPY --from=python-deps /ak-root/venv /ak-root/venv |  | ||||||
| COPY --from=web-builder /work/web/dist/ /web/dist/ | COPY --from=web-builder /work/web/dist/ /web/dist/ | ||||||
| COPY --from=web-builder /work/web/authentik/ /web/authentik/ | COPY --from=web-builder /work/web/authentik/ /web/authentik/ | ||||||
| COPY --from=website-builder /work/website/help/ /website/help/ | COPY --from=website-builder /work/website/help/ /website/help/ | ||||||
| COPY --from=geoip /usr/share/GeoIP /geoip |  | ||||||
|  |  | ||||||
| USER 1000 | USER 1000 | ||||||
|  |  | ||||||
| ENV TMPDIR=/dev/shm/ \ | ENV TMPDIR /dev/shm/ | ||||||
|     PYTHONDONTWRITEBYTECODE=1 \ | ENV PYTHONUNBUFFERED 1 | ||||||
|     PYTHONUNBUFFERED=1 \ | ENV PATH "/usr/local/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/lifecycle" | ||||||
|     PATH="/ak-root/venv/bin:/lifecycle:$PATH" \ |  | ||||||
|     VENV_PATH="/ak-root/venv" \ |  | ||||||
|     POETRY_VIRTUALENVS_CREATE=false |  | ||||||
|  |  | ||||||
| HEALTHCHECK --interval=30s --timeout=30s --start-period=60s --retries=3 CMD [ "ak", "healthcheck" ] | HEALTHCHECK --interval=30s --timeout=30s --start-period=60s --retries=3 CMD [ "/lifecycle/ak", "healthcheck" ] | ||||||
|  |  | ||||||
| ENTRYPOINT [ "dumb-init", "--", "ak" ] | ENTRYPOINT [ "/usr/local/bin/dumb-init", "--", "/lifecycle/ak" ] | ||||||
|  | |||||||
							
								
								
									
										180
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										180
									
								
								Makefile
									
									
									
									
									
								
							| @ -1,19 +1,9 @@ | |||||||
| .PHONY: gen dev-reset all clean test web website | .SHELLFLAGS += -x -e | ||||||
|  |  | ||||||
| .SHELLFLAGS += ${SHELLFLAGS} -e |  | ||||||
| PWD = $(shell pwd) | 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 | PY_SOURCES = authentik tests scripts lifecycle | ||||||
| DOCKER_IMAGE ?= "authentik:test" |  | ||||||
|  |  | ||||||
| GEN_API_TS = "gen-ts-api" |  | ||||||
| GEN_API_GO = "gen-go-api" |  | ||||||
|  |  | ||||||
| pg_user := $(shell python -m authentik.lib.config postgresql.user 2>/dev/null) |  | ||||||
| pg_host := $(shell python -m authentik.lib.config postgresql.host 2>/dev/null) |  | ||||||
| pg_name := $(shell python -m authentik.lib.config postgresql.name 2>/dev/null) |  | ||||||
|  |  | ||||||
| CODESPELL_ARGS = -D - -D .github/codespell-dictionary.txt \ | CODESPELL_ARGS = -D - -D .github/codespell-dictionary.txt \ | ||||||
| 		-I .github/codespell-words.txt \ | 		-I .github/codespell-words.txt \ | ||||||
| @ -29,173 +19,128 @@ CODESPELL_ARGS = -D - -D .github/codespell-dictionary.txt \ | |||||||
| 		website/integrations \ | 		website/integrations \ | ||||||
| 		website/src | 		website/src | ||||||
|  |  | ||||||
| all: lint-fix lint test gen web  ## Lint, build, and test everything | all: lint-fix lint test gen web | ||||||
|  |  | ||||||
| HELP_WIDTH := $(shell grep -h '^[a-z][^ ]*:.*\#\#' $(MAKEFILE_LIST) 2>/dev/null | \ |  | ||||||
| 	cut -d':' -f1 | awk '{printf "%d\n", length}' | sort -rn | head -1) |  | ||||||
|  |  | ||||||
| help:  ## Show this help |  | ||||||
| 	@echo "\nSpecify a command. The choices are:\n" |  | ||||||
| 	@grep -Eh '^[0-9a-zA-Z_-]+:.*?## .*$$' $(MAKEFILE_LIST) | \ |  | ||||||
| 		awk 'BEGIN {FS = ":.*?## "}; {printf "  \033[0;36m%-$(HELP_WIDTH)s  \033[m %s\n", $$1, $$2}' | \ |  | ||||||
| 		sort |  | ||||||
| 	@echo "" |  | ||||||
|  |  | ||||||
| test-go: | test-go: | ||||||
| 	go test -timeout 0 -v -race -cover ./... | 	go test -timeout 0 -v -race -cover ./... | ||||||
|  |  | ||||||
| test-docker:  ## Run all tests in a docker-compose | test-docker: | ||||||
| 	echo "PG_PASS=$(openssl rand -base64 32)" >> .env | 	echo "PG_PASS=$(openssl rand -base64 32)" >> .env | ||||||
| 	echo "AUTHENTIK_SECRET_KEY=$(openssl rand -base64 32)" >> .env | 	echo "AUTHENTIK_SECRET_KEY=$(openssl rand -base64 32)" >> .env | ||||||
| 	docker-compose pull -q | 	docker-compose pull -q | ||||||
| 	docker-compose up --no-start | 	docker-compose up --no-start | ||||||
| 	docker-compose start postgresql redis | 	docker-compose start postgresql redis | ||||||
| 	docker-compose run -u root server test-all | 	docker-compose run -u root server test | ||||||
| 	rm -f .env | 	rm -f .env | ||||||
|  |  | ||||||
| test: ## Run the server tests and produce a coverage report (locally) | test: | ||||||
| 	coverage run manage.py test --keepdb authentik | 	coverage run manage.py test --keepdb authentik | ||||||
| 	coverage html | 	coverage html | ||||||
| 	coverage report | 	coverage report | ||||||
|  |  | ||||||
| lint-fix:  ## Lint and automatically fix errors in the python source code. Reports spelling errors. | lint-fix: | ||||||
| 	black $(PY_SOURCES) | 	isort authentik $(PY_SOURCES) | ||||||
| 	ruff check --fix $(PY_SOURCES) | 	black authentik $(PY_SOURCES) | ||||||
|  | 	ruff authentik $(PY_SOURCES) | ||||||
| 	codespell -w $(CODESPELL_ARGS) | 	codespell -w $(CODESPELL_ARGS) | ||||||
|  |  | ||||||
| lint: ## Lint the python and golang sources | lint: | ||||||
|  | 	pylint $(PY_SOURCES) | ||||||
| 	bandit -r $(PY_SOURCES) -x node_modules | 	bandit -r $(PY_SOURCES) -x node_modules | ||||||
| 	golangci-lint run -v | 	golangci-lint run -v | ||||||
|  |  | ||||||
| core-install: | migrate: | ||||||
| 	poetry install |  | ||||||
|  |  | ||||||
| migrate: ## Run the Authentik Django server's migrations |  | ||||||
| 	python -m lifecycle.migrate | 	python -m lifecycle.migrate | ||||||
|  |  | ||||||
| i18n-extract: core-i18n-extract web-i18n-extract  ## Extract strings that require translation into files to send to a translation service | i18n-extract: i18n-extract-core web-i18n-extract | ||||||
|  |  | ||||||
| core-i18n-extract: | i18n-extract-core: | ||||||
| 	ak makemessages \ | 	ak makemessages --ignore web --ignore internal --ignore web --ignore web-api --ignore website -l en | ||||||
| 		--add-location file \ |  | ||||||
| 		--no-obsolete \ |  | ||||||
| 		--ignore web \ |  | ||||||
| 		--ignore internal \ |  | ||||||
| 		--ignore ${GEN_API_TS} \ |  | ||||||
| 		--ignore ${GEN_API_GO} \ |  | ||||||
| 		--ignore website \ |  | ||||||
| 		-l en |  | ||||||
|  |  | ||||||
| install: web-install website-install core-install  ## Install all requires dependencies for `web`, `website` and `core` |  | ||||||
|  |  | ||||||
| dev-drop-db: |  | ||||||
| 	dropdb -U ${pg_user} -h ${pg_host} ${pg_name} |  | ||||||
| 	# Also remove the test-db if it exists |  | ||||||
| 	dropdb -U ${pg_user} -h ${pg_host} test_${pg_name} || true |  | ||||||
| 	redis-cli -n 0 flushall |  | ||||||
|  |  | ||||||
| dev-create-db: |  | ||||||
| 	createdb -U ${pg_user} -h ${pg_host} ${pg_name} |  | ||||||
|  |  | ||||||
| dev-reset: dev-drop-db dev-create-db migrate  ## Drop and restore the Authentik PostgreSQL instance to a "fresh install" state. |  | ||||||
|  |  | ||||||
| ######################### | ######################### | ||||||
| ## API Schema | ## API Schema | ||||||
| ######################### | ######################### | ||||||
|  |  | ||||||
| gen-build:  ## Extract the schema from the database | gen-build: | ||||||
| 	AUTHENTIK_DEBUG=true \ | 	AUTHENTIK_DEBUG=true ak make_blueprint_schema > blueprints/schema.json | ||||||
| 		AUTHENTIK_TENANTS__ENABLED=true \ | 	AUTHENTIK_DEBUG=true ak spectacular --file schema.yml | ||||||
| 		AUTHENTIK_OUTPOSTS__DISABLE_EMBEDDED_OUTPOST=true \ |  | ||||||
| 		ak make_blueprint_schema > blueprints/schema.json |  | ||||||
| 	AUTHENTIK_DEBUG=true \ |  | ||||||
| 		AUTHENTIK_TENANTS__ENABLED=true \ |  | ||||||
| 		AUTHENTIK_OUTPOSTS__DISABLE_EMBEDDED_OUTPOST=true \ |  | ||||||
| 		ak spectacular --file schema.yml |  | ||||||
|  |  | ||||||
| gen-changelog:  ## (Release) generate the changelog based from the commits since the last tag | gen-changelog: | ||||||
| 	git log --pretty=format:" - %s" $(shell git describe --tags $(shell git rev-list --tags --max-count=1))...$(shell git branch --show-current) | sort > changelog.md | 	git log --pretty=format:" - %s" $(shell git describe --tags $(shell git rev-list --tags --max-count=1))...$(shell git branch --show-current) | sort > changelog.md | ||||||
| 	npx prettier --write changelog.md | 	npx prettier --write changelog.md | ||||||
|  |  | ||||||
| gen-diff:  ## (Release) generate the changelog diff between the current schema and the last tag | gen-diff: | ||||||
| 	git show $(shell git describe --tags $(shell git rev-list --tags --max-count=1)):schema.yml > old_schema.yml | 	git show $(shell git describe --tags $(shell git rev-list --tags --max-count=1)):schema.yml > old_schema.yml | ||||||
| 	docker run \ | 	docker run \ | ||||||
| 		--rm -v ${PWD}:/local \ | 		--rm -v ${PWD}:/local \ | ||||||
| 		--user ${UID}:${GID} \ | 		--user ${UID}:${GID} \ | ||||||
| 		docker.io/openapitools/openapi-diff:2.1.0-beta.8 \ | 		docker.io/openapitools/openapi-diff:2.1.0-beta.6 \ | ||||||
| 		--markdown /local/diff.md \ | 		--markdown /local/diff.md \ | ||||||
| 		/local/old_schema.yml /local/schema.yml | 		/local/old_schema.yml /local/schema.yml | ||||||
| 	rm old_schema.yml | 	rm old_schema.yml | ||||||
| 	sed -i 's/{/{/g' diff.md |  | ||||||
| 	sed -i 's/}/}/g' diff.md |  | ||||||
| 	npx prettier --write diff.md | 	npx prettier --write diff.md | ||||||
|  |  | ||||||
| gen-clean-ts:  ## Remove generated API client for Typescript | gen-clean: | ||||||
| 	rm -rf ./${GEN_API_TS}/ | 	rm -rf web/api/src/ | ||||||
| 	rm -rf ./web/node_modules/@goauthentik/api/ | 	rm -rf api/ | ||||||
|  |  | ||||||
| gen-clean-go:  ## Remove generated API client for Go | gen-client-ts: | ||||||
| 	rm -rf ./${GEN_API_GO}/ |  | ||||||
|  |  | ||||||
| gen-clean: gen-clean-ts gen-clean-go  ## Remove generated API clients |  | ||||||
|  |  | ||||||
| gen-client-ts: gen-clean-ts  ## Build and install the authentik API for Typescript into the authentik UI Application |  | ||||||
| 	docker run \ | 	docker run \ | ||||||
| 		--rm -v ${PWD}:/local \ | 		--rm -v ${PWD}:/local \ | ||||||
| 		--user ${UID}:${GID} \ | 		--user ${UID}:${GID} \ | ||||||
| 		docker.io/openapitools/openapi-generator-cli:v6.5.0 generate \ | 		docker.io/openapitools/openapi-generator-cli:v6.5.0 generate \ | ||||||
| 		-i /local/schema.yml \ | 		-i /local/schema.yml \ | ||||||
| 		-g typescript-fetch \ | 		-g typescript-fetch \ | ||||||
| 		-o /local/${GEN_API_TS} \ | 		-o /local/gen-ts-api \ | ||||||
| 		-c /local/scripts/api-ts-config.yaml \ | 		-c /local/scripts/api-ts-config.yaml \ | ||||||
| 		--additional-properties=npmVersion=${NPM_VERSION} \ | 		--additional-properties=npmVersion=${NPM_VERSION} \ | ||||||
| 		--git-repo-id authentik \ | 		--git-repo-id authentik \ | ||||||
| 		--git-user-id goauthentik | 		--git-user-id goauthentik | ||||||
| 	mkdir -p web/node_modules/@goauthentik/api | 	mkdir -p web/node_modules/@goauthentik/api | ||||||
| 	cd ./${GEN_API_TS} && npm i | 	cd gen-ts-api && npm i | ||||||
| 	\cp -rf ./${GEN_API_TS}/* web/node_modules/@goauthentik/api | 	\cp -rfv gen-ts-api/* web/node_modules/@goauthentik/api | ||||||
|  |  | ||||||
| gen-client-go: gen-clean-go  ## Build and install the authentik API for Golang | gen-client-go: | ||||||
| 	mkdir -p ./${GEN_API_GO} ./${GEN_API_GO}/templates | 	mkdir -p ./gen-go-api ./gen-go-api/templates | ||||||
| 	wget https://raw.githubusercontent.com/goauthentik/client-go/main/config.yaml -O ./${GEN_API_GO}/config.yaml | 	wget https://raw.githubusercontent.com/goauthentik/client-go/main/config.yaml -O ./gen-go-api/config.yaml | ||||||
| 	wget https://raw.githubusercontent.com/goauthentik/client-go/main/templates/README.mustache -O ./${GEN_API_GO}/templates/README.mustache | 	wget https://raw.githubusercontent.com/goauthentik/client-go/main/templates/README.mustache -O ./gen-go-api/templates/README.mustache | ||||||
| 	wget https://raw.githubusercontent.com/goauthentik/client-go/main/templates/go.mod.mustache -O ./${GEN_API_GO}/templates/go.mod.mustache | 	wget https://raw.githubusercontent.com/goauthentik/client-go/main/templates/go.mod.mustache -O ./gen-go-api/templates/go.mod.mustache | ||||||
| 	cp schema.yml ./${GEN_API_GO}/ | 	cp schema.yml ./gen-go-api/ | ||||||
| 	docker run \ | 	docker run \ | ||||||
| 		--rm -v ${PWD}/${GEN_API_GO}:/local \ | 		--rm -v ${PWD}/gen-go-api:/local \ | ||||||
| 		--user ${UID}:${GID} \ | 		--user ${UID}:${GID} \ | ||||||
| 		docker.io/openapitools/openapi-generator-cli:v6.5.0 generate \ | 		docker.io/openapitools/openapi-generator-cli:v6.5.0 generate \ | ||||||
| 		-i /local/schema.yml \ | 		-i /local/schema.yml \ | ||||||
| 		-g go \ | 		-g go \ | ||||||
| 		-o /local/ \ | 		-o /local/ \ | ||||||
| 		-c /local/config.yaml | 		-c /local/config.yaml | ||||||
| 	go mod edit -replace goauthentik.io/api/v3=./${GEN_API_GO} | 	go mod edit -replace goauthentik.io/api/v3=./gen-go-api | ||||||
| 	rm -rf ./${GEN_API_GO}/config.yaml ./${GEN_API_GO}/templates/ | 	rm -rf ./gen-go-api/config.yaml ./gen-go-api/templates/ | ||||||
|  |  | ||||||
| gen-dev-config:  ## Generate a local development config file | gen-dev-config: | ||||||
| 	python -m scripts.generate_config | 	python -m scripts.generate_config | ||||||
|  |  | ||||||
| gen: gen-build gen-client-ts | gen: gen-build gen-clean gen-client-ts | ||||||
|  |  | ||||||
| ######################### | ######################### | ||||||
| ## Web | ## Web | ||||||
| ######################### | ######################### | ||||||
|  |  | ||||||
| web-build: web-install  ## Build the Authentik UI | web-build: web-install | ||||||
| 	cd web && npm run build | 	cd web && npm run build | ||||||
|  |  | ||||||
| web: web-lint-fix web-lint web-check-compile  ## Automatically fix formatting issues in the Authentik UI source code, lint the code, and compile it | web: web-lint-fix web-lint web-check-compile | ||||||
|  |  | ||||||
| web-install:  ## Install the necessary libraries to build the Authentik UI | web-install: | ||||||
| 	cd web && npm ci | 	cd web && npm ci | ||||||
|  |  | ||||||
| web-watch:  ## Build and watch the Authentik UI for changes, updating automatically | web-watch: | ||||||
| 	rm -rf web/dist/ | 	rm -rf web/dist/ | ||||||
| 	mkdir web/dist/ | 	mkdir web/dist/ | ||||||
| 	touch web/dist/.gitkeep | 	touch web/dist/.gitkeep | ||||||
| 	cd web && npm run watch | 	cd web && npm run watch | ||||||
|  |  | ||||||
| web-storybook-watch:  ## Build and run the storybook documentation server | web-storybook-watch: | ||||||
| 	cd web && npm run storybook | 	cd web && npm run storybook | ||||||
|  |  | ||||||
| web-lint-fix: | web-lint-fix: | ||||||
| @ -215,7 +160,7 @@ web-i18n-extract: | |||||||
| ## Website | ## Website | ||||||
| ######################### | ######################### | ||||||
|  |  | ||||||
| website: website-lint-fix website-build  ## Automatically fix formatting issues in the Authentik website/docs source code, lint the code, and compile it | website: website-lint-fix website-build | ||||||
|  |  | ||||||
| website-install: | website-install: | ||||||
| 	cd website && npm ci | 	cd website && npm ci | ||||||
| @ -226,26 +171,18 @@ website-lint-fix: | |||||||
| website-build: | website-build: | ||||||
| 	cd website && npm run build | 	cd website && npm run build | ||||||
|  |  | ||||||
| website-watch:  ## Build and watch the documentation website, updating automatically | website-watch: | ||||||
| 	cd website && npm run watch | 	cd website && npm run watch | ||||||
|  |  | ||||||
| ######################### |  | ||||||
| ## Docker |  | ||||||
| ######################### |  | ||||||
|  |  | ||||||
| docker:  ## Build a docker image of the current source tree |  | ||||||
| 	DOCKER_BUILDKIT=1 docker build . --progress plain --tag ${DOCKER_IMAGE} |  | ||||||
|  |  | ||||||
| ######################### |  | ||||||
| ## CI |  | ||||||
| ######################### |  | ||||||
| # These targets are use by GitHub actions to allow usage of matrix | # These targets are use by GitHub actions to allow usage of matrix | ||||||
| # which makes the YAML File a lot smaller | # which makes the YAML File a lot smaller | ||||||
|  |  | ||||||
| ci--meta-debug: | ci--meta-debug: | ||||||
| 	python -V | 	python -V | ||||||
| 	node --version | 	node --version | ||||||
|  |  | ||||||
|  | ci-pylint: ci--meta-debug | ||||||
|  | 	pylint $(PY_SOURCES) | ||||||
|  |  | ||||||
| ci-black: ci--meta-debug | ci-black: ci--meta-debug | ||||||
| 	black --check $(PY_SOURCES) | 	black --check $(PY_SOURCES) | ||||||
|  |  | ||||||
| @ -255,8 +192,25 @@ ci-ruff: ci--meta-debug | |||||||
| ci-codespell: ci--meta-debug | ci-codespell: ci--meta-debug | ||||||
| 	codespell $(CODESPELL_ARGS) -s | 	codespell $(CODESPELL_ARGS) -s | ||||||
|  |  | ||||||
|  | ci-isort: ci--meta-debug | ||||||
|  | 	isort --check $(PY_SOURCES) | ||||||
|  |  | ||||||
| ci-bandit: ci--meta-debug | ci-bandit: ci--meta-debug | ||||||
| 	bandit -r $(PY_SOURCES) | 	bandit -r $(PY_SOURCES) | ||||||
|  |  | ||||||
|  | ci-pyright: ci--meta-debug | ||||||
|  | 	./web/node_modules/.bin/pyright $(PY_SOURCES) | ||||||
|  |  | ||||||
| ci-pending-migrations: ci--meta-debug | ci-pending-migrations: ci--meta-debug | ||||||
| 	ak makemigrations --check | 	ak makemigrations --check | ||||||
|  |  | ||||||
|  | install: web-install website-install | ||||||
|  | 	poetry install | ||||||
|  |  | ||||||
|  | dev-reset: | ||||||
|  | 	dropdb -U postgres -h localhost authentik | ||||||
|  | 	# Also remove the test-db if it exists | ||||||
|  | 	dropdb -U postgres -h localhost test_authentik || true | ||||||
|  | 	createdb -U postgres -h localhost authentik | ||||||
|  | 	redis-cli -n 0 flushall | ||||||
|  | 	make migrate | ||||||
|  | |||||||
							
								
								
									
										12
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										12
									
								
								README.md
									
									
									
									
									
								
							| @ -41,3 +41,15 @@ See [SECURITY.md](SECURITY.md) | |||||||
| ## Adoption and Contributions | ## Adoption and Contributions | ||||||
|  |  | ||||||
| Your organization uses authentik? We'd love to add your logo to the readme and our website! Email us @ hello@goauthentik.io or open a GitHub Issue/PR! For more information on how to contribute to authentik, please refer to our [CONTRIBUTING.md file](./CONTRIBUTING.md). | Your organization uses authentik? We'd love to add your logo to the readme and our website! Email us @ hello@goauthentik.io or open a GitHub Issue/PR! For more information on how to contribute to authentik, please refer to our [CONTRIBUTING.md file](./CONTRIBUTING.md). | ||||||
|  |  | ||||||
|  | ## Sponsors | ||||||
|  |  | ||||||
|  | This project is proudly sponsored by: | ||||||
|  |  | ||||||
|  | <p> | ||||||
|  |     <a href="https://www.digitalocean.com/?utm_medium=opensource&utm_source=goauthentik.io"> | ||||||
|  |         <img src="https://opensource.nyc3.cdn.digitaloceanspaces.com/attribution/assets/SVG/DO_Logo_horizontal_blue.svg" width="201px"> | ||||||
|  |     </a> | ||||||
|  | </p> | ||||||
|  |  | ||||||
|  | DigitalOcean provides development and testing resources for authentik. | ||||||
|  | |||||||
| @ -1,9 +1,5 @@ | |||||||
| authentik takes security very seriously. We follow the rules of [responsible disclosure](https://en.wikipedia.org/wiki/Responsible_disclosure), and we urge our community to do so as well, instead of reporting vulnerabilities publicly. This allows us to patch the issue quickly, announce it's existence and release the fixed version. | authentik takes security very seriously. We follow the rules of [responsible disclosure](https://en.wikipedia.org/wiki/Responsible_disclosure), and we urge our community to do so as well, instead of reporting vulnerabilities publicly. This allows us to patch the issue quickly, announce it's existence and release the fixed version. | ||||||
|  |  | ||||||
| ## Independent audits and pentests |  | ||||||
|  |  | ||||||
| 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 | ||||||
|  |  | ||||||
| CVE (Common Vulnerability and Exposure) is a system designed to aggregate all vulnerabilities. As such, a CVE will be issued when there is a either vulnerability or exposure. Per NIST, A vulnerability is: | CVE (Common Vulnerability and Exposure) is a system designed to aggregate all vulnerabilities. As such, a CVE will be issued when there is a either vulnerability or exposure. Per NIST, A vulnerability is: | ||||||
|  | |||||||
| @ -1,12 +1,12 @@ | |||||||
| """authentik root module""" | """authentik root module""" | ||||||
|  |  | ||||||
| from os import environ | from os import environ | ||||||
|  | from typing import Optional | ||||||
|  |  | ||||||
| __version__ = "2024.2.1" | __version__ = "2023.8.1" | ||||||
| ENV_GIT_HASH_KEY = "GIT_BUILD_HASH" | ENV_GIT_HASH_KEY = "GIT_BUILD_HASH" | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_build_hash(fallback: str | None = None) -> str: | def get_build_hash(fallback: Optional[str] = None) -> str: | ||||||
|     """Get build hash""" |     """Get build hash""" | ||||||
|     build_hash = environ.get(ENV_GIT_HASH_KEY, fallback if fallback else "") |     build_hash = environ.get(ENV_GIT_HASH_KEY, fallback if fallback else "") | ||||||
|     return fallback if build_hash == "" and fallback else build_hash |     return fallback if build_hash == "" and fallback else build_hash | ||||||
|  | |||||||
| @ -1,8 +1,7 @@ | |||||||
| """Meta API""" | """Meta API""" | ||||||
|  |  | ||||||
| from drf_spectacular.utils import extend_schema | from drf_spectacular.utils import extend_schema | ||||||
| from rest_framework.fields import CharField | from rest_framework.fields import CharField | ||||||
| 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.viewsets import ViewSet | from rest_framework.viewsets import ViewSet | ||||||
| @ -22,7 +21,7 @@ class AppSerializer(PassiveSerializer): | |||||||
| class AppsViewSet(ViewSet): | class AppsViewSet(ViewSet): | ||||||
|     """Read-only view list all installed apps""" |     """Read-only view list all installed apps""" | ||||||
|  |  | ||||||
|     permission_classes = [IsAuthenticated] |     permission_classes = [IsAdminUser] | ||||||
|  |  | ||||||
|     @extend_schema(responses={200: AppSerializer(many=True)}) |     @extend_schema(responses={200: AppSerializer(many=True)}) | ||||||
|     def list(self, request: Request) -> Response: |     def list(self, request: Request) -> Response: | ||||||
| @ -36,7 +35,7 @@ class AppsViewSet(ViewSet): | |||||||
| class ModelViewSet(ViewSet): | class ModelViewSet(ViewSet): | ||||||
|     """Read-only view list all installed models""" |     """Read-only view list all installed models""" | ||||||
|  |  | ||||||
|     permission_classes = [IsAuthenticated] |     permission_classes = [IsAdminUser] | ||||||
|  |  | ||||||
|     @extend_schema(responses={200: AppSerializer(many=True)}) |     @extend_schema(responses={200: AppSerializer(many=True)}) | ||||||
|     def list(self, request: Request) -> Response: |     def list(self, request: Request) -> Response: | ||||||
|  | |||||||
| @ -1,12 +1,11 @@ | |||||||
| """authentik administration metrics""" | """authentik administration metrics""" | ||||||
|  |  | ||||||
| from datetime import timedelta | from datetime import timedelta | ||||||
|  |  | ||||||
| from django.db.models.functions import ExtractHour | from django.db.models.functions import ExtractHour | ||||||
| from drf_spectacular.utils import extend_schema, extend_schema_field | from drf_spectacular.utils import extend_schema, extend_schema_field | ||||||
| from guardian.shortcuts import get_objects_for_user | from guardian.shortcuts import get_objects_for_user | ||||||
| from rest_framework.fields import IntegerField, SerializerMethodField | from rest_framework.fields import IntegerField, SerializerMethodField | ||||||
| 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 | ||||||
| @ -69,7 +68,7 @@ class LoginMetricsSerializer(PassiveSerializer): | |||||||
| class AdministrationMetricsViewSet(APIView): | class AdministrationMetricsViewSet(APIView): | ||||||
|     """Login Metrics per 1h""" |     """Login Metrics per 1h""" | ||||||
|  |  | ||||||
|     permission_classes = [IsAuthenticated] |     permission_classes = [IsAdminUser] | ||||||
|  |  | ||||||
|     @extend_schema(responses={200: LoginMetricsSerializer(many=False)}) |     @extend_schema(responses={200: LoginMetricsSerializer(many=False)}) | ||||||
|     def get(self, request: Request) -> Response: |     def get(self, request: Request) -> Response: | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """authentik administration overview""" | """authentik administration overview""" | ||||||
|  |  | ||||||
| import platform | import platform | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
| from sys import version as python_version | from sys import version as python_version | ||||||
| @ -9,16 +8,15 @@ from django.utils.timezone import now | |||||||
| from drf_spectacular.utils import extend_schema | from drf_spectacular.utils import extend_schema | ||||||
| from gunicorn import version_info as gunicorn_version | from gunicorn import version_info as gunicorn_version | ||||||
| from rest_framework.fields import SerializerMethodField | from rest_framework.fields import SerializerMethodField | ||||||
|  | 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 | ||||||
|  |  | ||||||
| from authentik.core.api.utils import PassiveSerializer | from authentik.core.api.utils import PassiveSerializer | ||||||
| from authentik.lib.config import CONFIG |  | ||||||
| from authentik.lib.utils.reflection import get_env | from authentik.lib.utils.reflection import get_env | ||||||
| from authentik.outposts.apps import MANAGED_OUTPOST | from authentik.outposts.apps import MANAGED_OUTPOST | ||||||
| from authentik.outposts.models import Outpost | from authentik.outposts.models import Outpost | ||||||
| from authentik.rbac.permissions import HasPermission |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class RuntimeDict(TypedDict): | class RuntimeDict(TypedDict): | ||||||
| @ -32,16 +30,15 @@ class RuntimeDict(TypedDict): | |||||||
|     uname: str |     uname: str | ||||||
|  |  | ||||||
|  |  | ||||||
| class SystemInfoSerializer(PassiveSerializer): | class SystemSerializer(PassiveSerializer): | ||||||
|     """Get system information.""" |     """Get system information.""" | ||||||
|  |  | ||||||
|     http_headers = SerializerMethodField() |     http_headers = SerializerMethodField() | ||||||
|     http_host = SerializerMethodField() |     http_host = SerializerMethodField() | ||||||
|     http_is_secure = SerializerMethodField() |     http_is_secure = SerializerMethodField() | ||||||
|     runtime = SerializerMethodField() |     runtime = SerializerMethodField() | ||||||
|     brand = SerializerMethodField() |     tenant = SerializerMethodField() | ||||||
|     server_time = SerializerMethodField() |     server_time = SerializerMethodField() | ||||||
|     embedded_outpost_disabled = SerializerMethodField() |  | ||||||
|     embedded_outpost_host = SerializerMethodField() |     embedded_outpost_host = SerializerMethodField() | ||||||
|  |  | ||||||
|     def get_http_headers(self, request: Request) -> dict[str, str]: |     def get_http_headers(self, request: Request) -> dict[str, str]: | ||||||
| @ -72,18 +69,14 @@ class SystemInfoSerializer(PassiveSerializer): | |||||||
|             "uname": " ".join(platform.uname()), |             "uname": " ".join(platform.uname()), | ||||||
|         } |         } | ||||||
|  |  | ||||||
|     def get_brand(self, request: Request) -> str: |     def get_tenant(self, request: Request) -> str: | ||||||
|         """Currently active brand""" |         """Currently active tenant""" | ||||||
|         return str(request._request.brand) |         return str(request._request.tenant) | ||||||
|  |  | ||||||
|     def get_server_time(self, request: Request) -> datetime: |     def get_server_time(self, request: Request) -> datetime: | ||||||
|         """Current server time""" |         """Current server time""" | ||||||
|         return now() |         return now() | ||||||
|  |  | ||||||
|     def get_embedded_outpost_disabled(self, request: Request) -> bool: |  | ||||||
|         """Whether the embedded outpost is disabled""" |  | ||||||
|         return CONFIG.get_bool("outposts.disable_embedded_outpost", False) |  | ||||||
|  |  | ||||||
|     def get_embedded_outpost_host(self, request: Request) -> str: |     def get_embedded_outpost_host(self, request: Request) -> str: | ||||||
|         """Get the FQDN configured on the embedded outpost""" |         """Get the FQDN configured on the embedded outpost""" | ||||||
|         outposts = Outpost.objects.filter(managed=MANAGED_OUTPOST) |         outposts = Outpost.objects.filter(managed=MANAGED_OUTPOST) | ||||||
| @ -95,17 +88,17 @@ class SystemInfoSerializer(PassiveSerializer): | |||||||
| class SystemView(APIView): | class SystemView(APIView): | ||||||
|     """Get system information.""" |     """Get system information.""" | ||||||
|  |  | ||||||
|     permission_classes = [HasPermission("authentik_rbac.view_system_info")] |     permission_classes = [IsAdminUser] | ||||||
|     pagination_class = None |     pagination_class = None | ||||||
|     filter_backends = [] |     filter_backends = [] | ||||||
|     serializer_class = SystemInfoSerializer |     serializer_class = SystemSerializer | ||||||
|  |  | ||||||
|     @extend_schema(responses={200: SystemInfoSerializer(many=False)}) |     @extend_schema(responses={200: SystemSerializer(many=False)}) | ||||||
|     def get(self, request: Request) -> Response: |     def get(self, request: Request) -> Response: | ||||||
|         """Get system information.""" |         """Get system information.""" | ||||||
|         return Response(SystemInfoSerializer(request).data) |         return Response(SystemSerializer(request).data) | ||||||
|  |  | ||||||
|     @extend_schema(responses={200: SystemInfoSerializer(many=False)}) |     @extend_schema(responses={200: SystemSerializer(many=False)}) | ||||||
|     def post(self, request: Request) -> Response: |     def post(self, request: Request) -> Response: | ||||||
|         """Get system information.""" |         """Get system information.""" | ||||||
|         return Response(SystemInfoSerializer(request).data) |         return Response(SystemSerializer(request).data) | ||||||
|  | |||||||
							
								
								
									
										132
									
								
								authentik/admin/api/tasks.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								authentik/admin/api/tasks.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,132 @@ | |||||||
|  | """Tasks API""" | ||||||
|  | from importlib import import_module | ||||||
|  |  | ||||||
|  | from django.contrib import messages | ||||||
|  | from django.http.response import Http404 | ||||||
|  | from django.utils.translation import gettext_lazy as _ | ||||||
|  | from drf_spectacular.types import OpenApiTypes | ||||||
|  | from drf_spectacular.utils import OpenApiParameter, OpenApiResponse, extend_schema | ||||||
|  | from rest_framework.decorators import action | ||||||
|  | from rest_framework.fields import ( | ||||||
|  |     CharField, | ||||||
|  |     ChoiceField, | ||||||
|  |     DateTimeField, | ||||||
|  |     ListField, | ||||||
|  |     SerializerMethodField, | ||||||
|  | ) | ||||||
|  | from rest_framework.permissions import IsAdminUser | ||||||
|  | from rest_framework.request import Request | ||||||
|  | from rest_framework.response import Response | ||||||
|  | from rest_framework.viewsets import ViewSet | ||||||
|  | from structlog.stdlib import get_logger | ||||||
|  |  | ||||||
|  | from authentik.core.api.utils import PassiveSerializer | ||||||
|  | from authentik.events.monitored_tasks import TaskInfo, TaskResultStatus | ||||||
|  |  | ||||||
|  | LOGGER = get_logger() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TaskSerializer(PassiveSerializer): | ||||||
|  |     """Serialize TaskInfo and TaskResult""" | ||||||
|  |  | ||||||
|  |     task_name = CharField() | ||||||
|  |     task_description = CharField() | ||||||
|  |     task_finish_timestamp = DateTimeField(source="finish_time") | ||||||
|  |     task_duration = SerializerMethodField() | ||||||
|  |  | ||||||
|  |     status = ChoiceField( | ||||||
|  |         source="result.status.name", | ||||||
|  |         choices=[(x.name, x.name) for x in TaskResultStatus], | ||||||
|  |     ) | ||||||
|  |     messages = ListField(source="result.messages") | ||||||
|  |  | ||||||
|  |     def get_task_duration(self, instance: TaskInfo) -> int: | ||||||
|  |         """Get the duration a task took to run""" | ||||||
|  |         return max(instance.finish_timestamp - instance.start_timestamp, 0) | ||||||
|  |  | ||||||
|  |     def to_representation(self, instance: TaskInfo): | ||||||
|  |         """When a new version of authentik adds fields to TaskInfo, | ||||||
|  |         the API will fail with an AttributeError, as the classes | ||||||
|  |         are pickled in cache. In that case, just delete the info""" | ||||||
|  |         try: | ||||||
|  |             return super().to_representation(instance) | ||||||
|  |         # pylint: disable=broad-except | ||||||
|  |         except Exception:  # pragma: no cover | ||||||
|  |             if isinstance(self.instance, list): | ||||||
|  |                 for inst in self.instance: | ||||||
|  |                     inst.delete() | ||||||
|  |             else: | ||||||
|  |                 self.instance.delete() | ||||||
|  |             return {} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TaskViewSet(ViewSet): | ||||||
|  |     """Read-only view set that returns all background tasks""" | ||||||
|  |  | ||||||
|  |     permission_classes = [IsAdminUser] | ||||||
|  |     serializer_class = TaskSerializer | ||||||
|  |  | ||||||
|  |     @extend_schema( | ||||||
|  |         responses={ | ||||||
|  |             200: TaskSerializer(many=False), | ||||||
|  |             404: OpenApiResponse(description="Task not found"), | ||||||
|  |         }, | ||||||
|  |         parameters=[ | ||||||
|  |             OpenApiParameter( | ||||||
|  |                 "id", | ||||||
|  |                 type=OpenApiTypes.STR, | ||||||
|  |                 location=OpenApiParameter.PATH, | ||||||
|  |                 required=True, | ||||||
|  |             ), | ||||||
|  |         ], | ||||||
|  |     ) | ||||||
|  |     def retrieve(self, request: Request, pk=None) -> Response: | ||||||
|  |         """Get a single system task""" | ||||||
|  |         task = TaskInfo.by_name(pk) | ||||||
|  |         if not task: | ||||||
|  |             raise Http404 | ||||||
|  |         return Response(TaskSerializer(task, many=False).data) | ||||||
|  |  | ||||||
|  |     @extend_schema(responses={200: TaskSerializer(many=True)}) | ||||||
|  |     def list(self, request: Request) -> Response: | ||||||
|  |         """List system tasks""" | ||||||
|  |         tasks = sorted(TaskInfo.all().values(), key=lambda task: task.task_name) | ||||||
|  |         return Response(TaskSerializer(tasks, many=True).data) | ||||||
|  |  | ||||||
|  |     @extend_schema( | ||||||
|  |         request=OpenApiTypes.NONE, | ||||||
|  |         responses={ | ||||||
|  |             204: OpenApiResponse(description="Task retried successfully"), | ||||||
|  |             404: OpenApiResponse(description="Task not found"), | ||||||
|  |             500: OpenApiResponse(description="Failed to retry task"), | ||||||
|  |         }, | ||||||
|  |         parameters=[ | ||||||
|  |             OpenApiParameter( | ||||||
|  |                 "id", | ||||||
|  |                 type=OpenApiTypes.STR, | ||||||
|  |                 location=OpenApiParameter.PATH, | ||||||
|  |                 required=True, | ||||||
|  |             ), | ||||||
|  |         ], | ||||||
|  |     ) | ||||||
|  |     @action(detail=True, methods=["post"]) | ||||||
|  |     def retry(self, request: Request, pk=None) -> Response: | ||||||
|  |         """Retry task""" | ||||||
|  |         task = TaskInfo.by_name(pk) | ||||||
|  |         if not task: | ||||||
|  |             raise Http404 | ||||||
|  |         try: | ||||||
|  |             task_module = import_module(task.task_call_module) | ||||||
|  |             task_func = getattr(task_module, task.task_call_func) | ||||||
|  |             LOGGER.debug("Running task", task=task_func) | ||||||
|  |             task_func.delay(*task.task_call_args, **task.task_call_kwargs) | ||||||
|  |             messages.success( | ||||||
|  |                 self.request, | ||||||
|  |                 _("Successfully re-scheduled Task %(name)s!" % {"name": task.task_name}), | ||||||
|  |             ) | ||||||
|  |             return Response(status=204) | ||||||
|  |         except (ImportError, AttributeError):  # pragma: no cover | ||||||
|  |             LOGGER.warning("Failed to run task, remove state", task=task) | ||||||
|  |             # if we get an import error, the module path has probably changed | ||||||
|  |             task.delete() | ||||||
|  |             return Response(status=500) | ||||||
| @ -1,5 +1,4 @@ | |||||||
| """authentik administration overview""" | """authentik administration overview""" | ||||||
|  |  | ||||||
| from django.core.cache import cache | from django.core.cache import cache | ||||||
| from drf_spectacular.utils import extend_schema | from drf_spectacular.utils import extend_schema | ||||||
| from packaging.version import parse | from packaging.version import parse | ||||||
|  | |||||||
| @ -1,20 +1,19 @@ | |||||||
| """authentik administration overview""" | """authentik administration overview""" | ||||||
|  |  | ||||||
| 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 rest_framework.fields import IntegerField | from rest_framework.fields import IntegerField | ||||||
|  | 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 | ||||||
|  |  | ||||||
| from authentik.rbac.permissions import HasPermission |  | ||||||
| from authentik.root.celery import CELERY_APP | from authentik.root.celery import CELERY_APP | ||||||
|  |  | ||||||
|  |  | ||||||
| class WorkerView(APIView): | class WorkerView(APIView): | ||||||
|     """Get currently connected worker count.""" |     """Get currently connected worker count.""" | ||||||
|  |  | ||||||
|     permission_classes = [HasPermission("authentik_rbac.view_system_info")] |     permission_classes = [IsAdminUser] | ||||||
|  |  | ||||||
|     @extend_schema(responses=inline_serializer("Workers", fields={"count": IntegerField()})) |     @extend_schema(responses=inline_serializer("Workers", fields={"count": IntegerField()})) | ||||||
|     def get(self, request: Request) -> Response: |     def get(self, request: Request) -> Response: | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """authentik admin app config""" | """authentik admin app config""" | ||||||
|  |  | ||||||
| from prometheus_client import Gauge, Info | from prometheus_client import Gauge, Info | ||||||
|  |  | ||||||
| from authentik.blueprints.apps import ManagedAppConfig | from authentik.blueprints.apps import ManagedAppConfig | ||||||
| @ -15,3 +14,7 @@ class AuthentikAdminConfig(ManagedAppConfig): | |||||||
|     label = "authentik_admin" |     label = "authentik_admin" | ||||||
|     verbose_name = "authentik Admin" |     verbose_name = "authentik Admin" | ||||||
|     default = True |     default = True | ||||||
|  |  | ||||||
|  |     def reconcile_load_admin_signals(self): | ||||||
|  |         """Load admin signals""" | ||||||
|  |         self.import_module("authentik.admin.signals") | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """authentik admin settings""" | """authentik admin settings""" | ||||||
|  |  | ||||||
| from celery.schedules import crontab | from celery.schedules import crontab | ||||||
|  |  | ||||||
| from authentik.lib.utils.time import fqdn_rand | from authentik.lib.utils.time import fqdn_rand | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| """admin signals""" | """admin signals""" | ||||||
|  |  | ||||||
| from django.dispatch import receiver | from django.dispatch import receiver | ||||||
|  |  | ||||||
|  | from authentik.admin.api.tasks import TaskInfo | ||||||
| from authentik.admin.apps import GAUGE_WORKERS | 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 | ||||||
| @ -12,3 +12,10 @@ def monitoring_set_workers(sender, **kwargs): | |||||||
|     """Set worker gauge""" |     """Set worker gauge""" | ||||||
|     count = len(CELERY_APP.control.ping(timeout=0.5)) |     count = len(CELERY_APP.control.ping(timeout=0.5)) | ||||||
|     GAUGE_WORKERS.set(count) |     GAUGE_WORKERS.set(count) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @receiver(monitoring_set) | ||||||
|  | def monitoring_set_tasks(sender, **kwargs): | ||||||
|  |     """Set task gauges""" | ||||||
|  |     for task in TaskInfo.all().values(): | ||||||
|  |         task.update_metrics() | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """authentik admin tasks""" | """authentik admin tasks""" | ||||||
|  |  | ||||||
| import re | import re | ||||||
|  |  | ||||||
| from django.core.cache import cache | from django.core.cache import cache | ||||||
| @ -12,7 +11,12 @@ from structlog.stdlib import get_logger | |||||||
| from authentik import __version__, get_build_hash | from authentik import __version__, get_build_hash | ||||||
| from authentik.admin.apps import PROM_INFO | from authentik.admin.apps import PROM_INFO | ||||||
| from authentik.events.models import Event, EventAction, Notification | from authentik.events.models import Event, EventAction, Notification | ||||||
| from authentik.events.system_tasks import SystemTask, TaskStatus, prefill_task | from authentik.events.monitored_tasks import ( | ||||||
|  |     MonitoredTask, | ||||||
|  |     TaskResult, | ||||||
|  |     TaskResultStatus, | ||||||
|  |     prefill_task, | ||||||
|  | ) | ||||||
| from authentik.lib.config import CONFIG | from authentik.lib.config import CONFIG | ||||||
| from authentik.lib.utils.http import get_http_session | from authentik.lib.utils.http import get_http_session | ||||||
| from authentik.root.celery import CELERY_APP | from authentik.root.celery import CELERY_APP | ||||||
| @ -50,13 +54,13 @@ def clear_update_notifications(): | |||||||
|             notification.delete() |             notification.delete() | ||||||
|  |  | ||||||
|  |  | ||||||
| @CELERY_APP.task(bind=True, base=SystemTask) | @CELERY_APP.task(bind=True, base=MonitoredTask) | ||||||
| @prefill_task | @prefill_task | ||||||
| def update_latest_version(self: SystemTask): | def update_latest_version(self: MonitoredTask): | ||||||
|     """Update latest version info""" |     """Update latest version info""" | ||||||
|     if CONFIG.get_bool("disable_update_check"): |     if CONFIG.get_bool("disable_update_check"): | ||||||
|         cache.set(VERSION_CACHE_KEY, "0.0.0", VERSION_CACHE_TIMEOUT) |         cache.set(VERSION_CACHE_KEY, "0.0.0", VERSION_CACHE_TIMEOUT) | ||||||
|         self.set_status(TaskStatus.WARNING, "Version check disabled.") |         self.set_status(TaskResult(TaskResultStatus.WARNING, messages=["Version check disabled."])) | ||||||
|         return |         return | ||||||
|     try: |     try: | ||||||
|         response = get_http_session().get( |         response = get_http_session().get( | ||||||
| @ -66,7 +70,9 @@ def update_latest_version(self: SystemTask): | |||||||
|         data = response.json() |         data = response.json() | ||||||
|         upstream_version = data.get("stable", {}).get("version") |         upstream_version = data.get("stable", {}).get("version") | ||||||
|         cache.set(VERSION_CACHE_KEY, upstream_version, VERSION_CACHE_TIMEOUT) |         cache.set(VERSION_CACHE_KEY, upstream_version, VERSION_CACHE_TIMEOUT) | ||||||
|         self.set_status(TaskStatus.SUCCESSFUL, "Successfully updated latest Version") |         self.set_status( | ||||||
|  |             TaskResult(TaskResultStatus.SUCCESSFUL, ["Successfully updated latest Version"]) | ||||||
|  |         ) | ||||||
|         _set_prom_info() |         _set_prom_info() | ||||||
|         # Check if upstream version is newer than what we're running, |         # Check if upstream version is newer than what we're running, | ||||||
|         # and if no event exists yet, create one. |         # and if no event exists yet, create one. | ||||||
| @ -83,7 +89,7 @@ def update_latest_version(self: SystemTask): | |||||||
|             Event.new(EventAction.UPDATE_AVAILABLE, **event_dict).save() |             Event.new(EventAction.UPDATE_AVAILABLE, **event_dict).save() | ||||||
|     except (RequestException, IndexError) as exc: |     except (RequestException, IndexError) as exc: | ||||||
|         cache.set(VERSION_CACHE_KEY, "0.0.0", VERSION_CACHE_TIMEOUT) |         cache.set(VERSION_CACHE_KEY, "0.0.0", VERSION_CACHE_TIMEOUT) | ||||||
|         self.set_error(exc) |         self.set_status(TaskResult(TaskResultStatus.ERROR).with_error(exc)) | ||||||
|  |  | ||||||
|  |  | ||||||
| _set_prom_info() | _set_prom_info() | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """test admin api""" | """test admin api""" | ||||||
|  |  | ||||||
| from json import loads | from json import loads | ||||||
|  |  | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
| @ -8,6 +7,8 @@ from django.urls import reverse | |||||||
| from authentik import __version__ | from authentik import __version__ | ||||||
| from authentik.blueprints.tests import reconcile_app | from authentik.blueprints.tests import reconcile_app | ||||||
| from authentik.core.models import Group, User | from authentik.core.models import Group, User | ||||||
|  | from authentik.core.tasks import clean_expired_models | ||||||
|  | from authentik.events.monitored_tasks import TaskResultStatus | ||||||
| from authentik.lib.generators import generate_id | from authentik.lib.generators import generate_id | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -22,6 +23,53 @@ class TestAdminAPI(TestCase): | |||||||
|         self.group.save() |         self.group.save() | ||||||
|         self.client.force_login(self.user) |         self.client.force_login(self.user) | ||||||
|  |  | ||||||
|  |     def test_tasks(self): | ||||||
|  |         """Test Task API""" | ||||||
|  |         clean_expired_models.delay() | ||||||
|  |         response = self.client.get(reverse("authentik_api:admin_system_tasks-list")) | ||||||
|  |         self.assertEqual(response.status_code, 200) | ||||||
|  |         body = loads(response.content) | ||||||
|  |         self.assertTrue(any(task["task_name"] == "clean_expired_models" for task in body)) | ||||||
|  |  | ||||||
|  |     def test_tasks_single(self): | ||||||
|  |         """Test Task API (read single)""" | ||||||
|  |         clean_expired_models.delay() | ||||||
|  |         response = self.client.get( | ||||||
|  |             reverse( | ||||||
|  |                 "authentik_api:admin_system_tasks-detail", | ||||||
|  |                 kwargs={"pk": "clean_expired_models"}, | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         self.assertEqual(response.status_code, 200) | ||||||
|  |         body = loads(response.content) | ||||||
|  |         self.assertEqual(body["status"], TaskResultStatus.SUCCESSFUL.name) | ||||||
|  |         self.assertEqual(body["task_name"], "clean_expired_models") | ||||||
|  |         response = self.client.get( | ||||||
|  |             reverse("authentik_api:admin_system_tasks-detail", kwargs={"pk": "qwerqwer"}) | ||||||
|  |         ) | ||||||
|  |         self.assertEqual(response.status_code, 404) | ||||||
|  |  | ||||||
|  |     def test_tasks_retry(self): | ||||||
|  |         """Test Task API (retry)""" | ||||||
|  |         clean_expired_models.delay() | ||||||
|  |         response = self.client.post( | ||||||
|  |             reverse( | ||||||
|  |                 "authentik_api:admin_system_tasks-retry", | ||||||
|  |                 kwargs={"pk": "clean_expired_models"}, | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         self.assertEqual(response.status_code, 204) | ||||||
|  |  | ||||||
|  |     def test_tasks_retry_404(self): | ||||||
|  |         """Test Task API (retry, 404)""" | ||||||
|  |         response = self.client.post( | ||||||
|  |             reverse( | ||||||
|  |                 "authentik_api:admin_system_tasks-retry", | ||||||
|  |                 kwargs={"pk": "qwerqewrqrqewrqewr"}, | ||||||
|  |             ) | ||||||
|  |         ) | ||||||
|  |         self.assertEqual(response.status_code, 404) | ||||||
|  |  | ||||||
|     def test_version(self): |     def test_version(self): | ||||||
|         """Test Version API""" |         """Test Version API""" | ||||||
|         response = self.client.get(reverse("authentik_api:admin_version")) |         response = self.client.get(reverse("authentik_api:admin_version")) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """test admin tasks""" | """test admin tasks""" | ||||||
|  |  | ||||||
| from django.core.cache import cache | from django.core.cache import cache | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
| from requests_mock import Mocker | from requests_mock import Mocker | ||||||
|  | |||||||
| @ -1,14 +1,15 @@ | |||||||
| """API URLs""" | """API URLs""" | ||||||
|  |  | ||||||
| from django.urls import path | from django.urls import path | ||||||
|  |  | ||||||
| from authentik.admin.api.meta import AppsViewSet, ModelViewSet | from authentik.admin.api.meta import AppsViewSet, ModelViewSet | ||||||
| from authentik.admin.api.metrics import AdministrationMetricsViewSet | from authentik.admin.api.metrics import AdministrationMetricsViewSet | ||||||
| from authentik.admin.api.system import SystemView | from authentik.admin.api.system import SystemView | ||||||
|  | from authentik.admin.api.tasks import TaskViewSet | ||||||
| from authentik.admin.api.version import VersionView | from authentik.admin.api.version import VersionView | ||||||
| from authentik.admin.api.workers import WorkerView | from authentik.admin.api.workers import WorkerView | ||||||
|  |  | ||||||
| api_urlpatterns = [ | api_urlpatterns = [ | ||||||
|  |     ("admin/system_tasks", TaskViewSet, "admin_system_tasks"), | ||||||
|     ("admin/apps", AppsViewSet, "apps"), |     ("admin/apps", AppsViewSet, "apps"), | ||||||
|     ("admin/models", ModelViewSet, "models"), |     ("admin/models", ModelViewSet, "models"), | ||||||
|     path( |     path( | ||||||
|  | |||||||
| @ -18,7 +18,7 @@ class AuthentikAPIConfig(AppConfig): | |||||||
|  |  | ||||||
|         # Class is defined here as it needs to be created early enough that drf-spectacular will |         # Class is defined here as it needs to be created early enough that drf-spectacular will | ||||||
|         # find it, but also won't cause any import issues |         # find it, but also won't cause any import issues | ||||||
|  |         # pylint: disable=unused-variable | ||||||
|         class TokenSchema(OpenApiAuthenticationExtension): |         class TokenSchema(OpenApiAuthenticationExtension): | ||||||
|             """Auth schema""" |             """Auth schema""" | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,7 +1,6 @@ | |||||||
| """API Authentication""" | """API Authentication""" | ||||||
|  |  | ||||||
| from hmac import compare_digest | from hmac import compare_digest | ||||||
| from typing import Any | from typing import Any, Optional | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from rest_framework.authentication import BaseAuthentication, get_authorization_header | from rest_framework.authentication import BaseAuthentication, get_authorization_header | ||||||
| @ -17,7 +16,7 @@ from authentik.providers.oauth2.constants import SCOPE_AUTHENTIK_API | |||||||
| LOGGER = get_logger() | LOGGER = get_logger() | ||||||
|  |  | ||||||
|  |  | ||||||
| def validate_auth(header: bytes) -> str | None: | def validate_auth(header: bytes) -> Optional[str]: | ||||||
|     """Validate that the header is in a correct format, |     """Validate that the header is in a correct format, | ||||||
|     returns type and credentials""" |     returns type and credentials""" | ||||||
|     auth_credentials = header.decode().strip() |     auth_credentials = header.decode().strip() | ||||||
| @ -32,7 +31,7 @@ def validate_auth(header: bytes) -> str | None: | |||||||
|     return auth_credentials |     return auth_credentials | ||||||
|  |  | ||||||
|  |  | ||||||
| def bearer_auth(raw_header: bytes) -> User | None: | def bearer_auth(raw_header: bytes) -> Optional[User]: | ||||||
|     """raw_header in the Format of `Bearer ....`""" |     """raw_header in the Format of `Bearer ....`""" | ||||||
|     user = auth_user_lookup(raw_header) |     user = auth_user_lookup(raw_header) | ||||||
|     if not user: |     if not user: | ||||||
| @ -42,7 +41,7 @@ def bearer_auth(raw_header: bytes) -> User | None: | |||||||
|     return user |     return user | ||||||
|  |  | ||||||
|  |  | ||||||
| def auth_user_lookup(raw_header: bytes) -> User | None: | def auth_user_lookup(raw_header: bytes) -> Optional[User]: | ||||||
|     """raw_header in the Format of `Bearer ....`""" |     """raw_header in the Format of `Bearer ....`""" | ||||||
|     from authentik.providers.oauth2.models import AccessToken |     from authentik.providers.oauth2.models import AccessToken | ||||||
|  |  | ||||||
| @ -75,7 +74,7 @@ def auth_user_lookup(raw_header: bytes) -> User | None: | |||||||
|     raise AuthenticationFailed("Token invalid/expired") |     raise AuthenticationFailed("Token invalid/expired") | ||||||
|  |  | ||||||
|  |  | ||||||
| def token_secret_key(value: str) -> User | None: | def token_secret_key(value: str) -> Optional[User]: | ||||||
|     """Check if the token is the secret key |     """Check if the token is the secret key | ||||||
|     and return the service account for the managed outpost""" |     and return the service account for the managed outpost""" | ||||||
|     from authentik.outposts.apps import MANAGED_OUTPOST |     from authentik.outposts.apps import MANAGED_OUTPOST | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """API Authorization""" | """API Authorization""" | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| from django.db.models import Model | from django.db.models import Model | ||||||
| from django.db.models.query import QuerySet | from django.db.models.query import QuerySet | ||||||
| @ -8,9 +7,9 @@ from rest_framework.authentication import get_authorization_header | |||||||
| from rest_framework.filters import BaseFilterBackend | from rest_framework.filters import BaseFilterBackend | ||||||
| from rest_framework.permissions import BasePermission | from rest_framework.permissions import BasePermission | ||||||
| from rest_framework.request import Request | from rest_framework.request import Request | ||||||
|  | from rest_framework_guardian.filters import ObjectPermissionsFilter | ||||||
|  |  | ||||||
| from authentik.api.authentication import validate_auth | from authentik.api.authentication import validate_auth | ||||||
| from authentik.rbac.filters import ObjectFilter |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class OwnerFilter(BaseFilterBackend): | class OwnerFilter(BaseFilterBackend): | ||||||
| @ -27,14 +26,14 @@ class OwnerFilter(BaseFilterBackend): | |||||||
| class SecretKeyFilter(DjangoFilterBackend): | class SecretKeyFilter(DjangoFilterBackend): | ||||||
|     """Allow access to all objects when authenticated with secret key as token. |     """Allow access to all objects when authenticated with secret key as token. | ||||||
|  |  | ||||||
|     Replaces both DjangoFilterBackend and ObjectFilter""" |     Replaces both DjangoFilterBackend and ObjectPermissionsFilter""" | ||||||
|  |  | ||||||
|     def filter_queryset(self, request: Request, queryset: QuerySet, view) -> QuerySet: |     def filter_queryset(self, request: Request, queryset: QuerySet, view) -> QuerySet: | ||||||
|         auth_header = get_authorization_header(request) |         auth_header = get_authorization_header(request) | ||||||
|         token = validate_auth(auth_header) |         token = validate_auth(auth_header) | ||||||
|         if token and token == settings.SECRET_KEY: |         if token and token == settings.SECRET_KEY: | ||||||
|             return queryset |             return queryset | ||||||
|         queryset = ObjectFilter().filter_queryset(request, queryset, view) |         queryset = ObjectPermissionsFilter().filter_queryset(request, queryset, view) | ||||||
|         return super().filter_queryset(request, queryset, view) |         return super().filter_queryset(request, queryset, view) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										35
									
								
								authentik/api/decorators.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								authentik/api/decorators.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | |||||||
|  | """API Decorators""" | ||||||
|  | from functools import wraps | ||||||
|  | from typing import Callable, Optional | ||||||
|  |  | ||||||
|  | from rest_framework.request import Request | ||||||
|  | from rest_framework.response import Response | ||||||
|  | from rest_framework.viewsets import ModelViewSet | ||||||
|  | from structlog.stdlib import get_logger | ||||||
|  |  | ||||||
|  | LOGGER = get_logger() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def permission_required(perm: Optional[str] = None, other_perms: Optional[list[str]] = None): | ||||||
|  |     """Check permissions for a single custom action""" | ||||||
|  |  | ||||||
|  |     def wrapper_outter(func: Callable): | ||||||
|  |         """Check permissions for a single custom action""" | ||||||
|  |  | ||||||
|  |         @wraps(func) | ||||||
|  |         def wrapper(self: ModelViewSet, request: Request, *args, **kwargs) -> Response: | ||||||
|  |             if perm: | ||||||
|  |                 obj = self.get_object() | ||||||
|  |                 if not request.user.has_perm(perm, obj): | ||||||
|  |                     LOGGER.debug("denying access for object", user=request.user, perm=perm, obj=obj) | ||||||
|  |                     return self.permission_denied(request) | ||||||
|  |             if other_perms: | ||||||
|  |                 for other_perm in other_perms: | ||||||
|  |                     if not request.user.has_perm(other_perm): | ||||||
|  |                         LOGGER.debug("denying access for other", user=request.user, perm=perm) | ||||||
|  |                         return self.permission_denied(request) | ||||||
|  |             return func(self, request, *args, **kwargs) | ||||||
|  |  | ||||||
|  |         return wrapper | ||||||
|  |  | ||||||
|  |     return wrapper_outter | ||||||
| @ -1,5 +1,4 @@ | |||||||
| """Pagination which includes total pages and current page""" | """Pagination which includes total pages and current page""" | ||||||
|  |  | ||||||
| from rest_framework import pagination | from rest_framework import pagination | ||||||
| from rest_framework.response import Response | from rest_framework.response import Response | ||||||
|  |  | ||||||
| @ -78,10 +77,3 @@ class Pagination(pagination.PageNumberPagination): | |||||||
|             }, |             }, | ||||||
|             "required": ["pagination", "results"], |             "required": ["pagination", "results"], | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
| class SmallerPagination(Pagination): |  | ||||||
|     """Smaller pagination for objects which might require a lot of queries |  | ||||||
|     to retrieve all data for.""" |  | ||||||
|  |  | ||||||
|     max_page_size = 10 |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Error Response schema, from https://github.com/axnsan12/drf-yasg/issues/224""" | """Error Response schema, from https://github.com/axnsan12/drf-yasg/issues/224""" | ||||||
|  |  | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from drf_spectacular.generators import SchemaGenerator | from drf_spectacular.generators import SchemaGenerator | ||||||
| from drf_spectacular.plumbing import ( | from drf_spectacular.plumbing import ( | ||||||
|  | |||||||
| @ -3,7 +3,7 @@ | |||||||
| {% load static %} | {% load static %} | ||||||
|  |  | ||||||
| {% block title %} | {% block title %} | ||||||
| API Browser - {{ brand.branding_title }} | API Browser - {{ tenant.branding_title }} | ||||||
| {% endblock %} | {% endblock %} | ||||||
|  |  | ||||||
| {% block head %} | {% block head %} | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test API Authentication""" | """Test API Authentication""" | ||||||
|  |  | ||||||
| import json | import json | ||||||
| from base64 import b64encode | from base64 import b64encode | ||||||
|  |  | ||||||
| @ -13,8 +12,6 @@ from authentik.blueprints.tests import reconcile_app | |||||||
| from authentik.core.models import Token, TokenIntents, User, UserTypes | from authentik.core.models import Token, TokenIntents, User, UserTypes | ||||||
| from authentik.core.tests.utils import create_test_admin_user, create_test_flow | 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.outposts.apps import MANAGED_OUTPOST |  | ||||||
| from authentik.outposts.models import Outpost |  | ||||||
| from authentik.providers.oauth2.constants import SCOPE_AUTHENTIK_API | from authentik.providers.oauth2.constants import SCOPE_AUTHENTIK_API | ||||||
| from authentik.providers.oauth2.models import AccessToken, OAuth2Provider | from authentik.providers.oauth2.models import AccessToken, OAuth2Provider | ||||||
|  |  | ||||||
| @ -25,17 +22,17 @@ class TestAPIAuth(TestCase): | |||||||
|     def test_invalid_type(self): |     def test_invalid_type(self): | ||||||
|         """Test invalid type""" |         """Test invalid type""" | ||||||
|         with self.assertRaises(AuthenticationFailed): |         with self.assertRaises(AuthenticationFailed): | ||||||
|             bearer_auth(b"foo bar") |             bearer_auth("foo bar".encode()) | ||||||
|  |  | ||||||
|     def test_invalid_empty(self): |     def test_invalid_empty(self): | ||||||
|         """Test invalid type""" |         """Test invalid type""" | ||||||
|         self.assertIsNone(bearer_auth(b"Bearer ")) |         self.assertIsNone(bearer_auth("Bearer ".encode())) | ||||||
|         self.assertIsNone(bearer_auth(b"")) |         self.assertIsNone(bearer_auth("".encode())) | ||||||
|  |  | ||||||
|     def test_invalid_no_token(self): |     def test_invalid_no_token(self): | ||||||
|         """Test invalid with no token""" |         """Test invalid with no token""" | ||||||
|         with self.assertRaises(AuthenticationFailed): |         with self.assertRaises(AuthenticationFailed): | ||||||
|             auth = b64encode(b":abc").decode() |             auth = b64encode(":abc".encode()).decode() | ||||||
|             self.assertIsNone(bearer_auth(f"Basic :{auth}".encode())) |             self.assertIsNone(bearer_auth(f"Basic :{auth}".encode())) | ||||||
|  |  | ||||||
|     def test_bearer_valid(self): |     def test_bearer_valid(self): | ||||||
| @ -52,12 +49,8 @@ class TestAPIAuth(TestCase): | |||||||
|         with self.assertRaises(AuthenticationFailed): |         with self.assertRaises(AuthenticationFailed): | ||||||
|             bearer_auth(f"Bearer {token.key}".encode()) |             bearer_auth(f"Bearer {token.key}".encode()) | ||||||
|  |  | ||||||
|     @reconcile_app("authentik_outposts") |     def test_managed_outpost(self): | ||||||
|     def test_managed_outpost_fail(self): |  | ||||||
|         """Test managed outpost""" |         """Test managed outpost""" | ||||||
|         outpost = Outpost.objects.filter(managed=MANAGED_OUTPOST).first() |  | ||||||
|         outpost.user.delete() |  | ||||||
|         outpost.delete() |  | ||||||
|         with self.assertRaises(AuthenticationFailed): |         with self.assertRaises(AuthenticationFailed): | ||||||
|             bearer_auth(f"Bearer {settings.SECRET_KEY}".encode()) |             bearer_auth(f"Bearer {settings.SECRET_KEY}".encode()) | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test config API""" | """Test config API""" | ||||||
|  |  | ||||||
| from json import loads | from json import loads | ||||||
|  |  | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
|  | |||||||
							
								
								
									
										34
									
								
								authentik/api/tests/test_decorators.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								authentik/api/tests/test_decorators.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | """test decorators api""" | ||||||
|  | from django.urls import reverse | ||||||
|  | from guardian.shortcuts import assign_perm | ||||||
|  | from rest_framework.test import APITestCase | ||||||
|  |  | ||||||
|  | from authentik.core.models import Application, User | ||||||
|  | from authentik.lib.generators import generate_id | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestAPIDecorators(APITestCase): | ||||||
|  |     """test decorators api""" | ||||||
|  |  | ||||||
|  |     def setUp(self) -> None: | ||||||
|  |         super().setUp() | ||||||
|  |         self.user = User.objects.create(username="test-user") | ||||||
|  |  | ||||||
|  |     def test_obj_perm_denied(self): | ||||||
|  |         """Test object perm denied""" | ||||||
|  |         self.client.force_login(self.user) | ||||||
|  |         app = Application.objects.create(name=generate_id(), slug=generate_id()) | ||||||
|  |         response = self.client.get( | ||||||
|  |             reverse("authentik_api:application-metrics", kwargs={"slug": app.slug}) | ||||||
|  |         ) | ||||||
|  |         self.assertEqual(response.status_code, 403) | ||||||
|  |  | ||||||
|  |     def test_other_perm_denied(self): | ||||||
|  |         """Test other perm denied""" | ||||||
|  |         self.client.force_login(self.user) | ||||||
|  |         app = Application.objects.create(name=generate_id(), slug=generate_id()) | ||||||
|  |         assign_perm("authentik_core.view_application", self.user, app) | ||||||
|  |         response = self.client.get( | ||||||
|  |             reverse("authentik_api:application-metrics", kwargs={"slug": app.slug}) | ||||||
|  |         ) | ||||||
|  |         self.assertEqual(response.status_code, 403) | ||||||
| @ -1,5 +1,4 @@ | |||||||
| """Schema generation tests""" | """Schema generation tests""" | ||||||
|  |  | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
| from rest_framework.test import APITestCase | from rest_framework.test import APITestCase | ||||||
| from yaml import safe_load | from yaml import safe_load | ||||||
|  | |||||||
| @ -1,6 +1,5 @@ | |||||||
| """authentik API Modelviewset tests""" | """authentik API Modelviewset tests""" | ||||||
|  | from typing import Callable | ||||||
| from collections.abc import Callable |  | ||||||
|  |  | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
| from rest_framework.viewsets import ModelViewSet, ReadOnlyModelViewSet | from rest_framework.viewsets import ModelViewSet, ReadOnlyModelViewSet | ||||||
| @ -17,7 +16,6 @@ def viewset_tester_factory(test_viewset: type[ModelViewSet]) -> Callable: | |||||||
|  |  | ||||||
|     def tester(self: TestModelViewSets): |     def tester(self: TestModelViewSets): | ||||||
|         self.assertIsNotNone(getattr(test_viewset, "search_fields", None)) |         self.assertIsNotNone(getattr(test_viewset, "search_fields", None)) | ||||||
|         self.assertIsNotNone(getattr(test_viewset, "ordering", None)) |  | ||||||
|         filterset_class = getattr(test_viewset, "filterset_class", None) |         filterset_class = getattr(test_viewset, "filterset_class", None) | ||||||
|         if not filterset_class: |         if not filterset_class: | ||||||
|             self.assertIsNotNone(getattr(test_viewset, "filterset_fields", None)) |             self.assertIsNotNone(getattr(test_viewset, "filterset_fields", None)) | ||||||
| @ -26,6 +24,6 @@ def viewset_tester_factory(test_viewset: type[ModelViewSet]) -> Callable: | |||||||
|  |  | ||||||
|  |  | ||||||
| for _, viewset, _ in router.registry: | for _, viewset, _ in router.registry: | ||||||
|     if not issubclass(viewset, ModelViewSet | ReadOnlyModelViewSet): |     if not issubclass(viewset, (ModelViewSet, ReadOnlyModelViewSet)): | ||||||
|         continue |         continue | ||||||
|     setattr(TestModelViewSets, f"test_viewset_{viewset.__name__}", viewset_tester_factory(viewset)) |     setattr(TestModelViewSets, f"test_viewset_{viewset.__name__}", viewset_tester_factory(viewset)) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """authentik api urls""" | """authentik api urls""" | ||||||
|  |  | ||||||
| from django.urls import include, path | from django.urls import include, path | ||||||
|  |  | ||||||
| from authentik.api.v3.urls import urlpatterns as v3_urls | from authentik.api.v3.urls import urlpatterns as v3_urls | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """core Configs API""" | """core Configs API""" | ||||||
|  |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  |  | ||||||
| from django.conf import settings | from django.conf import settings | ||||||
| @ -20,7 +19,7 @@ from rest_framework.response import Response | |||||||
| from rest_framework.views import APIView | from rest_framework.views import APIView | ||||||
|  |  | ||||||
| from authentik.core.api.utils import PassiveSerializer | from authentik.core.api.utils import PassiveSerializer | ||||||
| from authentik.events.context_processors.base import get_context_processors | from authentik.events.geo import GEOIP_READER | ||||||
| from authentik.lib.config import CONFIG | from authentik.lib.config import CONFIG | ||||||
|  |  | ||||||
| capabilities = Signal() | capabilities = Signal() | ||||||
| @ -31,7 +30,6 @@ class Capabilities(models.TextChoices): | |||||||
|  |  | ||||||
|     CAN_SAVE_MEDIA = "can_save_media" |     CAN_SAVE_MEDIA = "can_save_media" | ||||||
|     CAN_GEO_IP = "can_geo_ip" |     CAN_GEO_IP = "can_geo_ip" | ||||||
|     CAN_ASN = "can_asn" |  | ||||||
|     CAN_IMPERSONATE = "can_impersonate" |     CAN_IMPERSONATE = "can_impersonate" | ||||||
|     CAN_DEBUG = "can_debug" |     CAN_DEBUG = "can_debug" | ||||||
|     IS_ENTERPRISE = "is_enterprise" |     IS_ENTERPRISE = "is_enterprise" | ||||||
| @ -70,10 +68,9 @@ class ConfigView(APIView): | |||||||
|         deb_test = settings.DEBUG or settings.TEST |         deb_test = settings.DEBUG or settings.TEST | ||||||
|         if Path(settings.MEDIA_ROOT).is_mount() or deb_test: |         if Path(settings.MEDIA_ROOT).is_mount() or deb_test: | ||||||
|             caps.append(Capabilities.CAN_SAVE_MEDIA) |             caps.append(Capabilities.CAN_SAVE_MEDIA) | ||||||
|         for processor in get_context_processors(): |         if GEOIP_READER.enabled: | ||||||
|             if cap := processor.capability(): |             caps.append(Capabilities.CAN_GEO_IP) | ||||||
|                 caps.append(cap) |         if CONFIG.get_bool("impersonation"): | ||||||
|         if self.request.tenant.impersonation: |  | ||||||
|             caps.append(Capabilities.CAN_IMPERSONATE) |             caps.append(Capabilities.CAN_IMPERSONATE) | ||||||
|         if settings.DEBUG:  # pragma: no cover |         if settings.DEBUG:  # pragma: no cover | ||||||
|             caps.append(Capabilities.CAN_DEBUG) |             caps.append(Capabilities.CAN_DEBUG) | ||||||
| @ -96,10 +93,10 @@ class ConfigView(APIView): | |||||||
|                     "traces_sample_rate": float(CONFIG.get("error_reporting.sample_rate", 0.4)), |                     "traces_sample_rate": float(CONFIG.get("error_reporting.sample_rate", 0.4)), | ||||||
|                 }, |                 }, | ||||||
|                 "capabilities": self.get_capabilities(), |                 "capabilities": self.get_capabilities(), | ||||||
|                 "cache_timeout": CONFIG.get_int("cache.timeout"), |                 "cache_timeout": CONFIG.get_int("redis.cache_timeout"), | ||||||
|                 "cache_timeout_flows": CONFIG.get_int("cache.timeout_flows"), |                 "cache_timeout_flows": CONFIG.get_int("redis.cache_timeout_flows"), | ||||||
|                 "cache_timeout_policies": CONFIG.get_int("cache.timeout_policies"), |                 "cache_timeout_policies": CONFIG.get_int("redis.cache_timeout_policies"), | ||||||
|                 "cache_timeout_reputation": CONFIG.get_int("cache.timeout_reputation"), |                 "cache_timeout_reputation": CONFIG.get_int("redis.cache_timeout_reputation"), | ||||||
|             } |             } | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """api v3 urls""" | """api v3 urls""" | ||||||
|  |  | ||||||
| from importlib import import_module | from importlib import import_module | ||||||
|  |  | ||||||
| from django.urls import path | from django.urls import path | ||||||
| @ -22,9 +21,7 @@ _other_urls = [] | |||||||
| for _authentik_app in get_apps(): | for _authentik_app in get_apps(): | ||||||
|     try: |     try: | ||||||
|         api_urls = import_module(f"{_authentik_app.name}.urls") |         api_urls = import_module(f"{_authentik_app.name}.urls") | ||||||
|     except ModuleNotFoundError: |     except (ModuleNotFoundError, ImportError) as exc: | ||||||
|         continue |  | ||||||
|     except ImportError as exc: |  | ||||||
|         LOGGER.warning("Could not import app's URLs", app_name=_authentik_app.name, exc=exc) |         LOGGER.warning("Could not import app's URLs", app_name=_authentik_app.name, exc=exc) | ||||||
|         continue |         continue | ||||||
|     if not hasattr(api_urls, "api_urlpatterns"): |     if not hasattr(api_urls, "api_urlpatterns"): | ||||||
| @ -33,7 +30,7 @@ for _authentik_app in get_apps(): | |||||||
|             app_name=_authentik_app.name, |             app_name=_authentik_app.name, | ||||||
|         ) |         ) | ||||||
|         continue |         continue | ||||||
|     urls: list = api_urls.api_urlpatterns |     urls: list = getattr(api_urls, "api_urlpatterns") | ||||||
|     for url in urls: |     for url in urls: | ||||||
|         if isinstance(url, URLPattern): |         if isinstance(url, URLPattern): | ||||||
|             _other_urls.append(url) |             _other_urls.append(url) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """General API Views""" | """General API Views""" | ||||||
|  |  | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| from django.urls import reverse | from django.urls import reverse | ||||||
|  | |||||||
| @ -1,22 +1,22 @@ | |||||||
| """Serializer mixin for managed models""" | """Serializer mixin for managed models""" | ||||||
|  |  | ||||||
| from django.utils.translation import gettext_lazy as _ | from django.utils.translation import gettext_lazy as _ | ||||||
| from drf_spectacular.utils import extend_schema, inline_serializer | from drf_spectacular.utils import extend_schema, inline_serializer | ||||||
| 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, DateTimeField | from rest_framework.fields import CharField, DateTimeField, JSONField | ||||||
|  | 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.serializers import ListSerializer, ModelSerializer | from rest_framework.serializers import ListSerializer, ModelSerializer | ||||||
| from rest_framework.viewsets import ModelViewSet | from rest_framework.viewsets import ModelViewSet | ||||||
|  |  | ||||||
|  | from authentik.api.decorators import permission_required | ||||||
| from authentik.blueprints.models import BlueprintInstance | from authentik.blueprints.models import BlueprintInstance | ||||||
| from authentik.blueprints.v1.importer import Importer | from authentik.blueprints.v1.importer import Importer | ||||||
| from authentik.blueprints.v1.oci import OCI_PREFIX | from authentik.blueprints.v1.oci import OCI_PREFIX | ||||||
| from authentik.blueprints.v1.tasks import apply_blueprint, blueprints_find_dict | from authentik.blueprints.v1.tasks import apply_blueprint, blueprints_find_dict | ||||||
| from authentik.core.api.used_by import UsedByMixin | from authentik.core.api.used_by import UsedByMixin | ||||||
| from authentik.core.api.utils import JSONDictField, PassiveSerializer | from authentik.core.api.utils import PassiveSerializer | ||||||
| from authentik.rbac.decorators import permission_required |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ManagedSerializer: | class ManagedSerializer: | ||||||
| @ -29,7 +29,7 @@ class MetadataSerializer(PassiveSerializer): | |||||||
|     """Serializer for blueprint metadata""" |     """Serializer for blueprint metadata""" | ||||||
|  |  | ||||||
|     name = CharField() |     name = CharField() | ||||||
|     labels = JSONDictField() |     labels = JSONField() | ||||||
|  |  | ||||||
|  |  | ||||||
| class BlueprintInstanceSerializer(ModelSerializer): | class BlueprintInstanceSerializer(ModelSerializer): | ||||||
| @ -49,12 +49,10 @@ class BlueprintInstanceSerializer(ModelSerializer): | |||||||
|         if content == "": |         if content == "": | ||||||
|             return content |             return content | ||||||
|         context = self.instance.context if self.instance else {} |         context = self.instance.context if self.instance else {} | ||||||
|         valid, logs = Importer.from_string(content, context).validate() |         valid, logs = Importer(content, context).validate() | ||||||
|         if not valid: |         if not valid: | ||||||
|             text_logs = "\n".join([x["event"] for x in logs]) |             text_logs = "\n".join([x["event"] for x in logs]) | ||||||
|             raise ValidationError( |             raise ValidationError(_("Failed to validate blueprint: %(logs)s" % {"logs": text_logs})) | ||||||
|                 _("Failed to validate blueprint: {logs}".format_map({"logs": text_logs})) |  | ||||||
|             ) |  | ||||||
|         return content |         return content | ||||||
|  |  | ||||||
|     def validate(self, attrs: dict) -> dict: |     def validate(self, attrs: dict) -> dict: | ||||||
| @ -89,11 +87,11 @@ class BlueprintInstanceSerializer(ModelSerializer): | |||||||
| class BlueprintInstanceViewSet(UsedByMixin, ModelViewSet): | class BlueprintInstanceViewSet(UsedByMixin, ModelViewSet): | ||||||
|     """Blueprint instances""" |     """Blueprint instances""" | ||||||
|  |  | ||||||
|  |     permission_classes = [IsAdminUser] | ||||||
|     serializer_class = BlueprintInstanceSerializer |     serializer_class = BlueprintInstanceSerializer | ||||||
|     queryset = BlueprintInstance.objects.all() |     queryset = BlueprintInstance.objects.all() | ||||||
|     search_fields = ["name", "path"] |     search_fields = ["name", "path"] | ||||||
|     filterset_fields = ["name", "path"] |     filterset_fields = ["name", "path"] | ||||||
|     ordering = ["name"] |  | ||||||
|  |  | ||||||
|     @extend_schema( |     @extend_schema( | ||||||
|         responses={ |         responses={ | ||||||
|  | |||||||
| @ -1,6 +1,5 @@ | |||||||
| """authentik Blueprints app""" | """authentik Blueprints app""" | ||||||
|  |  | ||||||
| from collections.abc import Callable |  | ||||||
| from importlib import import_module | from importlib import import_module | ||||||
| from inspect import ismethod | from inspect import ismethod | ||||||
|  |  | ||||||
| @ -12,91 +11,36 @@ from structlog.stdlib import BoundLogger, get_logger | |||||||
| class ManagedAppConfig(AppConfig): | class ManagedAppConfig(AppConfig): | ||||||
|     """Basic reconciliation logic for apps""" |     """Basic reconciliation logic for apps""" | ||||||
|  |  | ||||||
|     logger: BoundLogger |     _logger: BoundLogger | ||||||
|  |  | ||||||
|     RECONCILE_GLOBAL_CATEGORY: str = "global" |  | ||||||
|     RECONCILE_TENANT_CATEGORY: str = "tenant" |  | ||||||
|  |  | ||||||
|     def __init__(self, app_name: str, *args, **kwargs) -> None: |     def __init__(self, app_name: str, *args, **kwargs) -> None: | ||||||
|         super().__init__(app_name, *args, **kwargs) |         super().__init__(app_name, *args, **kwargs) | ||||||
|         self.logger = get_logger().bind(app_name=app_name) |         self._logger = get_logger().bind(app_name=app_name) | ||||||
|  |  | ||||||
|     def ready(self) -> None: |     def ready(self) -> None: | ||||||
|         self.import_related() |         self.reconcile() | ||||||
|         self._reconcile_global() |  | ||||||
|         self._reconcile_tenant() |  | ||||||
|         return super().ready() |         return super().ready() | ||||||
|  |  | ||||||
|     def import_related(self): |  | ||||||
|         """Automatically import related modules which rely on just being imported |  | ||||||
|         to register themselves (mainly django signals and celery tasks)""" |  | ||||||
|  |  | ||||||
|         def import_relative(rel_module: str): |  | ||||||
|             try: |  | ||||||
|                 module_name = f"{self.name}.{rel_module}" |  | ||||||
|                 import_module(module_name) |  | ||||||
|                 self.logger.info("Imported related module", module=module_name) |  | ||||||
|             except ModuleNotFoundError: |  | ||||||
|                 pass |  | ||||||
|  |  | ||||||
|         import_relative("checks") |  | ||||||
|         import_relative("tasks") |  | ||||||
|         import_relative("signals") |  | ||||||
|  |  | ||||||
|     def import_module(self, path: str): |     def import_module(self, path: str): | ||||||
|         """Load module""" |         """Load module""" | ||||||
|         import_module(path) |         import_module(path) | ||||||
|  |  | ||||||
|     def _reconcile(self, prefix: str) -> None: |     def reconcile(self) -> None: | ||||||
|  |         """reconcile ourselves""" | ||||||
|  |         prefix = "reconcile_" | ||||||
|         for meth_name in dir(self): |         for meth_name in dir(self): | ||||||
|             meth = getattr(self, meth_name) |             meth = getattr(self, meth_name) | ||||||
|             if not ismethod(meth): |             if not ismethod(meth): | ||||||
|                 continue |                 continue | ||||||
|             category = getattr(meth, "_authentik_managed_reconcile", None) |             if not meth_name.startswith(prefix): | ||||||
|             if category != prefix: |  | ||||||
|                 continue |                 continue | ||||||
|             name = meth_name.replace(prefix, "") |             name = meth_name.replace(prefix, "") | ||||||
|             try: |             try: | ||||||
|                 self.logger.debug("Starting reconciler", name=name) |                 self._logger.debug("Starting reconciler", name=name) | ||||||
|                 meth() |                 meth() | ||||||
|                 self.logger.debug("Successfully reconciled", name=name) |                 self._logger.debug("Successfully reconciled", name=name) | ||||||
|             except (DatabaseError, ProgrammingError, InternalError) as exc: |             except (DatabaseError, ProgrammingError, InternalError) as exc: | ||||||
|                 self.logger.warning("Failed to run reconcile", name=name, exc=exc) |                 self._logger.debug("Failed to run reconcile", name=name, exc=exc) | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def reconcile_tenant(func: Callable): |  | ||||||
|         """Mark a function to be called on startup (for each tenant)""" |  | ||||||
|         func._authentik_managed_reconcile = ManagedAppConfig.RECONCILE_TENANT_CATEGORY |  | ||||||
|         return func |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def reconcile_global(func: Callable): |  | ||||||
|         """Mark a function to be called on startup (globally)""" |  | ||||||
|         func._authentik_managed_reconcile = ManagedAppConfig.RECONCILE_GLOBAL_CATEGORY |  | ||||||
|         return func |  | ||||||
|  |  | ||||||
|     def _reconcile_tenant(self) -> None: |  | ||||||
|         """reconcile ourselves for tenanted methods""" |  | ||||||
|         from authentik.tenants.models import Tenant |  | ||||||
|  |  | ||||||
|         try: |  | ||||||
|             tenants = list(Tenant.objects.filter(ready=True)) |  | ||||||
|         except (DatabaseError, ProgrammingError, InternalError) as exc: |  | ||||||
|             self.logger.debug("Failed to get tenants to run reconcile", exc=exc) |  | ||||||
|             return |  | ||||||
|         for tenant in tenants: |  | ||||||
|             with tenant: |  | ||||||
|                 self._reconcile(self.RECONCILE_TENANT_CATEGORY) |  | ||||||
|  |  | ||||||
|     def _reconcile_global(self) -> None: |  | ||||||
|         """ |  | ||||||
|         reconcile ourselves for global methods. |  | ||||||
|         Used for signals, tasks, etc. Database queries should not be made in here. |  | ||||||
|         """ |  | ||||||
|         from django_tenants.utils import get_public_schema_name, schema_context |  | ||||||
|  |  | ||||||
|         with schema_context(get_public_schema_name()): |  | ||||||
|             self._reconcile(self.RECONCILE_GLOBAL_CATEGORY) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthentikBlueprintsConfig(ManagedAppConfig): | class AuthentikBlueprintsConfig(ManagedAppConfig): | ||||||
| @ -107,13 +51,11 @@ class AuthentikBlueprintsConfig(ManagedAppConfig): | |||||||
|     verbose_name = "authentik Blueprints" |     verbose_name = "authentik Blueprints" | ||||||
|     default = True |     default = True | ||||||
|  |  | ||||||
|     @ManagedAppConfig.reconcile_global |     def reconcile_load_blueprints_v1_tasks(self): | ||||||
|     def load_blueprints_v1_tasks(self): |  | ||||||
|         """Load v1 tasks""" |         """Load v1 tasks""" | ||||||
|         self.import_module("authentik.blueprints.v1.tasks") |         self.import_module("authentik.blueprints.v1.tasks") | ||||||
|  |  | ||||||
|     @ManagedAppConfig.reconcile_tenant |     def reconcile_blueprints_discovery(self): | ||||||
|     def blueprints_discovery(self): |  | ||||||
|         """Run blueprint discovery""" |         """Run blueprint discovery""" | ||||||
|         from authentik.blueprints.v1.tasks import blueprints_discovery, clear_failed_blueprints |         from authentik.blueprints.v1.tasks import blueprints_discovery, clear_failed_blueprints | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Apply blueprint from commandline""" | """Apply blueprint from commandline""" | ||||||
|  |  | ||||||
| from sys import exit as sys_exit | from sys import exit as sys_exit | ||||||
|  |  | ||||||
| from django.core.management.base import BaseCommand, no_translations | from django.core.management.base import BaseCommand, no_translations | ||||||
| @ -7,7 +6,6 @@ from structlog.stdlib import get_logger | |||||||
|  |  | ||||||
| from authentik.blueprints.models import BlueprintInstance | from authentik.blueprints.models import BlueprintInstance | ||||||
| from authentik.blueprints.v1.importer import Importer | from authentik.blueprints.v1.importer import Importer | ||||||
| from authentik.tenants.models import Tenant |  | ||||||
|  |  | ||||||
| LOGGER = get_logger() | LOGGER = get_logger() | ||||||
|  |  | ||||||
| @ -18,11 +16,9 @@ class Command(BaseCommand): | |||||||
|     @no_translations |     @no_translations | ||||||
|     def handle(self, *args, **options): |     def handle(self, *args, **options): | ||||||
|         """Apply all blueprints in order, abort when one fails to import""" |         """Apply all blueprints in order, abort when one fails to import""" | ||||||
|         for tenant in Tenant.objects.filter(ready=True): |  | ||||||
|             with tenant: |  | ||||||
|         for blueprint_path in options.get("blueprints", []): |         for blueprint_path in options.get("blueprints", []): | ||||||
|             content = BlueprintInstance(path=blueprint_path).retrieve() |             content = BlueprintInstance(path=blueprint_path).retrieve() | ||||||
|                     importer = Importer.from_string(content) |             importer = Importer(content) | ||||||
|             valid, _ = importer.validate() |             valid, _ = importer.validate() | ||||||
|             if not valid: |             if not valid: | ||||||
|                 self.stderr.write("blueprint invalid") |                 self.stderr.write("blueprint invalid") | ||||||
|  | |||||||
| @ -1,19 +1,17 @@ | |||||||
| """Export blueprint of current authentik install""" | """Export blueprint of current authentik install""" | ||||||
|  | from django.core.management.base import BaseCommand, no_translations | ||||||
| from django.core.management.base import no_translations |  | ||||||
| from structlog.stdlib import get_logger | from structlog.stdlib import get_logger | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.exporter import Exporter | from authentik.blueprints.v1.exporter import Exporter | ||||||
| from authentik.tenants.management import TenantCommand |  | ||||||
|  |  | ||||||
| LOGGER = get_logger() | LOGGER = get_logger() | ||||||
|  |  | ||||||
|  |  | ||||||
| class Command(TenantCommand): | class Command(BaseCommand): | ||||||
|     """Export blueprint of current authentik install""" |     """Export blueprint of current authentik install""" | ||||||
|  |  | ||||||
|     @no_translations |     @no_translations | ||||||
|     def handle_per_tenant(self, *args, **options): |     def handle(self, *args, **options): | ||||||
|         """Export blueprint of current authentik install""" |         """Export blueprint of current authentik install""" | ||||||
|         exporter = Exporter() |         exporter = Exporter() | ||||||
|         self.stdout.write(exporter.export_to_string()) |         self.stdout.write(exporter.export_to_string()) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Generate JSON Schema for blueprints""" | """Generate JSON Schema for blueprints""" | ||||||
|  |  | ||||||
| from json import dumps | from json import dumps | ||||||
| from typing import Any | from typing import Any | ||||||
|  |  | ||||||
| @ -10,7 +9,6 @@ from rest_framework.fields import Field, JSONField, UUIDField | |||||||
| from rest_framework.serializers import Serializer | from rest_framework.serializers import Serializer | ||||||
| from structlog.stdlib import get_logger | from structlog.stdlib import get_logger | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.common import BlueprintEntryDesiredState |  | ||||||
| from authentik.blueprints.v1.importer import SERIALIZER_CONTEXT_BLUEPRINT, is_model_allowed | from authentik.blueprints.v1.importer import SERIALIZER_CONTEXT_BLUEPRINT, is_model_allowed | ||||||
| from authentik.blueprints.v1.meta.registry import BaseMetaModel, registry | from authentik.blueprints.v1.meta.registry import BaseMetaModel, registry | ||||||
| from authentik.lib.models import SerializerModel | from authentik.lib.models import SerializerModel | ||||||
| @ -112,7 +110,7 @@ class Command(BaseCommand): | |||||||
|                 "id": {"type": "string"}, |                 "id": {"type": "string"}, | ||||||
|                 "state": { |                 "state": { | ||||||
|                     "type": "string", |                     "type": "string", | ||||||
|                     "enum": [s.value for s in BlueprintEntryDesiredState], |                     "enum": ["absent", "present", "created"], | ||||||
|                     "default": "present", |                     "default": "present", | ||||||
|                 }, |                 }, | ||||||
|                 "conditions": {"type": "array", "items": {"type": "boolean"}}, |                 "conditions": {"type": "array", "items": {"type": "boolean"}}, | ||||||
|  | |||||||
| @ -14,7 +14,7 @@ from authentik.blueprints.v1.labels import LABEL_AUTHENTIK_SYSTEM | |||||||
| from authentik.lib.config import CONFIG | from authentik.lib.config import CONFIG | ||||||
|  |  | ||||||
|  |  | ||||||
| def check_blueprint_v1_file(BlueprintInstance: type, db_alias, path: Path): | def check_blueprint_v1_file(BlueprintInstance: type, path: Path): | ||||||
|     """Check if blueprint should be imported""" |     """Check if blueprint should be imported""" | ||||||
|     from authentik.blueprints.models import BlueprintInstanceStatus |     from authentik.blueprints.models import BlueprintInstanceStatus | ||||||
|     from authentik.blueprints.v1.common import BlueprintLoader, BlueprintMetadata |     from authentik.blueprints.v1.common import BlueprintLoader, BlueprintMetadata | ||||||
| @ -29,9 +29,7 @@ def check_blueprint_v1_file(BlueprintInstance: type, db_alias, path: Path): | |||||||
|         if version != 1: |         if version != 1: | ||||||
|             return |             return | ||||||
|         blueprint_file.seek(0) |         blueprint_file.seek(0) | ||||||
|     instance: BlueprintInstance = ( |     instance: BlueprintInstance = BlueprintInstance.objects.filter(path=path).first() | ||||||
|         BlueprintInstance.objects.using(db_alias).filter(path=path).first() |  | ||||||
|     ) |  | ||||||
|     rel_path = path.relative_to(Path(CONFIG.get("blueprints_dir"))) |     rel_path = path.relative_to(Path(CONFIG.get("blueprints_dir"))) | ||||||
|     meta = None |     meta = None | ||||||
|     if metadata: |     if metadata: | ||||||
| @ -39,7 +37,7 @@ def check_blueprint_v1_file(BlueprintInstance: type, db_alias, path: Path): | |||||||
|         if meta.labels.get(LABEL_AUTHENTIK_INSTANTIATE, "").lower() == "false": |         if meta.labels.get(LABEL_AUTHENTIK_INSTANTIATE, "").lower() == "false": | ||||||
|             return |             return | ||||||
|     if not instance: |     if not instance: | ||||||
|         BlueprintInstance.objects.using(db_alias).create( |         instance = BlueprintInstance( | ||||||
|             name=meta.name if meta else str(rel_path), |             name=meta.name if meta else str(rel_path), | ||||||
|             path=str(rel_path), |             path=str(rel_path), | ||||||
|             context={}, |             context={}, | ||||||
| @ -49,6 +47,7 @@ def check_blueprint_v1_file(BlueprintInstance: type, db_alias, path: Path): | |||||||
|             last_applied_hash="", |             last_applied_hash="", | ||||||
|             metadata=metadata or {}, |             metadata=metadata or {}, | ||||||
|         ) |         ) | ||||||
|  |         instance.save() | ||||||
|  |  | ||||||
|  |  | ||||||
| def migration_blueprint_import(apps: Apps, schema_editor: BaseDatabaseSchemaEditor): | def migration_blueprint_import(apps: Apps, schema_editor: BaseDatabaseSchemaEditor): | ||||||
| @ -57,7 +56,7 @@ def migration_blueprint_import(apps: Apps, schema_editor: BaseDatabaseSchemaEdit | |||||||
|  |  | ||||||
|     db_alias = schema_editor.connection.alias |     db_alias = schema_editor.connection.alias | ||||||
|     for file in glob(f"{CONFIG.get('blueprints_dir')}/**/*.yaml", recursive=True): |     for file in glob(f"{CONFIG.get('blueprints_dir')}/**/*.yaml", recursive=True): | ||||||
|         check_blueprint_v1_file(BlueprintInstance, db_alias, Path(file)) |         check_blueprint_v1_file(BlueprintInstance, Path(file)) | ||||||
|  |  | ||||||
|     for blueprint in BlueprintInstance.objects.using(db_alias).all(): |     for blueprint in BlueprintInstance.objects.using(db_alias).all(): | ||||||
|         # If we already have flows (and we should always run before flow migrations) |         # If we already have flows (and we should always run before flow migrations) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """blueprint models""" | """blueprint models""" | ||||||
|  |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from uuid import uuid4 | from uuid import uuid4 | ||||||
|  |  | ||||||
| @ -71,19 +70,6 @@ class BlueprintInstance(SerializerModel, ManagedModel, CreatedUpdatedModel): | |||||||
|     enabled = models.BooleanField(default=True) |     enabled = models.BooleanField(default=True) | ||||||
|     managed_models = ArrayField(models.TextField(), default=list) |     managed_models = ArrayField(models.TextField(), default=list) | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _("Blueprint Instance") |  | ||||||
|         verbose_name_plural = _("Blueprint Instances") |  | ||||||
|         unique_together = ( |  | ||||||
|             ( |  | ||||||
|                 "name", |  | ||||||
|                 "path", |  | ||||||
|             ), |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|     def __str__(self) -> str: |  | ||||||
|         return f"Blueprint Instance {self.name}" |  | ||||||
|  |  | ||||||
|     def retrieve_oci(self) -> str: |     def retrieve_oci(self) -> str: | ||||||
|         """Get blueprint from an OCI registry""" |         """Get blueprint from an OCI registry""" | ||||||
|         client = BlueprintOCIClient(self.path.replace(OCI_PREFIX, "https://")) |         client = BlueprintOCIClient(self.path.replace(OCI_PREFIX, "https://")) | ||||||
| @ -102,7 +88,7 @@ class BlueprintInstance(SerializerModel, ManagedModel, CreatedUpdatedModel): | |||||||
|                 raise BlueprintRetrievalFailed("Invalid blueprint path") |                 raise BlueprintRetrievalFailed("Invalid blueprint path") | ||||||
|             with full_path.open("r", encoding="utf-8") as _file: |             with full_path.open("r", encoding="utf-8") as _file: | ||||||
|                 return _file.read() |                 return _file.read() | ||||||
|         except OSError as exc: |         except (IOError, OSError) as exc: | ||||||
|             raise BlueprintRetrievalFailed(exc) from exc |             raise BlueprintRetrievalFailed(exc) from exc | ||||||
|  |  | ||||||
|     def retrieve(self) -> str: |     def retrieve(self) -> str: | ||||||
| @ -118,3 +104,16 @@ class BlueprintInstance(SerializerModel, ManagedModel, CreatedUpdatedModel): | |||||||
|         from authentik.blueprints.api import BlueprintInstanceSerializer |         from authentik.blueprints.api import BlueprintInstanceSerializer | ||||||
|  |  | ||||||
|         return BlueprintInstanceSerializer |         return BlueprintInstanceSerializer | ||||||
|  |  | ||||||
|  |     def __str__(self) -> str: | ||||||
|  |         return f"Blueprint Instance {self.name}" | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         verbose_name = _("Blueprint Instance") | ||||||
|  |         verbose_name_plural = _("Blueprint Instances") | ||||||
|  |         unique_together = ( | ||||||
|  |             ( | ||||||
|  |                 "name", | ||||||
|  |                 "path", | ||||||
|  |             ), | ||||||
|  |         ) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """blueprint Settings""" | """blueprint Settings""" | ||||||
|  |  | ||||||
| from celery.schedules import crontab | from celery.schedules import crontab | ||||||
|  |  | ||||||
| from authentik.lib.utils.time import fqdn_rand | from authentik.lib.utils.time import fqdn_rand | ||||||
|  | |||||||
| @ -1,7 +1,6 @@ | |||||||
| """Blueprint helpers""" | """Blueprint helpers""" | ||||||
|  |  | ||||||
| from collections.abc import Callable |  | ||||||
| from functools import wraps | from functools import wraps | ||||||
|  | from typing import Callable | ||||||
|  |  | ||||||
| from django.apps import apps | from django.apps import apps | ||||||
|  |  | ||||||
| @ -21,7 +20,7 @@ def apply_blueprint(*files: str): | |||||||
|         def wrapper(*args, **kwargs): |         def wrapper(*args, **kwargs): | ||||||
|             for file in files: |             for file in files: | ||||||
|                 content = BlueprintInstance(path=file).retrieve() |                 content = BlueprintInstance(path=file).retrieve() | ||||||
|                 Importer.from_string(content).apply() |                 Importer(content).apply() | ||||||
|             return func(*args, **kwargs) |             return func(*args, **kwargs) | ||||||
|  |  | ||||||
|         return wrapper |         return wrapper | ||||||
| @ -39,7 +38,7 @@ def reconcile_app(app_name: str): | |||||||
|         def wrapper(*args, **kwargs): |         def wrapper(*args, **kwargs): | ||||||
|             config = apps.get_app_config(app_name) |             config = apps.get_app_config(app_name) | ||||||
|             if isinstance(config, ManagedAppConfig): |             if isinstance(config, ManagedAppConfig): | ||||||
|                 config.ready() |                 config.reconcile() | ||||||
|             return func(*args, **kwargs) |             return func(*args, **kwargs) | ||||||
|  |  | ||||||
|         return wrapper |         return wrapper | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """authentik managed models tests""" | """authentik managed models tests""" | ||||||
|  |  | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
|  |  | ||||||
| from authentik.blueprints.models import BlueprintInstance, BlueprintRetrievalFailed | from authentik.blueprints.models import BlueprintInstance, BlueprintRetrievalFailed | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test blueprints OCI""" | """Test blueprints OCI""" | ||||||
|  |  | ||||||
| from django.test import TransactionTestCase | from django.test import TransactionTestCase | ||||||
| from requests_mock import Mocker | from requests_mock import Mocker | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,23 +1,22 @@ | |||||||
| """test packaged blueprints""" | """test packaged blueprints""" | ||||||
|  |  | ||||||
| from collections.abc import Callable |  | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
|  | from typing import Callable | ||||||
|  |  | ||||||
| from django.test import TransactionTestCase | from django.test import TransactionTestCase | ||||||
|  |  | ||||||
| from authentik.blueprints.models import BlueprintInstance | from authentik.blueprints.models import BlueprintInstance | ||||||
| from authentik.blueprints.tests import apply_blueprint | from authentik.blueprints.tests import apply_blueprint | ||||||
| from authentik.blueprints.v1.importer import Importer | from authentik.blueprints.v1.importer import Importer | ||||||
| from authentik.brands.models import Brand | from authentik.tenants.models import Tenant | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestPackaged(TransactionTestCase): | class TestPackaged(TransactionTestCase): | ||||||
|     """Empty class, test methods are added dynamically""" |     """Empty class, test methods are added dynamically""" | ||||||
|  |  | ||||||
|     @apply_blueprint("default/default-brand.yaml") |     @apply_blueprint("default/default-tenant.yaml") | ||||||
|     def test_decorator_static(self): |     def test_decorator_static(self): | ||||||
|         """Test @apply_blueprint decorator""" |         """Test @apply_blueprint decorator""" | ||||||
|         self.assertTrue(Brand.objects.filter(domain="authentik-default").exists()) |         self.assertTrue(Tenant.objects.filter(domain="authentik-default").exists()) | ||||||
|  |  | ||||||
|  |  | ||||||
| def blueprint_tester(file_name: Path) -> Callable: | def blueprint_tester(file_name: Path) -> Callable: | ||||||
| @ -26,7 +25,7 @@ def blueprint_tester(file_name: Path) -> Callable: | |||||||
|     def tester(self: TestPackaged): |     def tester(self: TestPackaged): | ||||||
|         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(BlueprintInstance(path=str(rel_path)).retrieve()) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,20 +1,18 @@ | |||||||
| """authentik managed models tests""" | """authentik managed models tests""" | ||||||
|  | from typing import Callable, Type | ||||||
| from collections.abc import Callable |  | ||||||
|  |  | ||||||
| from django.apps import apps | from django.apps import apps | ||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.importer import is_model_allowed | from authentik.blueprints.v1.importer import is_model_allowed | ||||||
| from authentik.lib.models import SerializerModel | from authentik.lib.models import SerializerModel | ||||||
| from authentik.providers.oauth2.models import RefreshToken |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TestModels(TestCase): | class TestModels(TestCase): | ||||||
|     """Test Models""" |     """Test Models""" | ||||||
|  |  | ||||||
|  |  | ||||||
| def serializer_tester_factory(test_model: type[SerializerModel]) -> Callable: | def serializer_tester_factory(test_model: Type[SerializerModel]) -> Callable: | ||||||
|     """Test serializer""" |     """Test serializer""" | ||||||
|  |  | ||||||
|     def tester(self: TestModels): |     def tester(self: TestModels): | ||||||
| @ -23,9 +21,6 @@ def serializer_tester_factory(test_model: type[SerializerModel]) -> Callable: | |||||||
|         model_class = test_model() |         model_class = test_model() | ||||||
|         self.assertTrue(isinstance(model_class, SerializerModel)) |         self.assertTrue(isinstance(model_class, SerializerModel)) | ||||||
|         self.assertIsNotNone(model_class.serializer) |         self.assertIsNotNone(model_class.serializer) | ||||||
|         if model_class.serializer.Meta().model == RefreshToken: |  | ||||||
|             return |  | ||||||
|         self.assertEqual(model_class.serializer.Meta().model, test_model) |  | ||||||
|  |  | ||||||
|     return tester |     return tester | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test blueprints v1""" | """Test blueprints v1""" | ||||||
|  |  | ||||||
| from os import environ | from os import environ | ||||||
|  |  | ||||||
| from django.test import TransactionTestCase | from django.test import TransactionTestCase | ||||||
| @ -22,14 +21,14 @@ class TestBlueprintsV1(TransactionTestCase): | |||||||
|  |  | ||||||
|     def test_blueprint_invalid_format(self): |     def test_blueprint_invalid_format(self): | ||||||
|         """Test blueprint with invalid format""" |         """Test blueprint with invalid format""" | ||||||
|         importer = Importer.from_string('{"version": 3}') |         importer = Importer('{"version": 3}') | ||||||
|         self.assertFalse(importer.validate()[0]) |         self.assertFalse(importer.validate()[0]) | ||||||
|         importer = Importer.from_string( |         importer = Importer( | ||||||
|             '{"version": 1,"entries":[{"identifiers":{},"attrs":{},' |             '{"version": 1,"entries":[{"identifiers":{},"attrs":{},' | ||||||
|             '"model": "authentik_core.User"}]}' |             '"model": "authentik_core.User"}]}' | ||||||
|         ) |         ) | ||||||
|         self.assertFalse(importer.validate()[0]) |         self.assertFalse(importer.validate()[0]) | ||||||
|         importer = Importer.from_string( |         importer = Importer( | ||||||
|             '{"version": 1, "entries": [{"attrs": {"name": "test"}, ' |             '{"version": 1, "entries": [{"attrs": {"name": "test"}, ' | ||||||
|             '"identifiers": {}, ' |             '"identifiers": {}, ' | ||||||
|             '"model": "authentik_core.Group"}]}' |             '"model": "authentik_core.Group"}]}' | ||||||
| @ -55,7 +54,7 @@ class TestBlueprintsV1(TransactionTestCase): | |||||||
|             }, |             }, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         importer = Importer.from_string( |         importer = Importer( | ||||||
|             '{"version": 1, "entries": [{"attrs": {"name": "test999", "attributes": ' |             '{"version": 1, "entries": [{"attrs": {"name": "test999", "attributes": ' | ||||||
|             '{"key": ["updated_value"]}}, "identifiers": {"attributes": {"other_key": ' |             '{"key": ["updated_value"]}}, "identifiers": {"attributes": {"other_key": ' | ||||||
|             '["other_value"]}}, "model": "authentik_core.Group"}]}' |             '["other_value"]}}, "model": "authentik_core.Group"}]}' | ||||||
| @ -104,7 +103,7 @@ class TestBlueprintsV1(TransactionTestCase): | |||||||
|             self.assertEqual(len(export.entries), 3) |             self.assertEqual(len(export.entries), 3) | ||||||
|             export_yaml = exporter.export_to_string() |             export_yaml = exporter.export_to_string() | ||||||
|  |  | ||||||
|         importer = Importer.from_string(export_yaml) |         importer = Importer(export_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|  |  | ||||||
| @ -114,14 +113,14 @@ class TestBlueprintsV1(TransactionTestCase): | |||||||
|         """Test export and import it twice""" |         """Test export and import it twice""" | ||||||
|         count_initial = Prompt.objects.filter(field_key="username").count() |         count_initial = Prompt.objects.filter(field_key="username").count() | ||||||
|  |  | ||||||
|         importer = Importer.from_string(load_fixture("fixtures/static_prompt_export.yaml")) |         importer = Importer(load_fixture("fixtures/static_prompt_export.yaml")) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|  |  | ||||||
|         count_before = Prompt.objects.filter(field_key="username").count() |         count_before = Prompt.objects.filter(field_key="username").count() | ||||||
|         self.assertEqual(count_initial + 1, count_before) |         self.assertEqual(count_initial + 1, count_before) | ||||||
|  |  | ||||||
|         importer = Importer.from_string(load_fixture("fixtures/static_prompt_export.yaml")) |         importer = Importer(load_fixture("fixtures/static_prompt_export.yaml")) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|  |  | ||||||
|         self.assertEqual(Prompt.objects.filter(field_key="username").count(), count_before) |         self.assertEqual(Prompt.objects.filter(field_key="username").count(), count_before) | ||||||
| @ -131,7 +130,7 @@ class TestBlueprintsV1(TransactionTestCase): | |||||||
|         ExpressionPolicy.objects.filter(name="foo-bar-baz-qux").delete() |         ExpressionPolicy.objects.filter(name="foo-bar-baz-qux").delete() | ||||||
|         Group.objects.filter(name="test").delete() |         Group.objects.filter(name="test").delete() | ||||||
|         environ["foo"] = generate_id() |         environ["foo"] = generate_id() | ||||||
|         importer = Importer.from_string(load_fixture("fixtures/tags.yaml"), {"bar": "baz"}) |         importer = Importer(load_fixture("fixtures/tags.yaml"), {"bar": "baz"}) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         policy = ExpressionPolicy.objects.filter(name="foo-bar-baz-qux").first() |         policy = ExpressionPolicy.objects.filter(name="foo-bar-baz-qux").first() | ||||||
| @ -249,7 +248,7 @@ class TestBlueprintsV1(TransactionTestCase): | |||||||
|             exporter = FlowExporter(flow) |             exporter = FlowExporter(flow) | ||||||
|             export_yaml = exporter.export_to_string() |             export_yaml = exporter.export_to_string() | ||||||
|  |  | ||||||
|         importer = Importer.from_string(export_yaml) |         importer = Importer(export_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         self.assertTrue(UserLoginStage.objects.filter(name=stage_name).exists()) |         self.assertTrue(UserLoginStage.objects.filter(name=stage_name).exists()) | ||||||
| @ -298,7 +297,7 @@ class TestBlueprintsV1(TransactionTestCase): | |||||||
|             exporter = FlowExporter(flow) |             exporter = FlowExporter(flow) | ||||||
|             export_yaml = exporter.export_to_string() |             export_yaml = exporter.export_to_string() | ||||||
|  |  | ||||||
|         importer = Importer.from_string(export_yaml) |         importer = Importer(export_yaml) | ||||||
|  |  | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test blueprints v1 api""" | """Test blueprints v1 api""" | ||||||
|  |  | ||||||
| from json import loads | from json import loads | ||||||
| from tempfile import NamedTemporaryFile, mkdtemp | from tempfile import NamedTemporaryFile, mkdtemp | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test blueprints v1""" | """Test blueprints v1""" | ||||||
|  |  | ||||||
| from django.test import TransactionTestCase | from django.test import TransactionTestCase | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.importer import Importer | from authentik.blueprints.v1.importer import Importer | ||||||
| @ -19,7 +18,7 @@ class TestBlueprintsV1ConditionalFields(TransactionTestCase): | |||||||
|         self.uid = generate_id() |         self.uid = generate_id() | ||||||
|         import_yaml = load_fixture("fixtures/conditional_fields.yaml", uid=self.uid, user=user.pk) |         import_yaml = load_fixture("fixtures/conditional_fields.yaml", uid=self.uid, user=user.pk) | ||||||
|  |  | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test blueprints v1""" | """Test blueprints v1""" | ||||||
|  |  | ||||||
| from django.test import TransactionTestCase | from django.test import TransactionTestCase | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.importer import Importer | from authentik.blueprints.v1.importer import Importer | ||||||
| @ -19,7 +18,7 @@ class TestBlueprintsV1Conditions(TransactionTestCase): | |||||||
|             "fixtures/conditions_fulfilled.yaml", id1=flow_slug1, id2=flow_slug2 |             "fixtures/conditions_fulfilled.yaml", id1=flow_slug1, id2=flow_slug2 | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         # Ensure objects exist |         # Ensure objects exist | ||||||
| @ -36,7 +35,7 @@ class TestBlueprintsV1Conditions(TransactionTestCase): | |||||||
|             "fixtures/conditions_not_fulfilled.yaml", id1=flow_slug1, id2=flow_slug2 |             "fixtures/conditions_not_fulfilled.yaml", id1=flow_slug1, id2=flow_slug2 | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         # Ensure objects do not exist |         # Ensure objects do not exist | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test blueprints v1""" | """Test blueprints v1""" | ||||||
|  |  | ||||||
| from django.test import TransactionTestCase | from django.test import TransactionTestCase | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.importer import Importer | from authentik.blueprints.v1.importer import Importer | ||||||
| @ -16,7 +15,7 @@ class TestBlueprintsV1State(TransactionTestCase): | |||||||
|         flow_slug = generate_id() |         flow_slug = generate_id() | ||||||
|         import_yaml = load_fixture("fixtures/state_present.yaml", id=flow_slug) |         import_yaml = load_fixture("fixtures/state_present.yaml", id=flow_slug) | ||||||
|  |  | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         # Ensure object exists |         # Ensure object exists | ||||||
| @ -31,7 +30,7 @@ class TestBlueprintsV1State(TransactionTestCase): | |||||||
|         self.assertEqual(flow.title, "bar") |         self.assertEqual(flow.title, "bar") | ||||||
|  |  | ||||||
|         # Ensure importer updates it |         # Ensure importer updates it | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         flow: Flow = Flow.objects.filter(slug=flow_slug).first() |         flow: Flow = Flow.objects.filter(slug=flow_slug).first() | ||||||
| @ -42,7 +41,7 @@ class TestBlueprintsV1State(TransactionTestCase): | |||||||
|         flow_slug = generate_id() |         flow_slug = generate_id() | ||||||
|         import_yaml = load_fixture("fixtures/state_created.yaml", id=flow_slug) |         import_yaml = load_fixture("fixtures/state_created.yaml", id=flow_slug) | ||||||
|  |  | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         # Ensure object exists |         # Ensure object exists | ||||||
| @ -57,7 +56,7 @@ class TestBlueprintsV1State(TransactionTestCase): | |||||||
|         self.assertEqual(flow.title, "bar") |         self.assertEqual(flow.title, "bar") | ||||||
|  |  | ||||||
|         # Ensure importer doesn't update it |         # Ensure importer doesn't update it | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         flow: Flow = Flow.objects.filter(slug=flow_slug).first() |         flow: Flow = Flow.objects.filter(slug=flow_slug).first() | ||||||
| @ -68,7 +67,7 @@ class TestBlueprintsV1State(TransactionTestCase): | |||||||
|         flow_slug = generate_id() |         flow_slug = generate_id() | ||||||
|         import_yaml = load_fixture("fixtures/state_created.yaml", id=flow_slug) |         import_yaml = load_fixture("fixtures/state_created.yaml", id=flow_slug) | ||||||
|  |  | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         # Ensure object exists |         # Ensure object exists | ||||||
| @ -76,7 +75,7 @@ class TestBlueprintsV1State(TransactionTestCase): | |||||||
|         self.assertEqual(flow.slug, flow_slug) |         self.assertEqual(flow.slug, flow_slug) | ||||||
|  |  | ||||||
|         import_yaml = load_fixture("fixtures/state_absent.yaml", id=flow_slug) |         import_yaml = load_fixture("fixtures/state_absent.yaml", id=flow_slug) | ||||||
|         importer = Importer.from_string(import_yaml) |         importer = Importer(import_yaml) | ||||||
|         self.assertTrue(importer.validate()[0]) |         self.assertTrue(importer.validate()[0]) | ||||||
|         self.assertTrue(importer.apply()) |         self.assertTrue(importer.apply()) | ||||||
|         flow: Flow = Flow.objects.filter(slug=flow_slug).first() |         flow: Flow = Flow.objects.filter(slug=flow_slug).first() | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Test blueprints v1 tasks""" | """Test blueprints v1 tasks""" | ||||||
|  |  | ||||||
| from hashlib import sha512 | from hashlib import sha512 | ||||||
| from tempfile import NamedTemporaryFile, mkdtemp | from tempfile import NamedTemporaryFile, mkdtemp | ||||||
|  |  | ||||||
| @ -54,7 +53,7 @@ class TestBlueprintsV1Tasks(TransactionTestCase): | |||||||
|             file.seek(0) |             file.seek(0) | ||||||
|             file_hash = sha512(file.read().encode()).hexdigest() |             file_hash = sha512(file.read().encode()).hexdigest() | ||||||
|             file.flush() |             file.flush() | ||||||
|             blueprints_discovery() |             blueprints_discovery()  # pylint: disable=no-value-for-parameter | ||||||
|             instance = BlueprintInstance.objects.filter(name=blueprint_id).first() |             instance = BlueprintInstance.objects.filter(name=blueprint_id).first() | ||||||
|             self.assertEqual(instance.last_applied_hash, file_hash) |             self.assertEqual(instance.last_applied_hash, file_hash) | ||||||
|             self.assertEqual( |             self.assertEqual( | ||||||
| @ -82,7 +81,7 @@ class TestBlueprintsV1Tasks(TransactionTestCase): | |||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
|             file.flush() |             file.flush() | ||||||
|             blueprints_discovery() |             blueprints_discovery()  # pylint: disable=no-value-for-parameter | ||||||
|             blueprint = BlueprintInstance.objects.filter(name="foo").first() |             blueprint = BlueprintInstance.objects.filter(name="foo").first() | ||||||
|             self.assertEqual( |             self.assertEqual( | ||||||
|                 blueprint.last_applied_hash, |                 blueprint.last_applied_hash, | ||||||
| @ -107,7 +106,7 @@ class TestBlueprintsV1Tasks(TransactionTestCase): | |||||||
|                 ) |                 ) | ||||||
|             ) |             ) | ||||||
|             file.flush() |             file.flush() | ||||||
|             blueprints_discovery() |             blueprints_discovery()  # pylint: disable=no-value-for-parameter | ||||||
|             blueprint.refresh_from_db() |             blueprint.refresh_from_db() | ||||||
|             self.assertEqual( |             self.assertEqual( | ||||||
|                 blueprint.last_applied_hash, |                 blueprint.last_applied_hash, | ||||||
| @ -149,7 +148,7 @@ class TestBlueprintsV1Tasks(TransactionTestCase): | |||||||
|                 instance.status, |                 instance.status, | ||||||
|                 BlueprintInstanceStatus.UNKNOWN, |                 BlueprintInstanceStatus.UNKNOWN, | ||||||
|             ) |             ) | ||||||
|             apply_blueprint(instance.pk) |             apply_blueprint(instance.pk)  # pylint: disable=no-value-for-parameter | ||||||
|             instance.refresh_from_db() |             instance.refresh_from_db() | ||||||
|             self.assertEqual(instance.last_applied_hash, "") |             self.assertEqual(instance.last_applied_hash, "") | ||||||
|             self.assertEqual( |             self.assertEqual( | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """API URLs""" | """API URLs""" | ||||||
|  |  | ||||||
| from authentik.blueprints.api import BlueprintInstanceViewSet | from authentik.blueprints.api import BlueprintInstanceViewSet | ||||||
|  |  | ||||||
| api_urlpatterns = [ | api_urlpatterns = [ | ||||||
|  | |||||||
| @ -1,20 +1,17 @@ | |||||||
| """transfer common classes""" | """transfer common classes""" | ||||||
|  |  | ||||||
| from collections import OrderedDict | from collections import OrderedDict | ||||||
| from collections.abc import Iterable, Mapping |  | ||||||
| from copy import copy | from copy import copy | ||||||
| from dataclasses import asdict, dataclass, field, is_dataclass | from dataclasses import asdict, dataclass, field, is_dataclass | ||||||
| from enum import Enum | from enum import Enum | ||||||
| from functools import reduce | from functools import reduce | ||||||
| from operator import ixor | from operator import ixor | ||||||
| from os import getenv | from os import getenv | ||||||
| from typing import Any, Literal, Union | from typing import Any, Iterable, Literal, Mapping, Optional, Union | ||||||
| from uuid import UUID | from uuid import UUID | ||||||
|  |  | ||||||
| from deepmerge import always_merger | from deepmerge import always_merger | ||||||
| from django.apps import apps | from django.apps import apps | ||||||
| from django.db.models import Model, Q | from django.db.models import Model, Q | ||||||
| from rest_framework.exceptions import ValidationError |  | ||||||
| from rest_framework.fields import Field | from rest_framework.fields import Field | ||||||
| from rest_framework.serializers import Serializer | from rest_framework.serializers import Serializer | ||||||
| from yaml import SafeDumper, SafeLoader, ScalarNode, SequenceNode | from yaml import SafeDumper, SafeLoader, ScalarNode, SequenceNode | ||||||
| @ -46,7 +43,7 @@ def get_attrs(obj: SerializerModel) -> dict[str, Any]: | |||||||
| class BlueprintEntryState: | class BlueprintEntryState: | ||||||
|     """State of a single instance""" |     """State of a single instance""" | ||||||
|  |  | ||||||
|     instance: Model | None = None |     instance: Optional[Model] = None | ||||||
|  |  | ||||||
|  |  | ||||||
| class BlueprintEntryDesiredState(Enum): | class BlueprintEntryDesiredState(Enum): | ||||||
| @ -55,7 +52,6 @@ class BlueprintEntryDesiredState(Enum): | |||||||
|     ABSENT = "absent" |     ABSENT = "absent" | ||||||
|     PRESENT = "present" |     PRESENT = "present" | ||||||
|     CREATED = "created" |     CREATED = "created" | ||||||
|     MUST_CREATED = "must_created" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @dataclass | @dataclass | ||||||
| @ -68,9 +64,9 @@ class BlueprintEntry: | |||||||
|     ) |     ) | ||||||
|     conditions: list[Any] = field(default_factory=list) |     conditions: list[Any] = field(default_factory=list) | ||||||
|     identifiers: dict[str, Any] = field(default_factory=dict) |     identifiers: dict[str, Any] = field(default_factory=dict) | ||||||
|     attrs: dict[str, Any] | None = field(default_factory=dict) |     attrs: Optional[dict[str, Any]] = field(default_factory=dict) | ||||||
|  |  | ||||||
|     id: str | None = None |     id: Optional[str] = None | ||||||
|  |  | ||||||
|     _state: BlueprintEntryState = field(default_factory=BlueprintEntryState) |     _state: BlueprintEntryState = field(default_factory=BlueprintEntryState) | ||||||
|  |  | ||||||
| @ -93,10 +89,10 @@ class BlueprintEntry: | |||||||
|             attrs=all_attrs, |             attrs=all_attrs, | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|     def get_tag_context( |     def _get_tag_context( | ||||||
|         self, |         self, | ||||||
|         depth: int = 0, |         depth: int = 0, | ||||||
|         context_tag_type: type["YAMLTagContext"] | tuple["YAMLTagContext", ...] | None = None, |         context_tag_type: Optional[type["YAMLTagContext"] | tuple["YAMLTagContext", ...]] = None, | ||||||
|     ) -> "YAMLTagContext": |     ) -> "YAMLTagContext": | ||||||
|         """Get a YAMLTagContext object located at a certain depth in the tag tree""" |         """Get a YAMLTagContext object located at a certain depth in the tag tree""" | ||||||
|         if depth < 0: |         if depth < 0: | ||||||
| @ -109,8 +105,8 @@ class BlueprintEntry: | |||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             return contexts[-(depth + 1)] |             return contexts[-(depth + 1)] | ||||||
|         except IndexError as exc: |         except IndexError: | ||||||
|             raise ValueError(f"invalid depth: {depth}. Max depth: {len(contexts) - 1}") from exc |             raise ValueError(f"invalid depth: {depth}. Max depth: {len(contexts) - 1}") | ||||||
|  |  | ||||||
|     def tag_resolver(self, value: Any, blueprint: "Blueprint") -> Any: |     def tag_resolver(self, value: Any, blueprint: "Blueprint") -> Any: | ||||||
|         """Check if we have any special tags that need handling""" |         """Check if we have any special tags that need handling""" | ||||||
| @ -171,7 +167,7 @@ class Blueprint: | |||||||
|     entries: list[BlueprintEntry] = field(default_factory=list) |     entries: list[BlueprintEntry] = field(default_factory=list) | ||||||
|     context: dict = field(default_factory=dict) |     context: dict = field(default_factory=dict) | ||||||
|  |  | ||||||
|     metadata: BlueprintMetadata | None = field(default=None) |     metadata: Optional[BlueprintMetadata] = field(default=None) | ||||||
|  |  | ||||||
|  |  | ||||||
| class YAMLTag: | class YAMLTag: | ||||||
| @ -210,8 +206,8 @@ class KeyOf(YAMLTag): | |||||||
|                 ): |                 ): | ||||||
|                     return _entry._state.instance.pbm_uuid |                     return _entry._state.instance.pbm_uuid | ||||||
|                 return _entry._state.instance.pk |                 return _entry._state.instance.pk | ||||||
|         raise EntryInvalidError.from_entry( |         raise EntryInvalidError( | ||||||
|             f"KeyOf: failed to find entry with `id` of `{self.id_from}` and a model instance", entry |             f"KeyOf: failed to find entry with `id` of `{self.id_from}` and a model instance" | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -219,7 +215,7 @@ class Env(YAMLTag): | |||||||
|     """Lookup environment variable with optional default""" |     """Lookup environment variable with optional default""" | ||||||
|  |  | ||||||
|     key: str |     key: str | ||||||
|     default: Any | None |     default: Optional[Any] | ||||||
|  |  | ||||||
|     def __init__(self, loader: "BlueprintLoader", node: ScalarNode | SequenceNode) -> None: |     def __init__(self, loader: "BlueprintLoader", node: ScalarNode | SequenceNode) -> None: | ||||||
|         super().__init__() |         super().__init__() | ||||||
| @ -238,7 +234,7 @@ class Context(YAMLTag): | |||||||
|     """Lookup key from instance context""" |     """Lookup key from instance context""" | ||||||
|  |  | ||||||
|     key: str |     key: str | ||||||
|     default: Any | None |     default: Optional[Any] | ||||||
|  |  | ||||||
|     def __init__(self, loader: "BlueprintLoader", node: ScalarNode | SequenceNode) -> None: |     def __init__(self, loader: "BlueprintLoader", node: ScalarNode | SequenceNode) -> None: | ||||||
|         super().__init__() |         super().__init__() | ||||||
| @ -282,7 +278,7 @@ class Format(YAMLTag): | |||||||
|         try: |         try: | ||||||
|             return self.format_string % tuple(args) |             return self.format_string % tuple(args) | ||||||
|         except TypeError as exc: |         except TypeError as exc: | ||||||
|             raise EntryInvalidError.from_entry(exc, entry) from exc |             raise EntryInvalidError(exc) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Find(YAMLTag): | class Find(YAMLTag): | ||||||
| @ -359,15 +355,13 @@ class Condition(YAMLTag): | |||||||
|                 args.append(arg) |                 args.append(arg) | ||||||
|  |  | ||||||
|         if not args: |         if not args: | ||||||
|             raise EntryInvalidError.from_entry( |             raise EntryInvalidError("At least one value is required after mode selection.") | ||||||
|                 "At least one value is required after mode selection.", entry |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             comparator = self._COMPARATORS[self.mode.upper()] |             comparator = self._COMPARATORS[self.mode.upper()] | ||||||
|             return comparator(tuple(bool(x) for x in args)) |             return comparator(tuple(bool(x) for x in args)) | ||||||
|         except (TypeError, KeyError) as exc: |         except (TypeError, KeyError) as exc: | ||||||
|             raise EntryInvalidError.from_entry(exc, entry) from exc |             raise EntryInvalidError(exc) | ||||||
|  |  | ||||||
|  |  | ||||||
| class If(YAMLTag): | class If(YAMLTag): | ||||||
| @ -399,7 +393,7 @@ class If(YAMLTag): | |||||||
|                 blueprint, |                 blueprint, | ||||||
|             ) |             ) | ||||||
|         except TypeError as exc: |         except TypeError as exc: | ||||||
|             raise EntryInvalidError.from_entry(exc, entry) from exc |             raise EntryInvalidError(exc) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Enumerate(YAMLTag, YAMLTagContext): | class Enumerate(YAMLTag, YAMLTagContext): | ||||||
| @ -413,7 +407,9 @@ class Enumerate(YAMLTag, YAMLTagContext): | |||||||
|         "SEQ": (list, lambda a, b: [*a, b]), |         "SEQ": (list, lambda a, b: [*a, b]), | ||||||
|         "MAP": ( |         "MAP": ( | ||||||
|             dict, |             dict, | ||||||
|             lambda a, b: always_merger.merge(a, {b[0]: b[1]} if isinstance(b, tuple | list) else b), |             lambda a, b: always_merger.merge( | ||||||
|  |                 a, {b[0]: b[1]} if isinstance(b, (tuple, list)) else b | ||||||
|  |             ), | ||||||
|         ), |         ), | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @ -429,10 +425,9 @@ class Enumerate(YAMLTag, YAMLTagContext): | |||||||
|  |  | ||||||
|     def resolve(self, entry: BlueprintEntry, blueprint: Blueprint) -> Any: |     def resolve(self, entry: BlueprintEntry, blueprint: Blueprint) -> Any: | ||||||
|         if isinstance(self.iterable, EnumeratedItem) and self.iterable.depth == 0: |         if isinstance(self.iterable, EnumeratedItem) and self.iterable.depth == 0: | ||||||
|             raise EntryInvalidError.from_entry( |             raise EntryInvalidError( | ||||||
|                 f"{self.__class__.__name__} tag's iterable references this tag's context. " |                 f"{self.__class__.__name__} tag's iterable references this tag's context. " | ||||||
|                 "This is a noop. Check you are setting depth bigger than 0.", |                 "This is a noop. Check you are setting depth bigger than 0." | ||||||
|                 entry, |  | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         if isinstance(self.iterable, YAMLTag): |         if isinstance(self.iterable, YAMLTag): | ||||||
| @ -441,10 +436,9 @@ class Enumerate(YAMLTag, YAMLTagContext): | |||||||
|             iterable = self.iterable |             iterable = self.iterable | ||||||
|  |  | ||||||
|         if not isinstance(iterable, Iterable): |         if not isinstance(iterable, Iterable): | ||||||
|             raise EntryInvalidError.from_entry( |             raise EntryInvalidError( | ||||||
|                 f"{self.__class__.__name__}'s iterable must be an iterable " |                 f"{self.__class__.__name__}'s iterable must be an iterable " | ||||||
|                 "such as a sequence or a mapping", |                 "such as a sequence or a mapping" | ||||||
|                 entry, |  | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|         if isinstance(iterable, Mapping): |         if isinstance(iterable, Mapping): | ||||||
| @ -455,7 +449,7 @@ class Enumerate(YAMLTag, YAMLTagContext): | |||||||
|         try: |         try: | ||||||
|             output_class, add_fn = self._OUTPUT_BODIES[self.output_body.upper()] |             output_class, add_fn = self._OUTPUT_BODIES[self.output_body.upper()] | ||||||
|         except KeyError as exc: |         except KeyError as exc: | ||||||
|             raise EntryInvalidError.from_entry(exc, entry) from exc |             raise EntryInvalidError(exc) | ||||||
|  |  | ||||||
|         result = output_class() |         result = output_class() | ||||||
|  |  | ||||||
| @ -467,8 +461,8 @@ class Enumerate(YAMLTag, YAMLTagContext): | |||||||
|                 resolved_body = entry.tag_resolver(self.item_body, blueprint) |                 resolved_body = entry.tag_resolver(self.item_body, blueprint) | ||||||
|                 result = add_fn(result, resolved_body) |                 result = add_fn(result, resolved_body) | ||||||
|                 if not isinstance(result, output_class): |                 if not isinstance(result, output_class): | ||||||
|                     raise EntryInvalidError.from_entry( |                     raise EntryInvalidError( | ||||||
|                         f"Invalid {self.__class__.__name__} item found: {resolved_body}", entry |                         f"Invalid {self.__class__.__name__} item found: {resolved_body}" | ||||||
|                     ) |                     ) | ||||||
|         finally: |         finally: | ||||||
|             self.__current_context = tuple() |             self.__current_context = tuple() | ||||||
| @ -483,27 +477,24 @@ class EnumeratedItem(YAMLTag): | |||||||
|  |  | ||||||
|     _SUPPORTED_CONTEXT_TAGS = (Enumerate,) |     _SUPPORTED_CONTEXT_TAGS = (Enumerate,) | ||||||
|  |  | ||||||
|     def __init__(self, _loader: "BlueprintLoader", node: ScalarNode) -> None: |     def __init__(self, loader: "BlueprintLoader", node: ScalarNode) -> None: | ||||||
|         super().__init__() |         super().__init__() | ||||||
|         self.depth = int(node.value) |         self.depth = int(node.value) | ||||||
|  |  | ||||||
|     def resolve(self, entry: BlueprintEntry, blueprint: Blueprint) -> Any: |     def resolve(self, entry: BlueprintEntry, blueprint: Blueprint) -> Any: | ||||||
|         try: |         try: | ||||||
|             context_tag: Enumerate = entry.get_tag_context( |             context_tag: Enumerate = entry._get_tag_context( | ||||||
|                 depth=self.depth, |                 depth=self.depth, | ||||||
|                 context_tag_type=EnumeratedItem._SUPPORTED_CONTEXT_TAGS, |                 context_tag_type=EnumeratedItem._SUPPORTED_CONTEXT_TAGS, | ||||||
|             ) |             ) | ||||||
|         except ValueError as exc: |         except ValueError as exc: | ||||||
|             if self.depth == 0: |             if self.depth == 0: | ||||||
|                 raise EntryInvalidError.from_entry( |                 raise EntryInvalidError( | ||||||
|                     f"{self.__class__.__name__} tags are only usable " |                     f"{self.__class__.__name__} tags are only usable " | ||||||
|                     f"inside an {Enumerate.__name__} tag", |                     f"inside an {Enumerate.__name__} tag" | ||||||
|                     entry, |                 ) | ||||||
|                 ) from exc |  | ||||||
|  |  | ||||||
|             raise EntryInvalidError.from_entry( |             raise EntryInvalidError(f"{self.__class__.__name__} tag: {exc}") | ||||||
|                 f"{self.__class__.__name__} tag: {exc}", entry |  | ||||||
|             ) from exc |  | ||||||
|  |  | ||||||
|         return context_tag.get_context(entry, blueprint) |         return context_tag.get_context(entry, blueprint) | ||||||
|  |  | ||||||
| @ -516,8 +507,8 @@ class Index(EnumeratedItem): | |||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             return context[0] |             return context[0] | ||||||
|         except IndexError as exc:  # pragma: no cover |         except IndexError:  # pragma: no cover | ||||||
|             raise EntryInvalidError.from_entry(f"Empty/invalid context: {context}", entry) from exc |             raise EntryInvalidError(f"Empty/invalid context: {context}") | ||||||
|  |  | ||||||
|  |  | ||||||
| class Value(EnumeratedItem): | class Value(EnumeratedItem): | ||||||
| @ -528,8 +519,8 @@ class Value(EnumeratedItem): | |||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             return context[1] |             return context[1] | ||||||
|         except IndexError as exc:  # pragma: no cover |         except IndexError:  # pragma: no cover | ||||||
|             raise EntryInvalidError.from_entry(f"Empty/invalid context: {context}", entry) from exc |             raise EntryInvalidError(f"Empty/invalid context: {context}") | ||||||
|  |  | ||||||
|  |  | ||||||
| class BlueprintDumper(SafeDumper): | class BlueprintDumper(SafeDumper): | ||||||
| @ -583,31 +574,8 @@ class BlueprintLoader(SafeLoader): | |||||||
| class EntryInvalidError(SentryIgnoredException): | class EntryInvalidError(SentryIgnoredException): | ||||||
|     """Error raised when an entry is invalid""" |     """Error raised when an entry is invalid""" | ||||||
|  |  | ||||||
|     entry_model: str | None |     serializer_errors: Optional[dict] | ||||||
|     entry_id: str | None |  | ||||||
|     validation_error: ValidationError | None |  | ||||||
|     serializer: Serializer | None = None |  | ||||||
|  |  | ||||||
|     def __init__( |     def __init__(self, *args: object, serializer_errors: Optional[dict] = None) -> None: | ||||||
|         self, *args: object, validation_error: ValidationError | None = None, **kwargs |  | ||||||
|     ) -> None: |  | ||||||
|         super().__init__(*args) |         super().__init__(*args) | ||||||
|         self.entry_model = None |         self.serializer_errors = serializer_errors | ||||||
|         self.entry_id = None |  | ||||||
|         self.validation_error = validation_error |  | ||||||
|         for key, value in kwargs.items(): |  | ||||||
|             setattr(self, key, value) |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def from_entry( |  | ||||||
|         msg_or_exc: str | Exception, entry: BlueprintEntry, *args, **kwargs |  | ||||||
|     ) -> "EntryInvalidError": |  | ||||||
|         """Create EntryInvalidError with the context of an entry""" |  | ||||||
|         error = EntryInvalidError(msg_or_exc, *args, **kwargs) |  | ||||||
|         if isinstance(msg_or_exc, ValidationError): |  | ||||||
|             error.validation_error = msg_or_exc |  | ||||||
|         # Make sure the model and id are strings, depending where the error happens |  | ||||||
|         # they might still be YAMLTag instances |  | ||||||
|         error.entry_model = str(entry.model) |  | ||||||
|         error.entry_id = str(entry.id) |  | ||||||
|         return error |  | ||||||
|  | |||||||
| @ -1,6 +1,5 @@ | |||||||
| """Blueprint exporter""" | """Blueprint exporter""" | ||||||
|  | from typing import Iterable | ||||||
| from collections.abc import Iterable |  | ||||||
| from uuid import UUID | from uuid import UUID | ||||||
|  |  | ||||||
| from django.apps import apps | from django.apps import apps | ||||||
| @ -8,6 +7,7 @@ from django.contrib.auth import get_user_model | |||||||
| from django.db.models import Model, Q, QuerySet | from django.db.models import Model, Q, QuerySet | ||||||
| from django.utils.timezone import now | from django.utils.timezone import now | ||||||
| from django.utils.translation import gettext as _ | from django.utils.translation import gettext as _ | ||||||
|  | from guardian.shortcuts import get_anonymous_user | ||||||
| from yaml import dump | from yaml import dump | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.common import ( | from authentik.blueprints.v1.common import ( | ||||||
| @ -48,7 +48,7 @@ class Exporter: | |||||||
|         """Return a queryset for `model`. Can be used to filter some |         """Return a queryset for `model`. Can be used to filter some | ||||||
|         objects on some models""" |         objects on some models""" | ||||||
|         if model == get_user_model(): |         if model == get_user_model(): | ||||||
|             return model.objects.exclude_anonymous() |             return model.objects.exclude(pk=get_anonymous_user().pk) | ||||||
|         return model.objects.all() |         return model.objects.all() | ||||||
|  |  | ||||||
|     def _pre_export(self, blueprint: Blueprint): |     def _pre_export(self, blueprint: Blueprint): | ||||||
| @ -59,7 +59,7 @@ class Exporter: | |||||||
|         blueprint = Blueprint() |         blueprint = Blueprint() | ||||||
|         self._pre_export(blueprint) |         self._pre_export(blueprint) | ||||||
|         blueprint.metadata = BlueprintMetadata( |         blueprint.metadata = BlueprintMetadata( | ||||||
|             name=_("authentik Export - {date}".format_map({"date": str(now())})), |             name=_("authentik Export - %(date)s" % {"date": str(now())}), | ||||||
|             labels={ |             labels={ | ||||||
|                 LABEL_AUTHENTIK_GENERATED: "true", |                 LABEL_AUTHENTIK_GENERATED: "true", | ||||||
|             }, |             }, | ||||||
| @ -74,7 +74,7 @@ class Exporter: | |||||||
|  |  | ||||||
|  |  | ||||||
| class FlowExporter(Exporter): | class FlowExporter(Exporter): | ||||||
|     """Exporter customized to only return objects related to `flow`""" |     """Exporter customised to only return objects related to `flow`""" | ||||||
|  |  | ||||||
|     flow: Flow |     flow: Flow | ||||||
|     with_policies: bool |     with_policies: bool | ||||||
|  | |||||||
| @ -1,21 +1,17 @@ | |||||||
| """Blueprint importer""" | """Blueprint importer""" | ||||||
|  |  | ||||||
| from contextlib import contextmanager | from contextlib import contextmanager | ||||||
| from copy import deepcopy | from copy import deepcopy | ||||||
| from typing import Any | from typing import Any, Optional | ||||||
|  |  | ||||||
| from dacite.config import Config | from dacite.config import Config | ||||||
| from dacite.core import from_dict | from dacite.core import from_dict | ||||||
| from dacite.exceptions import DaciteError | from dacite.exceptions import DaciteError | ||||||
| from deepmerge import always_merger | from deepmerge import always_merger | ||||||
| from django.contrib.auth.models import Permission |  | ||||||
| from django.contrib.contenttypes.models import ContentType |  | ||||||
| from django.core.exceptions import FieldError | from django.core.exceptions import FieldError | ||||||
|  | from django.db import transaction | ||||||
| from django.db.models import Model | from django.db.models import Model | ||||||
| from django.db.models.query_utils import Q | from django.db.models.query_utils import Q | ||||||
| from django.db.transaction import atomic |  | ||||||
| from django.db.utils import IntegrityError | from django.db.utils import IntegrityError | ||||||
| from guardian.models import UserObjectPermission |  | ||||||
| from rest_framework.exceptions import ValidationError | from rest_framework.exceptions import ValidationError | ||||||
| from rest_framework.serializers import BaseSerializer, Serializer | from rest_framework.serializers import BaseSerializer, Serializer | ||||||
| from structlog.stdlib import BoundLogger, get_logger | from structlog.stdlib import BoundLogger, get_logger | ||||||
| @ -39,40 +35,26 @@ from authentik.core.models import ( | |||||||
|     Source, |     Source, | ||||||
|     UserSourceConnection, |     UserSourceConnection, | ||||||
| ) | ) | ||||||
| from authentik.enterprise.license import LicenseKey |  | ||||||
| from authentik.enterprise.models import LicenseUsage |  | ||||||
| from authentik.enterprise.providers.rac.models import ConnectionToken |  | ||||||
| from authentik.events.models import SystemTask |  | ||||||
| from authentik.events.utils import cleanse_dict | from authentik.events.utils import cleanse_dict | ||||||
| from authentik.flows.models import FlowToken, Stage | from authentik.flows.models import FlowToken, Stage | ||||||
| from authentik.lib.models import SerializerModel | from authentik.lib.models import SerializerModel | ||||||
| from authentik.lib.sentry import SentryIgnoredException |  | ||||||
| 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.providers.oauth2.models import AccessToken, AuthorizationCode, RefreshToken |  | ||||||
| from authentik.providers.scim.models import SCIMGroup, SCIMUser |  | ||||||
| from authentik.tenants.models import Tenant |  | ||||||
|  |  | ||||||
| # Context set when the serializer is created in a blueprint context | # Context set when the serializer is created in a blueprint context | ||||||
| # Update website/developer-docs/blueprints/v1/models.md when used | # Update website/developer-docs/blueprints/v1/models.md when used | ||||||
| SERIALIZER_CONTEXT_BLUEPRINT = "blueprint_entry" | SERIALIZER_CONTEXT_BLUEPRINT = "blueprint_entry" | ||||||
|  |  | ||||||
|  |  | ||||||
| def excluded_models() -> list[type[Model]]: | def is_model_allowed(model: type[Model]) -> bool: | ||||||
|     """Return a list of all excluded models that shouldn't be exposed via API |     """Check if model is allowed""" | ||||||
|     or other means (internal only, base classes, non-used objects, etc)""" |     # pylint: disable=imported-auth-user | ||||||
|  |  | ||||||
|     from django.contrib.auth.models import Group as DjangoGroup |     from django.contrib.auth.models import Group as DjangoGroup | ||||||
|     from django.contrib.auth.models import User as DjangoUser |     from django.contrib.auth.models import User as DjangoUser | ||||||
|  |  | ||||||
|     return ( |     excluded_models = ( | ||||||
|         # Django only classes |  | ||||||
|         DjangoUser, |         DjangoUser, | ||||||
|         DjangoGroup, |         DjangoGroup, | ||||||
|         ContentType, |  | ||||||
|         Permission, |  | ||||||
|         UserObjectPermission, |  | ||||||
|         # Base classes |         # Base classes | ||||||
|         Provider, |         Provider, | ||||||
|         Source, |         Source, | ||||||
| @ -86,75 +68,45 @@ def excluded_models() -> list[type[Model]]: | |||||||
|         AuthenticatedSession, |         AuthenticatedSession, | ||||||
|         # Classes which are only internally managed |         # Classes which are only internally managed | ||||||
|         FlowToken, |         FlowToken, | ||||||
|         LicenseUsage, |  | ||||||
|         SCIMGroup, |  | ||||||
|         SCIMUser, |  | ||||||
|         Tenant, |  | ||||||
|         SystemTask, |  | ||||||
|         ConnectionToken, |  | ||||||
|         AuthorizationCode, |  | ||||||
|         AccessToken, |  | ||||||
|         RefreshToken, |  | ||||||
|         Reputation, |  | ||||||
|     ) |     ) | ||||||
|  |     return model not in excluded_models and issubclass(model, (SerializerModel, BaseMetaModel)) | ||||||
|  |  | ||||||
| def is_model_allowed(model: type[Model]) -> bool: |  | ||||||
|     """Check if model is allowed""" |  | ||||||
|     return model not in excluded_models() and issubclass(model, SerializerModel | BaseMetaModel) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class DoRollback(SentryIgnoredException): |  | ||||||
|     """Exception to trigger a rollback""" |  | ||||||
|  |  | ||||||
|  |  | ||||||
| @contextmanager | @contextmanager | ||||||
| def transaction_rollback(): | def transaction_rollback(): | ||||||
|     """Enters an atomic transaction and always triggers a rollback at the end of the block.""" |     """Enters an atomic transaction and always triggers a rollback at the end of the block.""" | ||||||
|     try: |     atomic = transaction.atomic() | ||||||
|         with atomic(): |     # pylint: disable=unnecessary-dunder-call | ||||||
|  |     atomic.__enter__() | ||||||
|     yield |     yield | ||||||
|             raise DoRollback() |     atomic.__exit__(IntegrityError, None, None) | ||||||
|     except DoRollback: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Importer: | class Importer: | ||||||
|     """Import Blueprint from raw dict or YAML/JSON""" |     """Import Blueprint from YAML""" | ||||||
|  |  | ||||||
|     logger: BoundLogger |     logger: BoundLogger | ||||||
|     _import: Blueprint |  | ||||||
|  |  | ||||||
|     def __init__(self, blueprint: Blueprint, context: dict | None = None): |     def __init__(self, yaml_input: str, context: Optional[dict] = None): | ||||||
|         self.__pk_map: dict[Any, Model] = {} |         self.__pk_map: dict[Any, Model] = {} | ||||||
|         self._import = blueprint |  | ||||||
|         self.logger = get_logger() |         self.logger = get_logger() | ||||||
|         ctx = self.default_context() |  | ||||||
|         always_merger.merge(ctx, self._import.context) |  | ||||||
|         if context: |  | ||||||
|             always_merger.merge(ctx, context) |  | ||||||
|         self._import.context = ctx |  | ||||||
|  |  | ||||||
|     def default_context(self): |  | ||||||
|         """Default context""" |  | ||||||
|         return {"goauthentik.io/enterprise/licensed": LicenseKey.get_total().is_valid()} |  | ||||||
|  |  | ||||||
|     @staticmethod |  | ||||||
|     def from_string(yaml_input: str, context: dict | None = None) -> "Importer": |  | ||||||
|         """Parse YAML string and create blueprint importer from it""" |  | ||||||
|         import_dict = load(yaml_input, BlueprintLoader) |         import_dict = load(yaml_input, BlueprintLoader) | ||||||
|         try: |         try: | ||||||
|             _import = from_dict( |             self.__import = from_dict( | ||||||
|                 Blueprint, import_dict, config=Config(cast=[BlueprintEntryDesiredState]) |                 Blueprint, import_dict, config=Config(cast=[BlueprintEntryDesiredState]) | ||||||
|             ) |             ) | ||||||
|         except DaciteError as exc: |         except DaciteError as exc: | ||||||
|             raise EntryInvalidError from exc |             raise EntryInvalidError from exc | ||||||
|         return Importer(_import, context) |         ctx = {} | ||||||
|  |         always_merger.merge(ctx, self.__import.context) | ||||||
|  |         if context: | ||||||
|  |             always_merger.merge(ctx, context) | ||||||
|  |         self.__import.context = ctx | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def blueprint(self) -> Blueprint: |     def blueprint(self) -> Blueprint: | ||||||
|         """Get imported blueprint""" |         """Get imported blueprint""" | ||||||
|         return self._import |         return self.__import | ||||||
|  |  | ||||||
|     def __update_pks_for_attrs(self, attrs: dict[str, Any]) -> dict[str, Any]: |     def __update_pks_for_attrs(self, attrs: dict[str, Any]) -> dict[str, Any]: | ||||||
|         """Replace any value if it is a known primary key of an other object""" |         """Replace any value if it is a known primary key of an other object""" | ||||||
| @ -168,7 +120,7 @@ class Importer: | |||||||
|         for key, value in attrs.items(): |         for key, value in attrs.items(): | ||||||
|             try: |             try: | ||||||
|                 if isinstance(value, dict): |                 if isinstance(value, dict): | ||||||
|                     for _, _inner_key in enumerate(value): |                     for idx, _inner_key in enumerate(value): | ||||||
|                         value[_inner_key] = updater(value[_inner_key]) |                         value[_inner_key] = updater(value[_inner_key]) | ||||||
|                 elif isinstance(value, list): |                 elif isinstance(value, list): | ||||||
|                     for idx, _inner_value in enumerate(value): |                     for idx, _inner_value in enumerate(value): | ||||||
| @ -197,21 +149,22 @@ class Importer: | |||||||
|  |  | ||||||
|         return main_query | sub_query |         return main_query | sub_query | ||||||
|  |  | ||||||
|     def _validate_single(self, entry: BlueprintEntry) -> BaseSerializer | None: |     # pylint: disable-msg=too-many-locals | ||||||
|  |     def _validate_single(self, entry: BlueprintEntry) -> Optional[BaseSerializer]: | ||||||
|         """Validate a single entry""" |         """Validate a single entry""" | ||||||
|         if not entry.check_all_conditions_match(self._import): |         if not entry.check_all_conditions_match(self.__import): | ||||||
|             self.logger.debug("One or more conditions of this entry are not fulfilled, skipping") |             self.logger.debug("One or more conditions of this entry are not fulfilled, skipping") | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         model_app_label, model_name = entry.get_model(self._import).split(".") |         model_app_label, model_name = entry.get_model(self.__import).split(".") | ||||||
|         model: type[SerializerModel] = registry.get_model(model_app_label, model_name) |         model: type[SerializerModel] = registry.get_model(model_app_label, model_name) | ||||||
|         # Don't use isinstance since we don't want to check for inheritance |         # Don't use isinstance since we don't want to check for inheritance | ||||||
|         if not is_model_allowed(model): |         if not is_model_allowed(model): | ||||||
|             raise EntryInvalidError.from_entry(f"Model {model} not allowed", entry) |             raise EntryInvalidError(f"Model {model} not allowed") | ||||||
|         if issubclass(model, BaseMetaModel): |         if issubclass(model, BaseMetaModel): | ||||||
|             serializer_class: type[Serializer] = model.serializer() |             serializer_class: type[Serializer] = model.serializer() | ||||||
|             serializer = serializer_class( |             serializer = serializer_class( | ||||||
|                 data=entry.get_attrs(self._import), |                 data=entry.get_attrs(self.__import), | ||||||
|                 context={ |                 context={ | ||||||
|                     SERIALIZER_CONTEXT_BLUEPRINT: entry, |                     SERIALIZER_CONTEXT_BLUEPRINT: entry, | ||||||
|                 }, |                 }, | ||||||
| @ -219,10 +172,8 @@ class Importer: | |||||||
|             try: |             try: | ||||||
|                 serializer.is_valid(raise_exception=True) |                 serializer.is_valid(raise_exception=True) | ||||||
|             except ValidationError as exc: |             except ValidationError as exc: | ||||||
|                 raise EntryInvalidError.from_entry( |                 raise EntryInvalidError( | ||||||
|                     f"Serializer errors {serializer.errors}", |                     f"Serializer errors {serializer.errors}", serializer_errors=serializer.errors | ||||||
|                     validation_error=exc, |  | ||||||
|                     entry=entry, |  | ||||||
|                 ) from exc |                 ) from exc | ||||||
|             return serializer |             return serializer | ||||||
|  |  | ||||||
| @ -231,7 +182,7 @@ class Importer: | |||||||
|         # the full serializer for later usage |         # the full serializer for later usage | ||||||
|         # Because a model might have multiple unique columns, we chain all identifiers together |         # Because a model might have multiple unique columns, we chain all identifiers together | ||||||
|         # to create an OR query. |         # to create an OR query. | ||||||
|         updated_identifiers = self.__update_pks_for_attrs(entry.get_identifiers(self._import)) |         updated_identifiers = self.__update_pks_for_attrs(entry.get_identifiers(self.__import)) | ||||||
|         for key, value in list(updated_identifiers.items()): |         for key, value in list(updated_identifiers.items()): | ||||||
|             if isinstance(value, dict) and "pk" in value: |             if isinstance(value, dict) and "pk" in value: | ||||||
|                 del updated_identifiers[key] |                 del updated_identifiers[key] | ||||||
| @ -239,12 +190,12 @@ class Importer: | |||||||
|  |  | ||||||
|         query = self.__query_from_identifier(updated_identifiers) |         query = self.__query_from_identifier(updated_identifiers) | ||||||
|         if not query: |         if not query: | ||||||
|             raise EntryInvalidError.from_entry("No or invalid identifiers", entry) |             raise EntryInvalidError("No or invalid identifiers") | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             existing_models = model.objects.filter(query) |             existing_models = model.objects.filter(query) | ||||||
|         except FieldError as exc: |         except FieldError as exc: | ||||||
|             raise EntryInvalidError.from_entry(f"Invalid identifier field: {exc}", entry) from exc |             raise EntryInvalidError(f"Invalid identifier field: {exc}") from exc | ||||||
|  |  | ||||||
|         serializer_kwargs = {} |         serializer_kwargs = {} | ||||||
|         model_instance = existing_models.first() |         model_instance = existing_models.first() | ||||||
| @ -257,14 +208,6 @@ 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: |  | ||||||
|             raise EntryInvalidError.from_entry( |  | ||||||
|                 ( |  | ||||||
|                     f"state is set to {BlueprintEntryDesiredState.MUST_CREATED} " |  | ||||||
|                     "and object exists already", |  | ||||||
|                 ), |  | ||||||
|                 entry, |  | ||||||
|             ) |  | ||||||
|         else: |         else: | ||||||
|             self.logger.debug( |             self.logger.debug( | ||||||
|                 "initialised new serializer instance", |                 "initialised new serializer instance", | ||||||
| @ -277,12 +220,9 @@ class Importer: | |||||||
|                 model_instance.pk = updated_identifiers["pk"] |                 model_instance.pk = updated_identifiers["pk"] | ||||||
|             serializer_kwargs["instance"] = model_instance |             serializer_kwargs["instance"] = model_instance | ||||||
|         try: |         try: | ||||||
|             full_data = self.__update_pks_for_attrs(entry.get_attrs(self._import)) |             full_data = self.__update_pks_for_attrs(entry.get_attrs(self.__import)) | ||||||
|         except ValueError as exc: |         except ValueError as exc: | ||||||
|             raise EntryInvalidError.from_entry( |             raise EntryInvalidError(exc) from exc | ||||||
|                 exc, |  | ||||||
|                 entry, |  | ||||||
|             ) from exc |  | ||||||
|         always_merger.merge(full_data, updated_identifiers) |         always_merger.merge(full_data, updated_identifiers) | ||||||
|         serializer_kwargs["data"] = full_data |         serializer_kwargs["data"] = full_data | ||||||
|  |  | ||||||
| @ -295,18 +235,15 @@ class Importer: | |||||||
|         try: |         try: | ||||||
|             serializer.is_valid(raise_exception=True) |             serializer.is_valid(raise_exception=True) | ||||||
|         except ValidationError as exc: |         except ValidationError as exc: | ||||||
|             raise EntryInvalidError.from_entry( |             raise EntryInvalidError( | ||||||
|                 f"Serializer errors {serializer.errors}", |                 f"Serializer errors {serializer.errors}", serializer_errors=serializer.errors | ||||||
|                 validation_error=exc, |  | ||||||
|                 entry=entry, |  | ||||||
|                 serializer=serializer, |  | ||||||
|             ) from exc |             ) from exc | ||||||
|         return serializer |         return serializer | ||||||
|  |  | ||||||
|     def apply(self) -> bool: |     def apply(self) -> bool: | ||||||
|         """Apply (create/update) models yaml, in database transaction""" |         """Apply (create/update) models yaml, in database transaction""" | ||||||
|         try: |         try: | ||||||
|             with atomic(): |             with transaction.atomic(): | ||||||
|                 if not self._apply_models(): |                 if not self._apply_models(): | ||||||
|                     self.logger.debug("Reverting changes due to error") |                     self.logger.debug("Reverting changes due to error") | ||||||
|                     raise IntegrityError |                     raise IntegrityError | ||||||
| @ -315,11 +252,11 @@ class Importer: | |||||||
|         self.logger.debug("Committing changes") |         self.logger.debug("Committing changes") | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|     def _apply_models(self, raise_errors=False) -> bool: |     def _apply_models(self) -> bool: | ||||||
|         """Apply (create/update) models yaml""" |         """Apply (create/update) models yaml""" | ||||||
|         self.__pk_map = {} |         self.__pk_map = {} | ||||||
|         for entry in self._import.entries: |         for entry in self.__import.entries: | ||||||
|             model_app_label, model_name = entry.get_model(self._import).split(".") |             model_app_label, model_name = entry.get_model(self.__import).split(".") | ||||||
|             try: |             try: | ||||||
|                 model: type[SerializerModel] = registry.get_model(model_app_label, model_name) |                 model: type[SerializerModel] = registry.get_model(model_app_label, model_name) | ||||||
|             except LookupError: |             except LookupError: | ||||||
| @ -328,27 +265,19 @@ class Importer: | |||||||
|                 ) |                 ) | ||||||
|                 return False |                 return False | ||||||
|             # Validate each single entry |             # Validate each single entry | ||||||
|             serializer = None |  | ||||||
|             try: |             try: | ||||||
|                 serializer = self._validate_single(entry) |                 serializer = self._validate_single(entry) | ||||||
|             except EntryInvalidError as exc: |             except EntryInvalidError as exc: | ||||||
|                 # For deleting objects we don't need the serializer to be valid |                 # For deleting objects we don't need the serializer to be valid | ||||||
|                 if entry.get_state(self._import) == BlueprintEntryDesiredState.ABSENT: |                 if entry.get_state(self.__import) == BlueprintEntryDesiredState.ABSENT: | ||||||
|                     serializer = exc.serializer |                     continue | ||||||
|                 else: |  | ||||||
|                 self.logger.warning(f"entry invalid: {exc}", entry=entry, error=exc) |                 self.logger.warning(f"entry invalid: {exc}", entry=entry, error=exc) | ||||||
|                     if raise_errors: |  | ||||||
|                         raise exc |  | ||||||
|                 return False |                 return False | ||||||
|             if not serializer: |             if not serializer: | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             state = entry.get_state(self._import) |             state = entry.get_state(self.__import) | ||||||
|             if state in [ |             if state in [BlueprintEntryDesiredState.PRESENT, BlueprintEntryDesiredState.CREATED]: | ||||||
|                 BlueprintEntryDesiredState.PRESENT, |  | ||||||
|                 BlueprintEntryDesiredState.CREATED, |  | ||||||
|                 BlueprintEntryDesiredState.MUST_CREATED, |  | ||||||
|             ]: |  | ||||||
|                 instance = serializer.instance |                 instance = serializer.instance | ||||||
|                 if ( |                 if ( | ||||||
|                     instance |                     instance | ||||||
| @ -368,7 +297,7 @@ class Importer: | |||||||
|                     self.__pk_map[entry.identifiers["pk"]] = instance.pk |                     self.__pk_map[entry.identifiers["pk"]] = instance.pk | ||||||
|                 entry._state = BlueprintEntryState(instance) |                 entry._state = BlueprintEntryState(instance) | ||||||
|             elif state == BlueprintEntryDesiredState.ABSENT: |             elif state == BlueprintEntryDesiredState.ABSENT: | ||||||
|                 instance: Model | None = serializer.instance |                 instance: Optional[Model] = serializer.instance | ||||||
|                 if instance.pk: |                 if instance.pk: | ||||||
|                     instance.delete() |                     instance.delete() | ||||||
|                     self.logger.debug("deleted model", mode=instance) |                     self.logger.debug("deleted model", mode=instance) | ||||||
| @ -376,23 +305,23 @@ class Importer: | |||||||
|                 self.logger.debug("entry to delete with no instance, skipping") |                 self.logger.debug("entry to delete with no instance, skipping") | ||||||
|         return True |         return True | ||||||
|  |  | ||||||
|     def validate(self, raise_validation_errors=False) -> tuple[bool, list[EventDict]]: |     def validate(self) -> tuple[bool, list[EventDict]]: | ||||||
|         """Validate loaded blueprint export, ensure all models are allowed |         """Validate loaded blueprint export, ensure all models are allowed | ||||||
|         and serializers have no errors""" |         and serializers have no errors""" | ||||||
|         self.logger.debug("Starting blueprint import validation") |         self.logger.debug("Starting blueprint import validation") | ||||||
|         orig_import = deepcopy(self._import) |         orig_import = deepcopy(self.__import) | ||||||
|         if self._import.version != 1: |         if self.__import.version != 1: | ||||||
|             self.logger.warning("Invalid blueprint version") |             self.logger.warning("Invalid blueprint version") | ||||||
|             return False, [{"event": "Invalid blueprint version"}] |             return False, [{"event": "Invalid blueprint version"}] | ||||||
|         with ( |         with ( | ||||||
|             transaction_rollback(), |             transaction_rollback(), | ||||||
|             capture_logs() as logs, |             capture_logs() as logs, | ||||||
|         ): |         ): | ||||||
|             successful = self._apply_models(raise_errors=raise_validation_errors) |             successful = self._apply_models() | ||||||
|             if not successful: |             if not successful: | ||||||
|                 self.logger.debug("Blueprint validation failed") |                 self.logger.debug("Blueprint validation failed") | ||||||
|         for log in logs: |         for log in logs: | ||||||
|             getattr(self.logger, log.get("log_level"))(**log) |             getattr(self.logger, log.get("log_level"))(**log) | ||||||
|         self.logger.debug("Finished blueprint import validation") |         self.logger.debug("Finished blueprint import validation") | ||||||
|         self._import = orig_import |         self.__import = orig_import | ||||||
|         return successful, logs |         return successful, logs | ||||||
|  | |||||||
| @ -1,13 +1,12 @@ | |||||||
| """Apply Blueprint meta model""" | """Apply Blueprint meta model""" | ||||||
|  |  | ||||||
| from typing import TYPE_CHECKING | from typing import TYPE_CHECKING | ||||||
|  |  | ||||||
| from rest_framework.exceptions import ValidationError | from rest_framework.exceptions import ValidationError | ||||||
| from rest_framework.fields import BooleanField | from rest_framework.fields import BooleanField, JSONField | ||||||
| from structlog.stdlib import get_logger | from structlog.stdlib import get_logger | ||||||
|  |  | ||||||
| from authentik.blueprints.v1.meta.registry import BaseMetaModel, MetaResult, registry | from authentik.blueprints.v1.meta.registry import BaseMetaModel, MetaResult, registry | ||||||
| from authentik.core.api.utils import JSONDictField, PassiveSerializer | from authentik.core.api.utils import PassiveSerializer, is_dict | ||||||
|  |  | ||||||
| if TYPE_CHECKING: | if TYPE_CHECKING: | ||||||
|     from authentik.blueprints.models import BlueprintInstance |     from authentik.blueprints.models import BlueprintInstance | ||||||
| @ -18,7 +17,7 @@ LOGGER = get_logger() | |||||||
| class ApplyBlueprintMetaSerializer(PassiveSerializer): | class ApplyBlueprintMetaSerializer(PassiveSerializer): | ||||||
|     """Serializer for meta apply blueprint model""" |     """Serializer for meta apply blueprint model""" | ||||||
|  |  | ||||||
|     identifiers = JSONDictField() |     identifiers = JSONField(validators=[is_dict]) | ||||||
|     required = BooleanField(default=True) |     required = BooleanField(default=True) | ||||||
|  |  | ||||||
|     # We cannot override `instance` as that will confuse rest_framework |     # We cannot override `instance` as that will confuse rest_framework | ||||||
| @ -43,7 +42,7 @@ class ApplyBlueprintMetaSerializer(PassiveSerializer): | |||||||
|             LOGGER.info("Blueprint does not exist, but not required") |             LOGGER.info("Blueprint does not exist, but not required") | ||||||
|             return MetaResult() |             return MetaResult() | ||||||
|         LOGGER.debug("Applying blueprint from meta model", blueprint=self.blueprint_instance) |         LOGGER.debug("Applying blueprint from meta model", blueprint=self.blueprint_instance) | ||||||
|  |         # pylint: disable=no-value-for-parameter | ||||||
|         apply_blueprint(str(self.blueprint_instance.pk)) |         apply_blueprint(str(self.blueprint_instance.pk)) | ||||||
|         return MetaResult() |         return MetaResult() | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """Base models""" | """Base models""" | ||||||
|  |  | ||||||
| from django.apps import apps | from django.apps import apps | ||||||
| from django.db.models import Model | from django.db.models import Model | ||||||
| from rest_framework.serializers import Serializer | from rest_framework.serializers import Serializer | ||||||
| @ -8,15 +7,15 @@ from rest_framework.serializers import Serializer | |||||||
| class BaseMetaModel(Model): | class BaseMetaModel(Model): | ||||||
|     """Base models""" |     """Base models""" | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         abstract = True |  | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def serializer() -> Serializer: |     def serializer() -> Serializer: | ||||||
|         """Serializer similar to SerializerModel, but as a static method since |         """Serializer similar to SerializerModel, but as a static method since | ||||||
|         this is an abstract model""" |         this is an abstract model""" | ||||||
|         raise NotImplementedError |         raise NotImplementedError | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         abstract = True | ||||||
|  |  | ||||||
|  |  | ||||||
| class MetaResult: | class MetaResult: | ||||||
|     """Result returned by Meta Models' serializers. Empty class but we can't return none as |     """Result returned by Meta Models' serializers. Empty class but we can't return none as | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| """OCI Client""" | """OCI Client""" | ||||||
|  |  | ||||||
| from typing import Any | from typing import Any | ||||||
| from urllib.parse import ParseResult, urlparse | from urllib.parse import ParseResult, urlparse | ||||||
|  |  | ||||||
|  | |||||||
| @ -1,9 +1,8 @@ | |||||||
| """v1 blueprints tasks""" | """v1 blueprints tasks""" | ||||||
|  |  | ||||||
| from dataclasses import asdict, dataclass, field | from dataclasses import asdict, dataclass, field | ||||||
| from hashlib import sha512 | from hashlib import sha512 | ||||||
| from pathlib import Path | from pathlib import Path | ||||||
| from sys import platform | from typing import Optional | ||||||
|  |  | ||||||
| from dacite.core import from_dict | from dacite.core import from_dict | ||||||
| from django.db import DatabaseError, InternalError, ProgrammingError | from django.db import DatabaseError, InternalError, ProgrammingError | ||||||
| @ -30,12 +29,15 @@ from authentik.blueprints.v1.common import BlueprintLoader, BlueprintMetadata, E | |||||||
| from authentik.blueprints.v1.importer import Importer | from authentik.blueprints.v1.importer import Importer | ||||||
| from authentik.blueprints.v1.labels import LABEL_AUTHENTIK_INSTANTIATE | from authentik.blueprints.v1.labels import LABEL_AUTHENTIK_INSTANTIATE | ||||||
| from authentik.blueprints.v1.oci import OCI_PREFIX | from authentik.blueprints.v1.oci import OCI_PREFIX | ||||||
| from authentik.events.models import TaskStatus | from authentik.events.monitored_tasks import ( | ||||||
| from authentik.events.system_tasks import SystemTask, prefill_task |     MonitoredTask, | ||||||
|  |     TaskResult, | ||||||
|  |     TaskResultStatus, | ||||||
|  |     prefill_task, | ||||||
|  | ) | ||||||
| from authentik.events.utils import sanitize_dict | from authentik.events.utils import sanitize_dict | ||||||
| from authentik.lib.config import CONFIG | from authentik.lib.config import CONFIG | ||||||
| from authentik.root.celery import CELERY_APP | from authentik.root.celery import CELERY_APP | ||||||
| from authentik.tenants.models import Tenant |  | ||||||
|  |  | ||||||
| LOGGER = get_logger() | LOGGER = get_logger() | ||||||
| _file_watcher_started = False | _file_watcher_started = False | ||||||
| @ -49,23 +51,18 @@ class BlueprintFile: | |||||||
|     version: int |     version: int | ||||||
|     hash: str |     hash: str | ||||||
|     last_m: int |     last_m: int | ||||||
|     meta: BlueprintMetadata | None = field(default=None) |     meta: Optional[BlueprintMetadata] = field(default=None) | ||||||
|  |  | ||||||
|  |  | ||||||
| def start_blueprint_watcher(): | def start_blueprint_watcher(): | ||||||
|     """Start blueprint watcher, if it's not running already.""" |     """Start blueprint watcher, if it's not running already.""" | ||||||
|     # This function might be called twice since it's called on celery startup |     # This function might be called twice since it's called on celery startup | ||||||
|  |     # pylint: disable=global-statement | ||||||
|     global _file_watcher_started  # noqa: PLW0603 |     global _file_watcher_started | ||||||
|     if _file_watcher_started: |     if _file_watcher_started: | ||||||
|         return |         return | ||||||
|     observer = Observer() |     observer = Observer() | ||||||
|     kwargs = {} |     observer.schedule(BlueprintEventHandler(), CONFIG.get("blueprints_dir"), recursive=True) | ||||||
|     if platform.startswith("linux"): |  | ||||||
|         kwargs["event_filter"] = (FileCreatedEvent, FileModifiedEvent) |  | ||||||
|     observer.schedule( |  | ||||||
|         BlueprintEventHandler(), CONFIG.get("blueprints_dir"), recursive=True, **kwargs |  | ||||||
|     ) |  | ||||||
|     observer.start() |     observer.start() | ||||||
|     _file_watcher_started = True |     _file_watcher_started = True | ||||||
|  |  | ||||||
| @ -73,34 +70,19 @@ def start_blueprint_watcher(): | |||||||
| class BlueprintEventHandler(FileSystemEventHandler): | class BlueprintEventHandler(FileSystemEventHandler): | ||||||
|     """Event handler for blueprint events""" |     """Event handler for blueprint events""" | ||||||
|  |  | ||||||
|     # We only ever get creation and modification events. |     def on_any_event(self, event: FileSystemEvent): | ||||||
|     # See the creation of the Observer instance above for the event filtering. |         if not isinstance(event, (FileCreatedEvent, FileModifiedEvent)): | ||||||
|  |             return | ||||||
|     # Even though we filter to only get file events, we might still get |  | ||||||
|     # directory events as some implementations such as inotify do not support |  | ||||||
|     # filtering on file/directory. |  | ||||||
|  |  | ||||||
|     def dispatch(self, event: FileSystemEvent) -> None: |  | ||||||
|         """Call specific event handler method. Ignores directory changes.""" |  | ||||||
|         if event.is_directory: |         if event.is_directory: | ||||||
|             return None |             return | ||||||
|         return super().dispatch(event) |         if isinstance(event, FileCreatedEvent): | ||||||
|  |  | ||||||
|     def on_created(self, event: FileSystemEvent): |  | ||||||
|         """Process file creation""" |  | ||||||
|             LOGGER.debug("new blueprint file created, starting discovery") |             LOGGER.debug("new blueprint file created, starting discovery") | ||||||
|         for tenant in Tenant.objects.filter(ready=True): |  | ||||||
|             with tenant: |  | ||||||
|             blueprints_discovery.delay() |             blueprints_discovery.delay() | ||||||
|  |         if isinstance(event, FileModifiedEvent): | ||||||
|     def on_modified(self, event: FileSystemEvent): |  | ||||||
|         """Process file modification""" |  | ||||||
|             path = Path(event.src_path) |             path = Path(event.src_path) | ||||||
|             root = Path(CONFIG.get("blueprints_dir")).absolute() |             root = Path(CONFIG.get("blueprints_dir")).absolute() | ||||||
|             rel_path = str(path.relative_to(root)) |             rel_path = str(path.relative_to(root)) | ||||||
|         for tenant in Tenant.objects.filter(ready=True): |             for instance in BlueprintInstance.objects.filter(path=rel_path): | ||||||
|             with tenant: |  | ||||||
|                 for instance in BlueprintInstance.objects.filter(path=rel_path, enabled=True): |  | ||||||
|                 LOGGER.debug("modified blueprint file, starting apply", instance=instance) |                 LOGGER.debug("modified blueprint file, starting apply", instance=instance) | ||||||
|                 apply_blueprint.delay(instance.pk.hex) |                 apply_blueprint.delay(instance.pk.hex) | ||||||
|  |  | ||||||
| @ -116,49 +98,57 @@ def blueprints_find_dict(): | |||||||
|     return blueprints |     return blueprints | ||||||
|  |  | ||||||
|  |  | ||||||
| def blueprints_find() -> list[BlueprintFile]: | def blueprints_find(): | ||||||
|     """Find blueprints and return valid ones""" |     """Find blueprints and return valid ones""" | ||||||
|     blueprints = [] |     blueprints = [] | ||||||
|     root = Path(CONFIG.get("blueprints_dir")) |     root = Path(CONFIG.get("blueprints_dir")) | ||||||
|     for path in root.rglob("**/*.yaml"): |     for path in root.rglob("**/*.yaml"): | ||||||
|         rel_path = path.relative_to(root) |  | ||||||
|         # Check if any part in the path starts with a dot and assume a hidden file |         # Check if any part in the path starts with a dot and assume a hidden file | ||||||
|         if any(part for part in path.parts if part.startswith(".")): |         if any(part for part in path.parts if part.startswith(".")): | ||||||
|             continue |             continue | ||||||
|         with open(path, encoding="utf-8") as blueprint_file: |         LOGGER.debug("found blueprint", path=str(path)) | ||||||
|  |         with open(path, "r", encoding="utf-8") as blueprint_file: | ||||||
|             try: |             try: | ||||||
|                 raw_blueprint = load(blueprint_file.read(), BlueprintLoader) |                 raw_blueprint = load(blueprint_file.read(), BlueprintLoader) | ||||||
|             except YAMLError as exc: |             except YAMLError as exc: | ||||||
|                 raw_blueprint = None |                 raw_blueprint = None | ||||||
|                 LOGGER.warning("failed to parse blueprint", exc=exc, path=str(rel_path)) |                 LOGGER.warning("failed to parse blueprint", exc=exc, path=str(path)) | ||||||
|             if not raw_blueprint: |             if not raw_blueprint: | ||||||
|                 continue |                 continue | ||||||
|             metadata = raw_blueprint.get("metadata", None) |             metadata = raw_blueprint.get("metadata", None) | ||||||
|             version = raw_blueprint.get("version", 1) |             version = raw_blueprint.get("version", 1) | ||||||
|             if version != 1: |             if version != 1: | ||||||
|                 LOGGER.warning("invalid blueprint version", version=version, path=str(rel_path)) |                 LOGGER.warning("invalid blueprint version", version=version, path=str(path)) | ||||||
|                 continue |                 continue | ||||||
|         file_hash = sha512(path.read_bytes()).hexdigest() |         file_hash = sha512(path.read_bytes()).hexdigest() | ||||||
|         blueprint = BlueprintFile(str(rel_path), version, file_hash, int(path.stat().st_mtime)) |         blueprint = BlueprintFile( | ||||||
|  |             str(path.relative_to(root)), version, file_hash, int(path.stat().st_mtime) | ||||||
|  |         ) | ||||||
|         blueprint.meta = from_dict(BlueprintMetadata, metadata) if metadata else None |         blueprint.meta = from_dict(BlueprintMetadata, metadata) if metadata else None | ||||||
|         blueprints.append(blueprint) |         blueprints.append(blueprint) | ||||||
|  |         LOGGER.debug( | ||||||
|  |             "parsed & loaded blueprint", | ||||||
|  |             hash=file_hash, | ||||||
|  |             path=str(path), | ||||||
|  |         ) | ||||||
|     return blueprints |     return blueprints | ||||||
|  |  | ||||||
|  |  | ||||||
| @CELERY_APP.task( | @CELERY_APP.task( | ||||||
|     throws=(DatabaseError, ProgrammingError, InternalError), base=SystemTask, bind=True |     throws=(DatabaseError, ProgrammingError, InternalError), base=MonitoredTask, bind=True | ||||||
| ) | ) | ||||||
| @prefill_task | @prefill_task | ||||||
| def blueprints_discovery(self: SystemTask, path: str | None = None): | def blueprints_discovery(self: MonitoredTask): | ||||||
|     """Find blueprints and check if they need to be created in the database""" |     """Find blueprints and check if they need to be created in the database""" | ||||||
|     count = 0 |     count = 0 | ||||||
|     for blueprint in blueprints_find(): |     for blueprint in blueprints_find(): | ||||||
|         if path and blueprint.path != path: |  | ||||||
|             continue |  | ||||||
|         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)d files." % {"count": count}) |         TaskResult( | ||||||
|  |             TaskResultStatus.SUCCESSFUL, | ||||||
|  |             messages=[_("Successfully imported %(count)d files." % {"count": count})], | ||||||
|  |         ) | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| @ -181,22 +171,18 @@ def check_blueprint_v1_file(blueprint: BlueprintFile): | |||||||
|             metadata={}, |             metadata={}, | ||||||
|         ) |         ) | ||||||
|         instance.save() |         instance.save() | ||||||
|         LOGGER.info( |  | ||||||
|             "Creating new blueprint instance from file", instance=instance, path=instance.path |  | ||||||
|         ) |  | ||||||
|     if instance.last_applied_hash != blueprint.hash: |     if instance.last_applied_hash != blueprint.hash: | ||||||
|         LOGGER.info("Applying blueprint due to changed file", instance=instance, path=instance.path) |  | ||||||
|         apply_blueprint.delay(str(instance.pk)) |         apply_blueprint.delay(str(instance.pk)) | ||||||
|  |  | ||||||
|  |  | ||||||
| @CELERY_APP.task( | @CELERY_APP.task( | ||||||
|     bind=True, |     bind=True, | ||||||
|     base=SystemTask, |     base=MonitoredTask, | ||||||
| ) | ) | ||||||
| def apply_blueprint(self: SystemTask, instance_pk: str): | def apply_blueprint(self: MonitoredTask, instance_pk: str): | ||||||
|     """Apply single blueprint""" |     """Apply single blueprint""" | ||||||
|     self.save_on_success = False |     self.save_on_success = False | ||||||
|     instance: BlueprintInstance | None = None |     instance: Optional[BlueprintInstance] = None | ||||||
|     try: |     try: | ||||||
|         instance: BlueprintInstance = BlueprintInstance.objects.filter(pk=instance_pk).first() |         instance: BlueprintInstance = BlueprintInstance.objects.filter(pk=instance_pk).first() | ||||||
|         if not instance or not instance.enabled: |         if not instance or not instance.enabled: | ||||||
| @ -204,36 +190,36 @@ def apply_blueprint(self: SystemTask, instance_pk: str): | |||||||
|         self.set_uid(slugify(instance.name)) |         self.set_uid(slugify(instance.name)) | ||||||
|         blueprint_content = instance.retrieve() |         blueprint_content = instance.retrieve() | ||||||
|         file_hash = sha512(blueprint_content.encode()).hexdigest() |         file_hash = sha512(blueprint_content.encode()).hexdigest() | ||||||
|         importer = Importer.from_string(blueprint_content, instance.context) |         importer = Importer(blueprint_content, instance.context) | ||||||
|         if importer.blueprint.metadata: |         if importer.blueprint.metadata: | ||||||
|             instance.metadata = asdict(importer.blueprint.metadata) |             instance.metadata = asdict(importer.blueprint.metadata) | ||||||
|         valid, logs = importer.validate() |         valid, logs = importer.validate() | ||||||
|         if not valid: |         if not valid: | ||||||
|             instance.status = BlueprintInstanceStatus.ERROR |             instance.status = BlueprintInstanceStatus.ERROR | ||||||
|             instance.save() |             instance.save() | ||||||
|             self.set_status(TaskStatus.ERROR, *[x["event"] for x in logs]) |             self.set_status(TaskResult(TaskResultStatus.ERROR, [x["event"] for x in logs])) | ||||||
|             return |             return | ||||||
|         applied = importer.apply() |         applied = importer.apply() | ||||||
|         if not applied: |         if not applied: | ||||||
|             instance.status = BlueprintInstanceStatus.ERROR |             instance.status = BlueprintInstanceStatus.ERROR | ||||||
|             instance.save() |             instance.save() | ||||||
|             self.set_status(TaskStatus.ERROR, "Failed to apply") |             self.set_status(TaskResult(TaskResultStatus.ERROR, "Failed to apply")) | ||||||
|             return |             return | ||||||
|         instance.status = BlueprintInstanceStatus.SUCCESSFUL |         instance.status = BlueprintInstanceStatus.SUCCESSFUL | ||||||
|         instance.last_applied_hash = file_hash |         instance.last_applied_hash = file_hash | ||||||
|         instance.last_applied = now() |         instance.last_applied = now() | ||||||
|         self.set_status(TaskStatus.SUCCESSFUL) |         self.set_status(TaskResult(TaskResultStatus.SUCCESSFUL)) | ||||||
|     except ( |     except ( | ||||||
|         OSError, |  | ||||||
|         DatabaseError, |         DatabaseError, | ||||||
|         ProgrammingError, |         ProgrammingError, | ||||||
|         InternalError, |         InternalError, | ||||||
|  |         IOError, | ||||||
|         BlueprintRetrievalFailed, |         BlueprintRetrievalFailed, | ||||||
|         EntryInvalidError, |         EntryInvalidError, | ||||||
|     ) as exc: |     ) as exc: | ||||||
|         if instance: |         if instance: | ||||||
|             instance.status = BlueprintInstanceStatus.ERROR |             instance.status = BlueprintInstanceStatus.ERROR | ||||||
|         self.set_error(exc) |         self.set_status(TaskResult(TaskResultStatus.ERROR).with_error(exc)) | ||||||
|     finally: |     finally: | ||||||
|         if instance: |         if instance: | ||||||
|             instance.save() |             instance.save() | ||||||
|  | |||||||
| @ -1,11 +0,0 @@ | |||||||
| """authentik brands app""" |  | ||||||
|  |  | ||||||
| from django.apps import AppConfig |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthentikBrandsConfig(AppConfig): |  | ||||||
|     """authentik Brand app""" |  | ||||||
|  |  | ||||||
|     name = "authentik.brands" |  | ||||||
|     label = "authentik_brands" |  | ||||||
|     verbose_name = "authentik Brands" |  | ||||||
| @ -1,27 +0,0 @@ | |||||||
| """Inject brand into current request""" |  | ||||||
|  |  | ||||||
| from collections.abc import Callable |  | ||||||
|  |  | ||||||
| from django.http.request import HttpRequest |  | ||||||
| from django.http.response import HttpResponse |  | ||||||
| from django.utils.translation import activate |  | ||||||
|  |  | ||||||
| from authentik.brands.utils import get_brand_for_request |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BrandMiddleware: |  | ||||||
|     """Add current brand to http request""" |  | ||||||
|  |  | ||||||
|     get_response: Callable[[HttpRequest], HttpResponse] |  | ||||||
|  |  | ||||||
|     def __init__(self, get_response: Callable[[HttpRequest], HttpResponse]): |  | ||||||
|         self.get_response = get_response |  | ||||||
|  |  | ||||||
|     def __call__(self, request: HttpRequest) -> HttpResponse: |  | ||||||
|         if not hasattr(request, "brand"): |  | ||||||
|             brand = get_brand_for_request(request) |  | ||||||
|             request.brand = brand |  | ||||||
|             locale = brand.default_locale |  | ||||||
|             if locale != "": |  | ||||||
|                 activate(locale) |  | ||||||
|         return self.get_response(request) |  | ||||||
| @ -1,21 +0,0 @@ | |||||||
| # Generated by Django 4.2.7 on 2023-12-12 06:41 |  | ||||||
|  |  | ||||||
| from django.db import migrations |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Migration(migrations.Migration): |  | ||||||
|     dependencies = [ |  | ||||||
|         ("authentik_brands", "0004_tenant_flow_device_code"), |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     operations = [ |  | ||||||
|         migrations.RenameField( |  | ||||||
|             model_name="brand", |  | ||||||
|             old_name="tenant_uuid", |  | ||||||
|             new_name="brand_uuid", |  | ||||||
|         ), |  | ||||||
|         migrations.RemoveField( |  | ||||||
|             model_name="brand", |  | ||||||
|             name="event_retention", |  | ||||||
|         ), |  | ||||||
|     ] |  | ||||||
| @ -1,86 +0,0 @@ | |||||||
| """brand models""" |  | ||||||
|  |  | ||||||
| from uuid import uuid4 |  | ||||||
|  |  | ||||||
| from django.db import models |  | ||||||
| from django.utils.translation import gettext_lazy as _ |  | ||||||
| from rest_framework.serializers import Serializer |  | ||||||
| from structlog.stdlib import get_logger |  | ||||||
|  |  | ||||||
| from authentik.crypto.models import CertificateKeyPair |  | ||||||
| from authentik.flows.models import Flow |  | ||||||
| from authentik.lib.models import SerializerModel |  | ||||||
|  |  | ||||||
| LOGGER = get_logger() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class Brand(SerializerModel): |  | ||||||
|     """Single brand""" |  | ||||||
|  |  | ||||||
|     brand_uuid = models.UUIDField(primary_key=True, editable=False, default=uuid4) |  | ||||||
|     domain = models.TextField( |  | ||||||
|         help_text=_( |  | ||||||
|             "Domain that activates this brand. Can be a superset, i.e. `a.b` for `aa.b` and `ba.b`" |  | ||||||
|         ) |  | ||||||
|     ) |  | ||||||
|     default = models.BooleanField( |  | ||||||
|         default=False, |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     branding_title = models.TextField(default="authentik") |  | ||||||
|  |  | ||||||
|     branding_logo = models.TextField(default="/static/dist/assets/icons/icon_left_brand.svg") |  | ||||||
|     branding_favicon = models.TextField(default="/static/dist/assets/icons/icon.png") |  | ||||||
|  |  | ||||||
|     flow_authentication = models.ForeignKey( |  | ||||||
|         Flow, null=True, on_delete=models.SET_NULL, related_name="brand_authentication" |  | ||||||
|     ) |  | ||||||
|     flow_invalidation = models.ForeignKey( |  | ||||||
|         Flow, null=True, on_delete=models.SET_NULL, related_name="brand_invalidation" |  | ||||||
|     ) |  | ||||||
|     flow_recovery = models.ForeignKey( |  | ||||||
|         Flow, null=True, on_delete=models.SET_NULL, related_name="brand_recovery" |  | ||||||
|     ) |  | ||||||
|     flow_unenrollment = models.ForeignKey( |  | ||||||
|         Flow, null=True, on_delete=models.SET_NULL, related_name="brand_unenrollment" |  | ||||||
|     ) |  | ||||||
|     flow_user_settings = models.ForeignKey( |  | ||||||
|         Flow, null=True, on_delete=models.SET_NULL, related_name="brand_user_settings" |  | ||||||
|     ) |  | ||||||
|     flow_device_code = models.ForeignKey( |  | ||||||
|         Flow, null=True, on_delete=models.SET_NULL, related_name="brand_device_code" |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     web_certificate = models.ForeignKey( |  | ||||||
|         CertificateKeyPair, |  | ||||||
|         null=True, |  | ||||||
|         default=None, |  | ||||||
|         on_delete=models.SET_DEFAULT, |  | ||||||
|         help_text=_("Web Certificate used by the authentik Core webserver."), |  | ||||||
|     ) |  | ||||||
|     attributes = models.JSONField(default=dict, blank=True) |  | ||||||
|  |  | ||||||
|     @property |  | ||||||
|     def serializer(self) -> Serializer: |  | ||||||
|         from authentik.brands.api import BrandSerializer |  | ||||||
|  |  | ||||||
|         return BrandSerializer |  | ||||||
|  |  | ||||||
|     @property |  | ||||||
|     def default_locale(self) -> str: |  | ||||||
|         """Get default locale""" |  | ||||||
|         try: |  | ||||||
|             return self.attributes.get("settings", {}).get("locale", "") |  | ||||||
|  |  | ||||||
|         except Exception as exc: |  | ||||||
|             LOGGER.warning("Failed to get default locale", exc=exc) |  | ||||||
|             return "" |  | ||||||
|  |  | ||||||
|     def __str__(self) -> str: |  | ||||||
|         if self.default: |  | ||||||
|             return "Default brand" |  | ||||||
|         return f"Brand {self.domain}" |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         verbose_name = _("Brand") |  | ||||||
|         verbose_name_plural = _("Brands") |  | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	