mirror of
				https://github.com/steveiliop56/tinyauth.git
				synced 2025-11-03 23:55:44 +00:00 
			
		
		
		
	Compare commits
	
		
			57 Commits
		
	
	
		
			b5eaf05629
			...
			refactor/d
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					725150f197 | ||
| 
						 | 
					ed3b1c49cb | ||
| 
						 | 
					c5bb389258 | ||
| 
						 | 
					6647c6cd78 | ||
| 
						 | 
					7231efcbc3 | ||
| 
						 | 
					5482430907 | ||
| 
						 | 
					97639ae903 | ||
| 
						 | 
					82350594c1 | ||
| 
						 | 
					57b7b66813 | ||
| 
						 | 
					2ea921f3ca | ||
| 
						 | 
					473109b36a | ||
| 
						 | 
					f628d1f0b3 | ||
| 
						 | 
					a9c1bf8865 | ||
| 
						 | 
					81136eeb42 | ||
| 
						 | 
					8ee331a564 | ||
| 
						 | 
					0996711f08 | ||
| 
						 | 
					64222b6d15 | ||
| 
						 | 
					1b87ed9b99 | ||
| 
						 | 
					dc67be2ba0 | ||
| 
						 | 
					9b76a84ee2 | ||
| 
						 | 
					ed20d2cf51 | ||
| 
						 | 
					fc7e395e66 | ||
| 
						 | 
					b940d681c3 | ||
| 
						 | 
					a1ec4a69cf | ||
| 
						 | 
					4047cea451 | ||
| 
						 | 
					5a4855c12c | ||
| 
						 | 
					05d4dbd68e | ||
| 
						 | 
					ae8347fd28 | ||
| 
						 | 
					76f2014444 | ||
| 
						 | 
					5b7bda3378 | ||
| 
						 | 
					e878516130 | ||
| 
						 | 
					e5f1df03c4 | ||
| 
						 | 
					c77da30d87 | ||
| 
						 | 
					287c6f975f | ||
| 
						 | 
					0255e954f7 | ||
| 
						 | 
					c5d70d7c93 | ||
| 
						 | 
					adffb4ac0a | ||
| 
						 | 
					cbe31d442d | ||
| 
						 | 
					4a530eebc9 | ||
| 
						 | 
					9ba1695274 | ||
| 
						 | 
					c337ba5b31 | ||
| 
						 | 
					bbf8112995 | ||
| 
						 | 
					103285855e | ||
| 
						 | 
					2cc6b6bdbb | ||
| 
						 | 
					adb1a9bee5 | ||
| 
						 | 
					1ee0cee171 | ||
| 
						 | 
					720f387908 | ||
| 
						 | 
					a629430a88 | ||
| 
						 | 
					f0a48cc91c | ||
| 
						 | 
					2f8fa39a9b | ||
| 
						 | 
					30fe695371 | ||
| 
						 | 
					121c629d51 | ||
| 
						 | 
					3ed180cb71 | ||
| 
						 | 
					2f1cb8dfe3 | ||
| 
						 | 
					dad0718091 | ||
| 
						 | 
					d4069900bc | ||
| 
						 | 
					a54996d72d | 
							
								
								
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							@@ -23,7 +23,7 @@ jobs:
 | 
			
		||||
      - name: Install frontend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd frontend
 | 
			
		||||
          bun install
 | 
			
		||||
          bun install --frozen-lockfile
 | 
			
		||||
 | 
			
		||||
      - name: Set version
 | 
			
		||||
        run: |
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										173
									
								
								.github/workflows/nightly.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										173
									
								
								.github/workflows/nightly.yml
									
									
									
									
										vendored
									
									
								
							@@ -66,7 +66,7 @@ jobs:
 | 
			
		||||
      - name: Install frontend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd frontend
 | 
			
		||||
          bun install
 | 
			
		||||
          bun install --frozen-lockfile
 | 
			
		||||
 | 
			
		||||
      - name: Install backend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -112,7 +112,7 @@ jobs:
 | 
			
		||||
      - name: Install frontend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd frontend
 | 
			
		||||
          bun install
 | 
			
		||||
          bun install --frozen-lockfile
 | 
			
		||||
 | 
			
		||||
      - name: Install backend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -171,6 +171,9 @@ jobs:
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
@@ -190,6 +193,65 @@ jobs:
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-build-distroless:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
      - create-release
 | 
			
		||||
      - generate-metadata
 | 
			
		||||
      - image-build
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
        with:
 | 
			
		||||
          ref: nightly
 | 
			
		||||
 | 
			
		||||
      - name: Docker meta
 | 
			
		||||
        id: meta
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
 | 
			
		||||
      - name: Login to GitHub Container Registry
 | 
			
		||||
        uses: docker/login-action@v3
 | 
			
		||||
        with:
 | 
			
		||||
          registry: ghcr.io
 | 
			
		||||
          username: ${{ github.repository_owner }}
 | 
			
		||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
      - name: Build and push
 | 
			
		||||
        uses: docker/build-push-action@v6
 | 
			
		||||
        id: build
 | 
			
		||||
        with:
 | 
			
		||||
          platforms: linux/amd64
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          file: Dockerfile.distroless
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
            BUILD_TIMESTAMP=${{ needs.generate-metadata.outputs.BUILD_TIMESTAMP }}
 | 
			
		||||
 | 
			
		||||
      - name: Export digest
 | 
			
		||||
        run: |
 | 
			
		||||
          mkdir -p ${{ runner.temp }}/digests
 | 
			
		||||
          digest="${{ steps.build.outputs.digest }}"
 | 
			
		||||
          touch "${{ runner.temp }}/digests/${digest#sha256:}"
 | 
			
		||||
 | 
			
		||||
      - name: Upload digest
 | 
			
		||||
        uses: actions/upload-artifact@v4
 | 
			
		||||
        with:
 | 
			
		||||
          name: digests-distroless-linux-amd64
 | 
			
		||||
          path: ${{ runner.temp }}/digests/*
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-build-arm:
 | 
			
		||||
    runs-on: ubuntu-24.04-arm
 | 
			
		||||
    needs:
 | 
			
		||||
@@ -217,10 +279,6 @@ jobs:
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
      - name: Set version
 | 
			
		||||
        run: |
 | 
			
		||||
          echo nightly > internal/assets/version
 | 
			
		||||
 | 
			
		||||
      - name: Build and push
 | 
			
		||||
        uses: docker/build-push-action@v6
 | 
			
		||||
        id: build
 | 
			
		||||
@@ -229,6 +287,9 @@ jobs:
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
@@ -248,6 +309,65 @@ jobs:
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-build-arm-distroless:
 | 
			
		||||
    runs-on: ubuntu-24.04-arm
 | 
			
		||||
    needs:
 | 
			
		||||
      - create-release
 | 
			
		||||
      - generate-metadata
 | 
			
		||||
      - image-build-arm
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
        with:
 | 
			
		||||
          ref: nightly
 | 
			
		||||
 | 
			
		||||
      - name: Docker meta
 | 
			
		||||
        id: meta
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
 | 
			
		||||
      - name: Login to GitHub Container Registry
 | 
			
		||||
        uses: docker/login-action@v3
 | 
			
		||||
        with:
 | 
			
		||||
          registry: ghcr.io
 | 
			
		||||
          username: ${{ github.repository_owner }}
 | 
			
		||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
      - name: Build and push
 | 
			
		||||
        uses: docker/build-push-action@v6
 | 
			
		||||
        id: build
 | 
			
		||||
        with:
 | 
			
		||||
          platforms: linux/arm64
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          file: Dockerfile.distroless
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
            BUILD_TIMESTAMP=${{ needs.generate-metadata.outputs.BUILD_TIMESTAMP }}
 | 
			
		||||
 | 
			
		||||
      - name: Export digest
 | 
			
		||||
        run: |
 | 
			
		||||
          mkdir -p ${{ runner.temp }}/digests
 | 
			
		||||
          digest="${{ steps.build.outputs.digest }}"
 | 
			
		||||
          touch "${{ runner.temp }}/digests/${digest#sha256:}"
 | 
			
		||||
 | 
			
		||||
      - name: Upload digest
 | 
			
		||||
        uses: actions/upload-artifact@v4
 | 
			
		||||
        with:
 | 
			
		||||
          name: digests-distroless-linux-arm64
 | 
			
		||||
          path: ${{ runner.temp }}/digests/*
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-merge:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
@@ -276,6 +396,8 @@ jobs:
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          flavor: |
 | 
			
		||||
            latest=false
 | 
			
		||||
          tags: |
 | 
			
		||||
            type=raw,nightly
 | 
			
		||||
 | 
			
		||||
@@ -285,6 +407,45 @@ jobs:
 | 
			
		||||
          docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \
 | 
			
		||||
            $(printf 'ghcr.io/${{ github.repository_owner }}/tinyauth@sha256:%s ' *)
 | 
			
		||||
 | 
			
		||||
  image-merge-distroless:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
      - image-build-distroless
 | 
			
		||||
      - image-build-arm-distroless
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Download digests
 | 
			
		||||
        uses: actions/download-artifact@v4
 | 
			
		||||
        with:
 | 
			
		||||
          path: ${{ runner.temp }}/digests
 | 
			
		||||
          pattern: digests-distroless-*
 | 
			
		||||
          merge-multiple: true
 | 
			
		||||
 | 
			
		||||
      - name: Login to GitHub Container Registry
 | 
			
		||||
        uses: docker/login-action@v3
 | 
			
		||||
        with:
 | 
			
		||||
          registry: ghcr.io
 | 
			
		||||
          username: ${{ github.repository_owner }}
 | 
			
		||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
      - name: Docker meta
 | 
			
		||||
        id: meta
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          flavor: |
 | 
			
		||||
            latest=false
 | 
			
		||||
          tags: |
 | 
			
		||||
            type=raw,nightly-distroless
 | 
			
		||||
 | 
			
		||||
      - name: Create manifest list and push
 | 
			
		||||
        working-directory: ${{ runner.temp }}/digests
 | 
			
		||||
        run: |
 | 
			
		||||
          docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \
 | 
			
		||||
            $(printf 'ghcr.io/${{ github.repository_owner }}/tinyauth@sha256:%s ' *)
 | 
			
		||||
 | 
			
		||||
  update-release:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										173
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										173
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							@@ -44,7 +44,7 @@ jobs:
 | 
			
		||||
      - name: Install frontend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd frontend
 | 
			
		||||
          bun install
 | 
			
		||||
          bun install --frozen-lockfile
 | 
			
		||||
 | 
			
		||||
      - name: Install backend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -87,7 +87,7 @@ jobs:
 | 
			
		||||
      - name: Install frontend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
          cd frontend
 | 
			
		||||
          bun install
 | 
			
		||||
          bun install --frozen-lockfile
 | 
			
		||||
 | 
			
		||||
      - name: Install backend dependencies
 | 
			
		||||
        run: |
 | 
			
		||||
@@ -143,6 +143,9 @@ jobs:
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
@@ -162,6 +165,62 @@ jobs:
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-build-distroless:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
      - generate-metadata
 | 
			
		||||
      - image-build
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
 | 
			
		||||
      - name: Docker meta
 | 
			
		||||
        id: meta
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
 | 
			
		||||
      - name: Login to GitHub Container Registry
 | 
			
		||||
        uses: docker/login-action@v3
 | 
			
		||||
        with:
 | 
			
		||||
          registry: ghcr.io
 | 
			
		||||
          username: ${{ github.repository_owner }}
 | 
			
		||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
      - name: Build and push
 | 
			
		||||
        uses: docker/build-push-action@v6
 | 
			
		||||
        id: build
 | 
			
		||||
        with:
 | 
			
		||||
          platforms: linux/amd64
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          file: Dockerfile.distroless
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
            BUILD_TIMESTAMP=${{ needs.generate-metadata.outputs.BUILD_TIMESTAMP }}
 | 
			
		||||
 | 
			
		||||
      - name: Export digest
 | 
			
		||||
        run: |
 | 
			
		||||
          mkdir -p ${{ runner.temp }}/digests
 | 
			
		||||
          digest="${{ steps.build.outputs.digest }}"
 | 
			
		||||
          touch "${{ runner.temp }}/digests/${digest#sha256:}"
 | 
			
		||||
 | 
			
		||||
      - name: Upload digest
 | 
			
		||||
        uses: actions/upload-artifact@v4
 | 
			
		||||
        with:
 | 
			
		||||
          name: digests-distroless-linux-amd64
 | 
			
		||||
          path: ${{ runner.temp }}/digests/*
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-build-arm:
 | 
			
		||||
    runs-on: ubuntu-24.04-arm
 | 
			
		||||
    needs:
 | 
			
		||||
@@ -194,6 +253,9 @@ jobs:
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
@@ -213,6 +275,62 @@ jobs:
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-build-arm-distroless:
 | 
			
		||||
    runs-on: ubuntu-24.04-arm
 | 
			
		||||
    needs:
 | 
			
		||||
      - generate-metadata
 | 
			
		||||
      - image-build-arm
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Checkout
 | 
			
		||||
        uses: actions/checkout@v4
 | 
			
		||||
 | 
			
		||||
      - name: Docker meta
 | 
			
		||||
        id: meta
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
 | 
			
		||||
      - name: Login to GitHub Container Registry
 | 
			
		||||
        uses: docker/login-action@v3
 | 
			
		||||
        with:
 | 
			
		||||
          registry: ghcr.io
 | 
			
		||||
          username: ${{ github.repository_owner }}
 | 
			
		||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
      - name: Build and push
 | 
			
		||||
        uses: docker/build-push-action@v6
 | 
			
		||||
        id: build
 | 
			
		||||
        with:
 | 
			
		||||
          platforms: linux/arm64
 | 
			
		||||
          labels: ${{ steps.meta.outputs.labels }}
 | 
			
		||||
          tags: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          outputs: type=image,push-by-digest=true,name-canonical=true,push=true
 | 
			
		||||
          file: Dockerfile.distroless
 | 
			
		||||
          cache-from: type=gha
 | 
			
		||||
          cache-to: type=gha,mode=max
 | 
			
		||||
          github-token: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
          build-args: |
 | 
			
		||||
            VERSION=${{ needs.generate-metadata.outputs.VERSION }}
 | 
			
		||||
            COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }}
 | 
			
		||||
            BUILD_TIMESTAMP=${{ needs.generate-metadata.outputs.BUILD_TIMESTAMP }}
 | 
			
		||||
 | 
			
		||||
      - name: Export digest
 | 
			
		||||
        run: |
 | 
			
		||||
          mkdir -p ${{ runner.temp }}/digests
 | 
			
		||||
          digest="${{ steps.build.outputs.digest }}"
 | 
			
		||||
          touch "${{ runner.temp }}/digests/${digest#sha256:}"
 | 
			
		||||
 | 
			
		||||
      - name: Upload digest
 | 
			
		||||
        uses: actions/upload-artifact@v4
 | 
			
		||||
        with:
 | 
			
		||||
          name: digests-distroless-linux-arm64
 | 
			
		||||
          path: ${{ runner.temp }}/digests/*
 | 
			
		||||
          if-no-files-found: error
 | 
			
		||||
          retention-days: 1
 | 
			
		||||
 | 
			
		||||
  image-merge:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
@@ -241,10 +359,55 @@ jobs:
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          flavor: |
 | 
			
		||||
            prefix=v,onlatest=false
 | 
			
		||||
          tags: |
 | 
			
		||||
            type=semver,pattern={{version}},prefix=v
 | 
			
		||||
            type=semver,pattern={{major}},prefix=v
 | 
			
		||||
            type=semver,pattern={{major}}.{{minor}},prefix=v
 | 
			
		||||
            type=semver,pattern={{version}}
 | 
			
		||||
            type=semver,pattern={{major}}
 | 
			
		||||
            type=semver,pattern={{major}}.{{minor}}
 | 
			
		||||
 | 
			
		||||
      - name: Create manifest list and push
 | 
			
		||||
        working-directory: ${{ runner.temp }}/digests
 | 
			
		||||
        run: |
 | 
			
		||||
          docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \
 | 
			
		||||
            $(printf 'ghcr.io/${{ github.repository_owner }}/tinyauth@sha256:%s ' *)
 | 
			
		||||
 | 
			
		||||
  image-merge-distroless:
 | 
			
		||||
    runs-on: ubuntu-latest
 | 
			
		||||
    needs:
 | 
			
		||||
      - image-build-distroless
 | 
			
		||||
      - image-build-arm-distroless
 | 
			
		||||
    steps:
 | 
			
		||||
      - name: Download digests
 | 
			
		||||
        uses: actions/download-artifact@v4
 | 
			
		||||
        with:
 | 
			
		||||
          path: ${{ runner.temp }}/digests
 | 
			
		||||
          pattern: digests-distroless-*
 | 
			
		||||
          merge-multiple: true
 | 
			
		||||
 | 
			
		||||
      - name: Login to GitHub Container Registry
 | 
			
		||||
        uses: docker/login-action@v3
 | 
			
		||||
        with:
 | 
			
		||||
          registry: ghcr.io
 | 
			
		||||
          username: ${{ github.repository_owner }}
 | 
			
		||||
          password: ${{ secrets.GITHUB_TOKEN }}
 | 
			
		||||
 | 
			
		||||
      - name: Set up Docker Buildx
 | 
			
		||||
        uses: docker/setup-buildx-action@v3
 | 
			
		||||
 | 
			
		||||
      - name: Docker meta
 | 
			
		||||
        id: meta
 | 
			
		||||
        uses: docker/metadata-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          images: ghcr.io/${{ github.repository_owner }}/tinyauth
 | 
			
		||||
          flavor: |
 | 
			
		||||
            latest=false
 | 
			
		||||
            prefix=v
 | 
			
		||||
            suffix=-distroless
 | 
			
		||||
          tags: |
 | 
			
		||||
            type=semver,pattern={{version}}
 | 
			
		||||
            type=semver,pattern={{major}}
 | 
			
		||||
            type=semver,pattern={{major}}.{{minor}}
 | 
			
		||||
 | 
			
		||||
      - name: Create manifest list and push
 | 
			
		||||
        working-directory: ${{ runner.temp }}/digests
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										16
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								Dockerfile
									
									
									
									
									
								
							@@ -1,12 +1,12 @@
 | 
			
		||||
# Site builder
 | 
			
		||||
FROM oven/bun:1.2.22-alpine AS frontend-builder
 | 
			
		||||
FROM oven/bun:1.3.0-alpine AS frontend-builder
 | 
			
		||||
 | 
			
		||||
WORKDIR /frontend
 | 
			
		||||
 | 
			
		||||
COPY ./frontend/package.json ./
 | 
			
		||||
COPY ./frontend/bun.lock ./
 | 
			
		||||
 | 
			
		||||
RUN bun install
 | 
			
		||||
RUN bun install --frozen-lockfile
 | 
			
		||||
 | 
			
		||||
COPY ./frontend/public ./public
 | 
			
		||||
COPY ./frontend/src ./src
 | 
			
		||||
@@ -45,12 +45,18 @@ FROM alpine:3.22 AS runner
 | 
			
		||||
 | 
			
		||||
WORKDIR /tinyauth
 | 
			
		||||
 | 
			
		||||
RUN apk add --no-cache curl
 | 
			
		||||
 | 
			
		||||
COPY --from=builder /tinyauth/tinyauth ./
 | 
			
		||||
 | 
			
		||||
RUN mkdir -p /data
 | 
			
		||||
 | 
			
		||||
EXPOSE 3000
 | 
			
		||||
 | 
			
		||||
VOLUME ["/data"]
 | 
			
		||||
 | 
			
		||||
ENTRYPOINT ["./tinyauth"]
 | 
			
		||||
ENV GIN_MODE=release
 | 
			
		||||
 | 
			
		||||
ENV PATH=$PATH:/tinyauth
 | 
			
		||||
 | 
			
		||||
HEALTHCHECK --interval=30s --timeout=5s --start-period=5s --retries=3 CMD ["tinyauth", "healthcheck"]
 | 
			
		||||
 | 
			
		||||
ENTRYPOINT ["tinyauth"]
 | 
			
		||||
							
								
								
									
										65
									
								
								Dockerfile.distroless
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								Dockerfile.distroless
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,65 @@
 | 
			
		||||
# Site builder
 | 
			
		||||
FROM oven/bun:1.3.0-alpine AS frontend-builder
 | 
			
		||||
 | 
			
		||||
WORKDIR /frontend
 | 
			
		||||
 | 
			
		||||
COPY ./frontend/package.json ./
 | 
			
		||||
COPY ./frontend/bun.lock ./
 | 
			
		||||
 | 
			
		||||
RUN bun install --frozen-lockfile
 | 
			
		||||
 | 
			
		||||
COPY ./frontend/public ./public
 | 
			
		||||
COPY ./frontend/src ./src
 | 
			
		||||
COPY ./frontend/eslint.config.js ./
 | 
			
		||||
COPY ./frontend/index.html ./
 | 
			
		||||
COPY ./frontend/tsconfig.json ./
 | 
			
		||||
COPY ./frontend/tsconfig.app.json ./
 | 
			
		||||
COPY ./frontend/tsconfig.node.json ./
 | 
			
		||||
COPY ./frontend/vite.config.ts ./
 | 
			
		||||
 | 
			
		||||
RUN bun run build
 | 
			
		||||
 | 
			
		||||
# Builder
 | 
			
		||||
FROM golang:1.25-alpine3.21 AS builder
 | 
			
		||||
 | 
			
		||||
ARG VERSION
 | 
			
		||||
ARG COMMIT_HASH
 | 
			
		||||
ARG BUILD_TIMESTAMP
 | 
			
		||||
 | 
			
		||||
WORKDIR /tinyauth
 | 
			
		||||
 | 
			
		||||
COPY go.mod ./
 | 
			
		||||
COPY go.sum ./
 | 
			
		||||
 | 
			
		||||
RUN go mod download
 | 
			
		||||
 | 
			
		||||
COPY ./main.go ./
 | 
			
		||||
COPY ./cmd ./cmd
 | 
			
		||||
COPY ./internal ./internal
 | 
			
		||||
COPY --from=frontend-builder /frontend/dist ./internal/assets/dist
 | 
			
		||||
 | 
			
		||||
RUN mkdir -p data
 | 
			
		||||
 | 
			
		||||
RUN CGO_ENABLED=0 go build -ldflags "-s -w -X tinyauth/internal/config.Version=${VERSION} -X tinyauth/internal/config.CommitHash=${COMMIT_HASH} -X tinyauth/internal/config.BuildTimestamp=${BUILD_TIMESTAMP}" 
 | 
			
		||||
 
 | 
			
		||||
# Runner
 | 
			
		||||
FROM gcr.io/distroless/static-debian12:latest AS runner
 | 
			
		||||
 | 
			
		||||
WORKDIR /tinyauth
 | 
			
		||||
 | 
			
		||||
COPY --from=builder /tinyauth/tinyauth ./
 | 
			
		||||
 | 
			
		||||
# Since it's distroless, we need to copy the data directory from the builder stage
 | 
			
		||||
COPY --from=builder /tinyauth/data /data
 | 
			
		||||
 | 
			
		||||
EXPOSE 3000
 | 
			
		||||
 | 
			
		||||
VOLUME ["/data"]
 | 
			
		||||
 | 
			
		||||
ENV GIN_MODE=release
 | 
			
		||||
 | 
			
		||||
ENV PATH=$PATH:/tinyauth
 | 
			
		||||
 | 
			
		||||
HEALTHCHECK --interval=30s --timeout=5s --start-period=5s --retries=3 CMD ["tinyauth", "healthcheck"]
 | 
			
		||||
 | 
			
		||||
ENTRYPOINT ["tinyauth"]
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
<div align="center">
 | 
			
		||||
    <img alt="Tinyauth" title="Tinyauth" width="96" src="assets/logo-rounded.png">
 | 
			
		||||
    <h1>Tinyauth</h1>
 | 
			
		||||
    <p>The easiest way to secure your apps with a login screen.</p>
 | 
			
		||||
    <p>The simplest way to protect your apps with a login screen.</p>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
<div align="center">
 | 
			
		||||
@@ -14,7 +14,7 @@
 | 
			
		||||
 | 
			
		||||
<br />
 | 
			
		||||
 | 
			
		||||
Tinyauth is a simple authentication middleware that adds a simple login screen or OAuth with Google, Github and any provider to all of your docker apps. It supports all the popular proxies like Traefik, Nginx and Caddy.
 | 
			
		||||
Tinyauth is a simple authentication middleware that adds a simple login screen or OAuth with Google, Github or any other provider to all of your apps. It supports all the popular proxies like Traefik, Nginx and Caddy.
 | 
			
		||||
 | 
			
		||||

 | 
			
		||||
 | 
			
		||||
@@ -23,7 +23,7 @@ Tinyauth is a simple authentication middleware that adds a simple login screen o
 | 
			
		||||
 | 
			
		||||
## Getting Started
 | 
			
		||||
 | 
			
		||||
You can easily get started with Tinyauth by following the guide in the [documentation](https://tinyauth.app/docs/getting-started.html). There is also an available [docker compose](./docker-compose.example.yml) file that has Traefik, Whoami and Tinyauth to demonstrate its capabilities.
 | 
			
		||||
You can easily get started with Tinyauth by following the guide in the [documentation](https://tinyauth.app/docs/getting-started). There is also an available [docker compose](./docker-compose.example.yml) file that has Traefik, Whoami and Tinyauth to demonstrate its capabilities.
 | 
			
		||||
 | 
			
		||||
## Demo
 | 
			
		||||
 | 
			
		||||
@@ -53,7 +53,7 @@ Tinyauth is licensed under the GNU General Public License v3.0. TL;DR — You ma
 | 
			
		||||
 | 
			
		||||
A big thank you to the following people for providing me with more coffee:
 | 
			
		||||
 | 
			
		||||
<!-- sponsors --><a href="https://github.com/erwinkramer"><img src="https://github.com/erwinkramer.png" width="64px" alt="User avatar: erwinkramer" /></a>  <a href="https://github.com/nicotsx"><img src="https://github.com/nicotsx.png" width="64px" alt="User avatar: nicotsx" /></a>  <a href="https://github.com/SimpleHomelab"><img src="https://github.com/SimpleHomelab.png" width="64px" alt="User avatar: SimpleHomelab" /></a>  <a href="https://github.com/jmadden91"><img src="https://github.com/jmadden91.png" width="64px" alt="User avatar: jmadden91" /></a>  <a href="https://github.com/tribor"><img src="https://github.com/tribor.png" width="64px" alt="User avatar: tribor" /></a>  <a href="https://github.com/eliasbenb"><img src="https://github.com/eliasbenb.png" width="64px" alt="User avatar: eliasbenb" /></a>  <a href="https://github.com/afunworm"><img src="https://github.com/afunworm.png" width="64px" alt="User avatar: afunworm" /></a>  <!-- sponsors -->
 | 
			
		||||
<!-- sponsors --><a href="https://github.com/erwinkramer"><img src="https://github.com/erwinkramer.png" width="64px" alt="User avatar: erwinkramer" /></a>  <a href="https://github.com/nicotsx"><img src="https://github.com/nicotsx.png" width="64px" alt="User avatar: nicotsx" /></a>  <a href="https://github.com/SimpleHomelab"><img src="https://github.com/SimpleHomelab.png" width="64px" alt="User avatar: SimpleHomelab" /></a>  <a href="https://github.com/jmadden91"><img src="https://github.com/jmadden91.png" width="64px" alt="User avatar: jmadden91" /></a>  <a href="https://github.com/tribor"><img src="https://github.com/tribor.png" width="64px" alt="User avatar: tribor" /></a>  <a href="https://github.com/eliasbenb"><img src="https://github.com/eliasbenb.png" width="64px" alt="User avatar: eliasbenb" /></a>  <a href="https://github.com/afunworm"><img src="https://github.com/afunworm.png" width="64px" alt="User avatar: afunworm" /></a>  <a href="https://github.com/chip-well"><img src="https://github.com/chip-well.png" width="64px" alt="User avatar: chip-well" /></a>  <a href="https://github.com/Lancelot-Enguerrand"><img src="https://github.com/Lancelot-Enguerrand.png" width="64px" alt="User avatar: Lancelot-Enguerrand" /></a>  <!-- sponsors -->
 | 
			
		||||
 | 
			
		||||
## Acknowledgements
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										2
									
								
								air.toml
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								air.toml
									
									
									
									
									
								
							@@ -2,7 +2,7 @@ root = "/tinyauth"
 | 
			
		||||
tmp_dir = "tmp"
 | 
			
		||||
 | 
			
		||||
[build]
 | 
			
		||||
pre_cmd = ["mkdir -p internal/assets/dist", "echo 'backend running' > internal/assets/dist/index.html", "go install github.com/go-delve/delve/cmd/dlv@v1.25.0"]
 | 
			
		||||
pre_cmd = ["mkdir -p internal/assets/dist", "mkdir -p /data", "echo 'backend running' > internal/assets/dist/index.html", "go install github.com/go-delve/delve/cmd/dlv@v1.25.0"]
 | 
			
		||||
cmd = "CGO_ENABLED=0 go build -gcflags=\"all=-N -l\" -o tmp/tinyauth ."
 | 
			
		||||
bin = "/go/bin/dlv --listen :4000 --headless=true --api-version=2 --accept-multiclient --log=true exec tmp/tinyauth --continue --check-go-version=false"
 | 
			
		||||
include_ext = ["go"]
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										99
									
								
								cmd/create.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								cmd/create.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,99 @@
 | 
			
		||||
package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type createUserCmd struct {
 | 
			
		||||
	root *cobra.Command
 | 
			
		||||
	cmd  *cobra.Command
 | 
			
		||||
 | 
			
		||||
	interactive bool
 | 
			
		||||
	docker      bool
 | 
			
		||||
	username    string
 | 
			
		||||
	password    string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newCreateUserCmd(root *cobra.Command) *createUserCmd {
 | 
			
		||||
	return &createUserCmd{
 | 
			
		||||
		root: root,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *createUserCmd) Register() {
 | 
			
		||||
	c.cmd = &cobra.Command{
 | 
			
		||||
		Use:   "create",
 | 
			
		||||
		Short: "Create a user",
 | 
			
		||||
		Long:  `Create a user either interactively or by passing flags.`,
 | 
			
		||||
		Run:   c.run,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	c.cmd.Flags().BoolVarP(&c.interactive, "interactive", "i", false, "Create a user interactively")
 | 
			
		||||
	c.cmd.Flags().BoolVar(&c.docker, "docker", false, "Format output for docker")
 | 
			
		||||
	c.cmd.Flags().StringVar(&c.username, "username", "", "Username")
 | 
			
		||||
	c.cmd.Flags().StringVar(&c.password, "password", "", "Password")
 | 
			
		||||
 | 
			
		||||
	if c.root != nil {
 | 
			
		||||
		c.root.AddCommand(c.cmd)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *createUserCmd) GetCmd() *cobra.Command {
 | 
			
		||||
	return c.cmd
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *createUserCmd) run(cmd *cobra.Command, args []string) {
 | 
			
		||||
	log.Logger = log.Level(zerolog.InfoLevel)
 | 
			
		||||
 | 
			
		||||
	if c.interactive {
 | 
			
		||||
		form := huh.NewForm(
 | 
			
		||||
			huh.NewGroup(
 | 
			
		||||
				huh.NewInput().Title("Username").Value(&c.username).Validate((func(s string) error {
 | 
			
		||||
					if s == "" {
 | 
			
		||||
						return errors.New("username cannot be empty")
 | 
			
		||||
					}
 | 
			
		||||
					return nil
 | 
			
		||||
				})),
 | 
			
		||||
				huh.NewInput().Title("Password").Value(&c.password).Validate((func(s string) error {
 | 
			
		||||
					if s == "" {
 | 
			
		||||
						return errors.New("password cannot be empty")
 | 
			
		||||
					}
 | 
			
		||||
					return nil
 | 
			
		||||
				})),
 | 
			
		||||
				huh.NewSelect[bool]().Title("Format the output for Docker?").Options(huh.NewOption("Yes", true), huh.NewOption("No", false)).Value(&c.docker),
 | 
			
		||||
			),
 | 
			
		||||
		)
 | 
			
		||||
		var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
		err := form.WithTheme(baseTheme).Run()
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Form failed")
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if c.username == "" || c.password == "" {
 | 
			
		||||
		log.Fatal().Err(errors.New("error invalid input")).Msg("Username and password cannot be empty")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Info().Str("username", c.username).Msg("Creating user")
 | 
			
		||||
 | 
			
		||||
	passwd, err := bcrypt.GenerateFromPassword([]byte(c.password), bcrypt.DefaultCost)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to hash password")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If docker format is enabled, escape the dollar sign
 | 
			
		||||
	passwdStr := string(passwd)
 | 
			
		||||
	if c.docker {
 | 
			
		||||
		passwdStr = strings.ReplaceAll(passwdStr, "$", "$$")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Info().Str("user", fmt.Sprintf("%s:%s", c.username, passwdStr)).Msg("User created")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										120
									
								
								cmd/generate.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										120
									
								
								cmd/generate.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,120 @@
 | 
			
		||||
package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/mdp/qrterminal/v3"
 | 
			
		||||
	"github.com/pquerna/otp/totp"
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type generateTotpCmd struct {
 | 
			
		||||
	root *cobra.Command
 | 
			
		||||
	cmd  *cobra.Command
 | 
			
		||||
 | 
			
		||||
	interactive bool
 | 
			
		||||
	user        string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newGenerateTotpCmd(root *cobra.Command) *generateTotpCmd {
 | 
			
		||||
	return &generateTotpCmd{
 | 
			
		||||
		root: root,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *generateTotpCmd) Register() {
 | 
			
		||||
	c.cmd = &cobra.Command{
 | 
			
		||||
		Use:   "generate",
 | 
			
		||||
		Short: "Generate a totp secret",
 | 
			
		||||
		Long:  `Generate a totp secret for a user either interactively or by passing flags.`,
 | 
			
		||||
		Run:   c.run,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	c.cmd.Flags().BoolVarP(&c.interactive, "interactive", "i", false, "Run in interactive mode")
 | 
			
		||||
	c.cmd.Flags().StringVar(&c.user, "user", "", "Your current user (username:hash)")
 | 
			
		||||
 | 
			
		||||
	if c.root != nil {
 | 
			
		||||
		c.root.AddCommand(c.cmd)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *generateTotpCmd) GetCmd() *cobra.Command {
 | 
			
		||||
	return c.cmd
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *generateTotpCmd) run(cmd *cobra.Command, args []string) {
 | 
			
		||||
	log.Logger = log.Level(zerolog.InfoLevel)
 | 
			
		||||
 | 
			
		||||
	if c.interactive {
 | 
			
		||||
		form := huh.NewForm(
 | 
			
		||||
			huh.NewGroup(
 | 
			
		||||
				huh.NewInput().Title("Current user (username:hash)").Value(&c.user).Validate((func(s string) error {
 | 
			
		||||
					if s == "" {
 | 
			
		||||
						return errors.New("user cannot be empty")
 | 
			
		||||
					}
 | 
			
		||||
					return nil
 | 
			
		||||
				})),
 | 
			
		||||
			),
 | 
			
		||||
		)
 | 
			
		||||
		var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
		err := form.WithTheme(baseTheme).Run()
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Form failed")
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	user, err := utils.ParseUser(c.user)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to parse user")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	docker := false
 | 
			
		||||
	if strings.Contains(c.user, "$$") {
 | 
			
		||||
		docker = true
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if user.TotpSecret != "" {
 | 
			
		||||
		log.Fatal().Msg("User already has a TOTP secret")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	key, err := totp.Generate(totp.GenerateOpts{
 | 
			
		||||
		Issuer:      "Tinyauth",
 | 
			
		||||
		AccountName: user.Username,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to generate TOTP secret")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	secret := key.Secret()
 | 
			
		||||
 | 
			
		||||
	log.Info().Str("secret", secret).Msg("Generated TOTP secret")
 | 
			
		||||
 | 
			
		||||
	log.Info().Msg("Generated QR code")
 | 
			
		||||
 | 
			
		||||
	config := qrterminal.Config{
 | 
			
		||||
		Level:     qrterminal.L,
 | 
			
		||||
		Writer:    os.Stdout,
 | 
			
		||||
		BlackChar: qrterminal.BLACK,
 | 
			
		||||
		WhiteChar: qrterminal.WHITE,
 | 
			
		||||
		QuietZone: 2,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	qrterminal.GenerateWithConfig(key.URL(), config)
 | 
			
		||||
 | 
			
		||||
	user.TotpSecret = secret
 | 
			
		||||
 | 
			
		||||
	// If using docker escape re-escape it
 | 
			
		||||
	if docker {
 | 
			
		||||
		user.Password = strings.ReplaceAll(user.Password, "$", "$$")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Info().Str("user", fmt.Sprintf("%s:%s:%s", user.Username, user.Password, user.TotpSecret)).Msg("Add the totp secret to your authenticator app then use the verify command to ensure everything is working correctly.")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										112
									
								
								cmd/healthcheck.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								cmd/healthcheck.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,112 @@
 | 
			
		||||
package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"encoding/json"
 | 
			
		||||
	"errors"
 | 
			
		||||
	"io"
 | 
			
		||||
	"net/http"
 | 
			
		||||
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"github.com/spf13/viper"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type healthzResponse struct {
 | 
			
		||||
	Status  string `json:"status"`
 | 
			
		||||
	Message string `json:"message"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type healthcheckCmd struct {
 | 
			
		||||
	root *cobra.Command
 | 
			
		||||
	cmd  *cobra.Command
 | 
			
		||||
 | 
			
		||||
	viper *viper.Viper
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newHealthcheckCmd(root *cobra.Command) *healthcheckCmd {
 | 
			
		||||
	return &healthcheckCmd{
 | 
			
		||||
		root:  root,
 | 
			
		||||
		viper: viper.New(),
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *healthcheckCmd) Register() {
 | 
			
		||||
	c.cmd = &cobra.Command{
 | 
			
		||||
		Use:   "healthcheck [app-url]",
 | 
			
		||||
		Short: "Perform a health check",
 | 
			
		||||
		Long:  `Use the health check endpoint to verify that Tinyauth is running and it's healthy.`,
 | 
			
		||||
		Run:   c.run,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	c.viper.AutomaticEnv()
 | 
			
		||||
 | 
			
		||||
	if c.root != nil {
 | 
			
		||||
		c.root.AddCommand(c.cmd)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *healthcheckCmd) GetCmd() *cobra.Command {
 | 
			
		||||
	return c.cmd
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *healthcheckCmd) run(cmd *cobra.Command, args []string) {
 | 
			
		||||
	log.Logger = log.Level(zerolog.InfoLevel)
 | 
			
		||||
 | 
			
		||||
	var appUrl string
 | 
			
		||||
 | 
			
		||||
	port := c.viper.GetString("PORT")
 | 
			
		||||
	address := c.viper.GetString("ADDRESS")
 | 
			
		||||
 | 
			
		||||
	if port == "" {
 | 
			
		||||
		port = "3000"
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if address == "" {
 | 
			
		||||
		address = "127.0.0.1"
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	appUrl = "http://" + address + ":" + port
 | 
			
		||||
 | 
			
		||||
	if len(args) > 0 {
 | 
			
		||||
		appUrl = args[0]
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Info().Str("app_url", appUrl).Msg("Performing health check")
 | 
			
		||||
 | 
			
		||||
	client := http.Client{}
 | 
			
		||||
 | 
			
		||||
	req, err := http.NewRequest("GET", appUrl+"/api/healthz", nil)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to create request")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	resp, err := client.Do(req)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to perform request")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if resp.StatusCode != http.StatusOK {
 | 
			
		||||
		log.Fatal().Err(errors.New("service is not healthy")).Msgf("Service is not healthy. Status code: %d", resp.StatusCode)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	defer resp.Body.Close()
 | 
			
		||||
 | 
			
		||||
	var healthResp healthzResponse
 | 
			
		||||
 | 
			
		||||
	body, err := io.ReadAll(resp.Body)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to read response")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = json.Unmarshal(body, &healthResp)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to decode response")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Info().Interface("response", healthResp).Msg("Tinyauth is healthy")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										146
									
								
								cmd/root.go
									
									
									
									
									
								
							
							
						
						
									
										146
									
								
								cmd/root.go
									
									
									
									
									
								
							@@ -2,8 +2,6 @@ package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"strings"
 | 
			
		||||
	totpCmd "tinyauth/cmd/totp"
 | 
			
		||||
	userCmd "tinyauth/cmd/user"
 | 
			
		||||
	"tinyauth/internal/bootstrap"
 | 
			
		||||
	"tinyauth/internal/config"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
@@ -15,55 +13,28 @@ import (
 | 
			
		||||
	"github.com/spf13/viper"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var rootCmd = &cobra.Command{
 | 
			
		||||
type rootCmd struct {
 | 
			
		||||
	root *cobra.Command
 | 
			
		||||
	cmd  *cobra.Command
 | 
			
		||||
 | 
			
		||||
	viper *viper.Viper
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newRootCmd() *rootCmd {
 | 
			
		||||
	return &rootCmd{
 | 
			
		||||
		viper: viper.New(),
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *rootCmd) Register() {
 | 
			
		||||
	c.cmd = &cobra.Command{
 | 
			
		||||
		Use:   "tinyauth",
 | 
			
		||||
	Short: "The simplest way to protect your apps with a login screen.",
 | 
			
		||||
	Long:  `Tinyauth is a simple authentication middleware that adds simple username/password login or OAuth with Google, Github and any generic OAuth provider to all of your docker apps.`,
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		var conf config.Config
 | 
			
		||||
 | 
			
		||||
		err := viper.Unmarshal(&conf)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Failed to parse config")
 | 
			
		||||
		Short: "The simplest way to protect your apps with a login screen",
 | 
			
		||||
		Long:  `Tinyauth is a simple authentication middleware that adds a simple login screen or OAuth with Google, Github or any other provider to all of your docker apps.`,
 | 
			
		||||
		Run:   c.run,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
		// Validate config
 | 
			
		||||
		v := validator.New()
 | 
			
		||||
 | 
			
		||||
		err = v.Struct(conf)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Invalid config")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Logger = log.Level(zerolog.Level(utils.GetLogLevel(conf.LogLevel)))
 | 
			
		||||
		log.Info().Str("version", strings.TrimSpace(config.Version)).Msg("Starting tinyauth")
 | 
			
		||||
 | 
			
		||||
		// Create bootstrap app
 | 
			
		||||
		app := bootstrap.NewBootstrapApp(conf)
 | 
			
		||||
 | 
			
		||||
		// Run
 | 
			
		||||
		err = app.Setup()
 | 
			
		||||
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Failed to setup app")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func Execute() {
 | 
			
		||||
	rootCmd.FParseErrWhitelist.UnknownFlags = true
 | 
			
		||||
	err := rootCmd.Execute()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to execute command")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	rootCmd.AddCommand(userCmd.UserCmd())
 | 
			
		||||
	rootCmd.AddCommand(totpCmd.TotpCmd())
 | 
			
		||||
 | 
			
		||||
	viper.AutomaticEnv()
 | 
			
		||||
	c.viper.AutomaticEnv()
 | 
			
		||||
 | 
			
		||||
	configOptions := []struct {
 | 
			
		||||
		name        string
 | 
			
		||||
@@ -101,17 +72,86 @@ func init() {
 | 
			
		||||
	for _, opt := range configOptions {
 | 
			
		||||
		switch v := opt.defaultVal.(type) {
 | 
			
		||||
		case bool:
 | 
			
		||||
			rootCmd.Flags().Bool(opt.name, v, opt.description)
 | 
			
		||||
			c.cmd.Flags().Bool(opt.name, v, opt.description)
 | 
			
		||||
		case int:
 | 
			
		||||
			rootCmd.Flags().Int(opt.name, v, opt.description)
 | 
			
		||||
			c.cmd.Flags().Int(opt.name, v, opt.description)
 | 
			
		||||
		case string:
 | 
			
		||||
			rootCmd.Flags().String(opt.name, v, opt.description)
 | 
			
		||||
			c.cmd.Flags().String(opt.name, v, opt.description)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Create uppercase env var name
 | 
			
		||||
		envVar := strings.ReplaceAll(strings.ToUpper(opt.name), "-", "_")
 | 
			
		||||
		viper.BindEnv(opt.name, envVar)
 | 
			
		||||
		c.viper.BindEnv(opt.name, envVar)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	viper.BindPFlags(rootCmd.Flags())
 | 
			
		||||
	c.viper.BindPFlags(c.cmd.Flags())
 | 
			
		||||
 | 
			
		||||
	if c.root != nil {
 | 
			
		||||
		c.root.AddCommand(c.cmd)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *rootCmd) GetCmd() *cobra.Command {
 | 
			
		||||
	return c.cmd
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *rootCmd) run(cmd *cobra.Command, args []string) {
 | 
			
		||||
	var conf config.Config
 | 
			
		||||
 | 
			
		||||
	err := c.viper.Unmarshal(&conf)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to parse config")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	v := validator.New()
 | 
			
		||||
	err = v.Struct(conf)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Invalid config")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Logger = log.Level(zerolog.Level(utils.GetLogLevel(conf.LogLevel)))
 | 
			
		||||
	log.Info().Str("version", strings.TrimSpace(config.Version)).Msg("Starting Tinyauth")
 | 
			
		||||
 | 
			
		||||
	if log.Logger.GetLevel() == zerolog.TraceLevel {
 | 
			
		||||
		log.Warn().Msg("Log level set to trace, this will log sensitive information!")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	app := bootstrap.NewBootstrapApp(conf)
 | 
			
		||||
 | 
			
		||||
	err = app.Setup()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to setup app")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func Run() {
 | 
			
		||||
	rootCmd := newRootCmd()
 | 
			
		||||
	rootCmd.Register()
 | 
			
		||||
	root := rootCmd.GetCmd()
 | 
			
		||||
 | 
			
		||||
	userCmd := &cobra.Command{
 | 
			
		||||
		Use:   "user",
 | 
			
		||||
		Short: "User utilities",
 | 
			
		||||
		Long:  `Utilities for creating and verifying tinyauth compatible users.`,
 | 
			
		||||
	}
 | 
			
		||||
	totpCmd := &cobra.Command{
 | 
			
		||||
		Use:   "totp",
 | 
			
		||||
		Short: "Totp utilities",
 | 
			
		||||
		Long:  `Utilities for creating and verifying totp codes.`,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	newCreateUserCmd(userCmd).Register()
 | 
			
		||||
	newVerifyUserCmd(userCmd).Register()
 | 
			
		||||
	newGenerateTotpCmd(totpCmd).Register()
 | 
			
		||||
	newVersionCmd(root).Register()
 | 
			
		||||
	newHealthcheckCmd(root).Register()
 | 
			
		||||
 | 
			
		||||
	root.AddCommand(userCmd)
 | 
			
		||||
	root.AddCommand(totpCmd)
 | 
			
		||||
 | 
			
		||||
	err := root.Execute()
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to execute root command")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,99 +0,0 @@
 | 
			
		||||
package generate
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/mdp/qrterminal/v3"
 | 
			
		||||
	"github.com/pquerna/otp/totp"
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var interactive bool
 | 
			
		||||
 | 
			
		||||
// Input user
 | 
			
		||||
var iUser string
 | 
			
		||||
 | 
			
		||||
var GenerateCmd = &cobra.Command{
 | 
			
		||||
	Use:   "generate",
 | 
			
		||||
	Short: "Generate a totp secret",
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		log.Logger = log.Level(zerolog.InfoLevel)
 | 
			
		||||
 | 
			
		||||
		if interactive {
 | 
			
		||||
			form := huh.NewForm(
 | 
			
		||||
				huh.NewGroup(
 | 
			
		||||
					huh.NewInput().Title("Current username:hash").Value(&iUser).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("user cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
				),
 | 
			
		||||
			)
 | 
			
		||||
			var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
			err := form.WithTheme(baseTheme).Run()
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				log.Fatal().Err(err).Msg("Form failed")
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		user, err := utils.ParseUser(iUser)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Failed to parse user")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		dockerEscape := false
 | 
			
		||||
		if strings.Contains(iUser, "$$") {
 | 
			
		||||
			dockerEscape = true
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if user.TotpSecret != "" {
 | 
			
		||||
			log.Fatal().Msg("User already has a totp secret")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		key, err := totp.Generate(totp.GenerateOpts{
 | 
			
		||||
			Issuer:      "Tinyauth",
 | 
			
		||||
			AccountName: user.Username,
 | 
			
		||||
		})
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Failed to generate totp secret")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		secret := key.Secret()
 | 
			
		||||
 | 
			
		||||
		log.Info().Str("secret", secret).Msg("Generated totp secret")
 | 
			
		||||
 | 
			
		||||
		log.Info().Msg("Generated QR code")
 | 
			
		||||
 | 
			
		||||
		config := qrterminal.Config{
 | 
			
		||||
			Level:     qrterminal.L,
 | 
			
		||||
			Writer:    os.Stdout,
 | 
			
		||||
			BlackChar: qrterminal.BLACK,
 | 
			
		||||
			WhiteChar: qrterminal.WHITE,
 | 
			
		||||
			QuietZone: 2,
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		qrterminal.GenerateWithConfig(key.URL(), config)
 | 
			
		||||
 | 
			
		||||
		user.TotpSecret = secret
 | 
			
		||||
 | 
			
		||||
		// If using docker escape re-escape it
 | 
			
		||||
		if dockerEscape {
 | 
			
		||||
			user.Password = strings.ReplaceAll(user.Password, "$", "$$")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Info().Str("user", fmt.Sprintf("%s:%s:%s", user.Username, user.Password, user.TotpSecret)).Msg("Add the totp secret to your authenticator app then use the verify command to ensure everything is working correctly.")
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	GenerateCmd.Flags().BoolVarP(&interactive, "interactive", "i", false, "Run in interactive mode")
 | 
			
		||||
	GenerateCmd.Flags().StringVar(&iUser, "user", "", "Your current username:hash")
 | 
			
		||||
}
 | 
			
		||||
@@ -1,17 +0,0 @@
 | 
			
		||||
package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"tinyauth/cmd/totp/generate"
 | 
			
		||||
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TotpCmd() *cobra.Command {
 | 
			
		||||
	totpCmd := &cobra.Command{
 | 
			
		||||
		Use:   "totp",
 | 
			
		||||
		Short: "Totp utilities",
 | 
			
		||||
		Long:  `Utilities for creating and verifying totp codes.`,
 | 
			
		||||
	}
 | 
			
		||||
	totpCmd.AddCommand(generate.GenerateCmd)
 | 
			
		||||
	return totpCmd
 | 
			
		||||
}
 | 
			
		||||
@@ -1,80 +0,0 @@
 | 
			
		||||
package create
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var interactive bool
 | 
			
		||||
var docker bool
 | 
			
		||||
 | 
			
		||||
// i stands for input
 | 
			
		||||
var iUsername string
 | 
			
		||||
var iPassword string
 | 
			
		||||
 | 
			
		||||
var CreateCmd = &cobra.Command{
 | 
			
		||||
	Use:   "create",
 | 
			
		||||
	Short: "Create a user",
 | 
			
		||||
	Long:  `Create a user either interactively or by passing flags.`,
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		log.Logger = log.Level(zerolog.InfoLevel)
 | 
			
		||||
 | 
			
		||||
		if interactive {
 | 
			
		||||
			form := huh.NewForm(
 | 
			
		||||
				huh.NewGroup(
 | 
			
		||||
					huh.NewInput().Title("Username").Value(&iUsername).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("username cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewInput().Title("Password").Value(&iPassword).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("password cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewSelect[bool]().Title("Format the output for docker?").Options(huh.NewOption("Yes", true), huh.NewOption("No", false)).Value(&docker),
 | 
			
		||||
				),
 | 
			
		||||
			)
 | 
			
		||||
			var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
			err := form.WithTheme(baseTheme).Run()
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				log.Fatal().Err(err).Msg("Form failed")
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if iUsername == "" || iPassword == "" {
 | 
			
		||||
			log.Fatal().Err(errors.New("error invalid input")).Msg("Username and password cannot be empty")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Info().Str("username", iUsername).Str("password", iPassword).Bool("docker", docker).Msg("Creating user")
 | 
			
		||||
 | 
			
		||||
		password, err := bcrypt.GenerateFromPassword([]byte(iPassword), bcrypt.DefaultCost)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Failed to hash password")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// If docker format is enabled, escape the dollar sign
 | 
			
		||||
		passwordString := string(password)
 | 
			
		||||
		if docker {
 | 
			
		||||
			passwordString = strings.ReplaceAll(passwordString, "$", "$$")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Info().Str("user", fmt.Sprintf("%s:%s", iUsername, passwordString)).Msg("User created")
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	CreateCmd.Flags().BoolVarP(&interactive, "interactive", "i", false, "Create a user interactively")
 | 
			
		||||
	CreateCmd.Flags().BoolVar(&docker, "docker", false, "Format output for docker")
 | 
			
		||||
	CreateCmd.Flags().StringVar(&iUsername, "username", "", "Username")
 | 
			
		||||
	CreateCmd.Flags().StringVar(&iPassword, "password", "", "Password")
 | 
			
		||||
}
 | 
			
		||||
@@ -1,19 +0,0 @@
 | 
			
		||||
package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"tinyauth/cmd/user/create"
 | 
			
		||||
	"tinyauth/cmd/user/verify"
 | 
			
		||||
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func UserCmd() *cobra.Command {
 | 
			
		||||
	userCmd := &cobra.Command{
 | 
			
		||||
		Use:   "user",
 | 
			
		||||
		Short: "User utilities",
 | 
			
		||||
		Long:  `Utilities for creating and verifying tinyauth compatible users.`,
 | 
			
		||||
	}
 | 
			
		||||
	userCmd.AddCommand(create.CreateCmd)
 | 
			
		||||
	userCmd.AddCommand(verify.VerifyCmd)
 | 
			
		||||
	return userCmd
 | 
			
		||||
}
 | 
			
		||||
@@ -1,101 +0,0 @@
 | 
			
		||||
package verify
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/pquerna/otp/totp"
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var interactive bool
 | 
			
		||||
var docker bool
 | 
			
		||||
 | 
			
		||||
// i stands for input
 | 
			
		||||
var iUsername string
 | 
			
		||||
var iPassword string
 | 
			
		||||
var iTotp string
 | 
			
		||||
var iUser string
 | 
			
		||||
 | 
			
		||||
var VerifyCmd = &cobra.Command{
 | 
			
		||||
	Use:   "verify",
 | 
			
		||||
	Short: "Verify a user is set up correctly",
 | 
			
		||||
	Long:  `Verify a user is set up correctly meaning that it has a correct username, password and totp code.`,
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		log.Logger = log.Level(zerolog.InfoLevel)
 | 
			
		||||
 | 
			
		||||
		if interactive {
 | 
			
		||||
			form := huh.NewForm(
 | 
			
		||||
				huh.NewGroup(
 | 
			
		||||
					huh.NewInput().Title("User (username:hash:totp)").Value(&iUser).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("user cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewInput().Title("Username").Value(&iUsername).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("username cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewInput().Title("Password").Value(&iPassword).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("password cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewInput().Title("Totp Code (if setup)").Value(&iTotp),
 | 
			
		||||
				),
 | 
			
		||||
			)
 | 
			
		||||
			var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
			err := form.WithTheme(baseTheme).Run()
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				log.Fatal().Err(err).Msg("Form failed")
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		user, err := utils.ParseUser(iUser)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Failed to parse user")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if user.Username != iUsername {
 | 
			
		||||
			log.Fatal().Msg("Username is incorrect")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		err = bcrypt.CompareHashAndPassword([]byte(user.Password), []byte(iPassword))
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Msg("Password is incorrect")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if user.TotpSecret == "" {
 | 
			
		||||
			if iTotp != "" {
 | 
			
		||||
				log.Warn().Msg("User does not have 2fa secret")
 | 
			
		||||
			}
 | 
			
		||||
			log.Info().Msg("User verified")
 | 
			
		||||
			return
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		ok := totp.Validate(iTotp, user.TotpSecret)
 | 
			
		||||
		if !ok {
 | 
			
		||||
			log.Fatal().Msg("Totp code incorrect")
 | 
			
		||||
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Info().Msg("User verified")
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	VerifyCmd.Flags().BoolVarP(&interactive, "interactive", "i", false, "Create a user interactively")
 | 
			
		||||
	VerifyCmd.Flags().BoolVar(&docker, "docker", false, "Is the user formatted for docker?")
 | 
			
		||||
	VerifyCmd.Flags().StringVar(&iUsername, "username", "", "Username")
 | 
			
		||||
	VerifyCmd.Flags().StringVar(&iPassword, "password", "", "Password")
 | 
			
		||||
	VerifyCmd.Flags().StringVar(&iTotp, "totp", "", "Totp code")
 | 
			
		||||
	VerifyCmd.Flags().StringVar(&iUser, "user", "", "Hash (username:hash:totp combination)")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										118
									
								
								cmd/verify.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								cmd/verify.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,118 @@
 | 
			
		||||
package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/pquerna/otp/totp"
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type verifyUserCmd struct {
 | 
			
		||||
	root *cobra.Command
 | 
			
		||||
	cmd  *cobra.Command
 | 
			
		||||
 | 
			
		||||
	interactive bool
 | 
			
		||||
	username    string
 | 
			
		||||
	password    string
 | 
			
		||||
	totp        string
 | 
			
		||||
	user        string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newVerifyUserCmd(root *cobra.Command) *verifyUserCmd {
 | 
			
		||||
	return &verifyUserCmd{
 | 
			
		||||
		root: root,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *verifyUserCmd) Register() {
 | 
			
		||||
	c.cmd = &cobra.Command{
 | 
			
		||||
		Use:   "verify",
 | 
			
		||||
		Short: "Verify a user is set up correctly",
 | 
			
		||||
		Long:  `Verify a user is set up correctly meaning that it has a correct username, password and TOTP code.`,
 | 
			
		||||
		Run:   c.run,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	c.cmd.Flags().BoolVarP(&c.interactive, "interactive", "i", false, "Validate a user interactively")
 | 
			
		||||
	c.cmd.Flags().StringVar(&c.username, "username", "", "Username")
 | 
			
		||||
	c.cmd.Flags().StringVar(&c.password, "password", "", "Password")
 | 
			
		||||
	c.cmd.Flags().StringVar(&c.totp, "totp", "", "TOTP code")
 | 
			
		||||
	c.cmd.Flags().StringVar(&c.user, "user", "", "Hash (username:hash:totp)")
 | 
			
		||||
 | 
			
		||||
	if c.root != nil {
 | 
			
		||||
		c.root.AddCommand(c.cmd)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *verifyUserCmd) GetCmd() *cobra.Command {
 | 
			
		||||
	return c.cmd
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *verifyUserCmd) run(cmd *cobra.Command, args []string) {
 | 
			
		||||
	log.Logger = log.Level(zerolog.InfoLevel)
 | 
			
		||||
 | 
			
		||||
	if c.interactive {
 | 
			
		||||
		form := huh.NewForm(
 | 
			
		||||
			huh.NewGroup(
 | 
			
		||||
				huh.NewInput().Title("User (username:hash:totp)").Value(&c.user).Validate((func(s string) error {
 | 
			
		||||
					if s == "" {
 | 
			
		||||
						return errors.New("user cannot be empty")
 | 
			
		||||
					}
 | 
			
		||||
					return nil
 | 
			
		||||
				})),
 | 
			
		||||
				huh.NewInput().Title("Username").Value(&c.username).Validate((func(s string) error {
 | 
			
		||||
					if s == "" {
 | 
			
		||||
						return errors.New("username cannot be empty")
 | 
			
		||||
					}
 | 
			
		||||
					return nil
 | 
			
		||||
				})),
 | 
			
		||||
				huh.NewInput().Title("Password").Value(&c.password).Validate((func(s string) error {
 | 
			
		||||
					if s == "" {
 | 
			
		||||
						return errors.New("password cannot be empty")
 | 
			
		||||
					}
 | 
			
		||||
					return nil
 | 
			
		||||
				})),
 | 
			
		||||
				huh.NewInput().Title("TOTP Code (optional)").Value(&c.totp),
 | 
			
		||||
			),
 | 
			
		||||
		)
 | 
			
		||||
		var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
		err := form.WithTheme(baseTheme).Run()
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Fatal().Err(err).Msg("Form failed")
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	user, err := utils.ParseUser(c.user)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to parse user")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if user.Username != c.username {
 | 
			
		||||
		log.Fatal().Msg("Username is incorrect")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = bcrypt.CompareHashAndPassword([]byte(user.Password), []byte(c.password))
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Msg("Password is incorrect")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if user.TotpSecret == "" {
 | 
			
		||||
		if c.totp != "" {
 | 
			
		||||
			log.Warn().Msg("User does not have TOTP secret")
 | 
			
		||||
		}
 | 
			
		||||
		log.Info().Msg("User verified")
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	ok := totp.Validate(c.totp, user.TotpSecret)
 | 
			
		||||
	if !ok {
 | 
			
		||||
		log.Fatal().Msg("TOTP code incorrect")
 | 
			
		||||
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Info().Msg("User verified")
 | 
			
		||||
}
 | 
			
		||||
@@ -7,17 +7,36 @@ import (
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var versionCmd = &cobra.Command{
 | 
			
		||||
type versionCmd struct {
 | 
			
		||||
	root *cobra.Command
 | 
			
		||||
	cmd  *cobra.Command
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func newVersionCmd(root *cobra.Command) *versionCmd {
 | 
			
		||||
	return &versionCmd{
 | 
			
		||||
		root: root,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *versionCmd) Register() {
 | 
			
		||||
	c.cmd = &cobra.Command{
 | 
			
		||||
		Use:   "version",
 | 
			
		||||
		Short: "Print the version number of Tinyauth",
 | 
			
		||||
	Long:  `All software has versions. This is Tinyauth's`,
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		Long:  `All software has versions. This is Tinyauth's.`,
 | 
			
		||||
		Run:   c.run,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if c.root != nil {
 | 
			
		||||
		c.root.AddCommand(c.cmd)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *versionCmd) GetCmd() *cobra.Command {
 | 
			
		||||
	return c.cmd
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (c *versionCmd) run(cmd *cobra.Command, args []string) {
 | 
			
		||||
	fmt.Printf("Version: %s\n", config.Version)
 | 
			
		||||
	fmt.Printf("Commit Hash: %s\n", config.CommitHash)
 | 
			
		||||
	fmt.Printf("Build Timestamp: %s\n", config.BuildTimestamp)
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	rootCmd.AddCommand(versionCmd)
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -9,44 +9,44 @@
 | 
			
		||||
        "@radix-ui/react-select": "^2.2.6",
 | 
			
		||||
        "@radix-ui/react-separator": "^1.1.7",
 | 
			
		||||
        "@radix-ui/react-slot": "^1.2.3",
 | 
			
		||||
        "@tailwindcss/vite": "^4.1.13",
 | 
			
		||||
        "@tanstack/react-query": "^5.89.0",
 | 
			
		||||
        "@tailwindcss/vite": "^4.1.14",
 | 
			
		||||
        "@tanstack/react-query": "^5.90.3",
 | 
			
		||||
        "axios": "^1.12.2",
 | 
			
		||||
        "class-variance-authority": "^0.7.1",
 | 
			
		||||
        "clsx": "^2.1.1",
 | 
			
		||||
        "i18next": "^25.5.2",
 | 
			
		||||
        "i18next": "^25.6.0",
 | 
			
		||||
        "i18next-browser-languagedetector": "^8.2.0",
 | 
			
		||||
        "i18next-resources-to-backend": "^1.2.1",
 | 
			
		||||
        "input-otp": "^1.4.2",
 | 
			
		||||
        "lucide-react": "^0.544.0",
 | 
			
		||||
        "lucide-react": "^0.545.0",
 | 
			
		||||
        "next-themes": "^0.4.6",
 | 
			
		||||
        "react": "^19.1.1",
 | 
			
		||||
        "react-dom": "^19.1.1",
 | 
			
		||||
        "react-hook-form": "^7.62.0",
 | 
			
		||||
        "react-i18next": "^15.7.3",
 | 
			
		||||
        "react": "^19.2.0",
 | 
			
		||||
        "react-dom": "^19.2.0",
 | 
			
		||||
        "react-hook-form": "^7.65.0",
 | 
			
		||||
        "react-i18next": "^16.0.1",
 | 
			
		||||
        "react-markdown": "^10.1.0",
 | 
			
		||||
        "react-router": "^7.9.1",
 | 
			
		||||
        "react-router": "^7.9.4",
 | 
			
		||||
        "sonner": "^2.0.7",
 | 
			
		||||
        "tailwind-merge": "^3.3.1",
 | 
			
		||||
        "tailwindcss": "^4.1.13",
 | 
			
		||||
        "zod": "^4.1.9",
 | 
			
		||||
        "tailwindcss": "^4.1.14",
 | 
			
		||||
        "zod": "^4.1.12",
 | 
			
		||||
      },
 | 
			
		||||
      "devDependencies": {
 | 
			
		||||
        "@eslint/js": "^9.35.0",
 | 
			
		||||
        "@tanstack/eslint-plugin-query": "^5.89.0",
 | 
			
		||||
        "@types/node": "^24.5.2",
 | 
			
		||||
        "@types/react": "^19.1.13",
 | 
			
		||||
        "@types/react-dom": "^19.1.9",
 | 
			
		||||
        "@vitejs/plugin-react": "^5.0.3",
 | 
			
		||||
        "eslint": "^9.35.0",
 | 
			
		||||
        "eslint-plugin-react-hooks": "^5.2.0",
 | 
			
		||||
        "eslint-plugin-react-refresh": "^0.4.19",
 | 
			
		||||
        "@eslint/js": "^9.37.0",
 | 
			
		||||
        "@tanstack/eslint-plugin-query": "^5.91.0",
 | 
			
		||||
        "@types/node": "^24.7.2",
 | 
			
		||||
        "@types/react": "^19.2.2",
 | 
			
		||||
        "@types/react-dom": "^19.2.2",
 | 
			
		||||
        "@vitejs/plugin-react": "^5.0.4",
 | 
			
		||||
        "eslint": "^9.37.0",
 | 
			
		||||
        "eslint-plugin-react-hooks": "^7.0.0",
 | 
			
		||||
        "eslint-plugin-react-refresh": "^0.4.23",
 | 
			
		||||
        "globals": "^16.4.0",
 | 
			
		||||
        "prettier": "3.6.2",
 | 
			
		||||
        "tw-animate-css": "^1.3.8",
 | 
			
		||||
        "typescript": "~5.9.2",
 | 
			
		||||
        "typescript-eslint": "^8.44.0",
 | 
			
		||||
        "vite": "^7.1.6",
 | 
			
		||||
        "tw-animate-css": "^1.4.0",
 | 
			
		||||
        "typescript": "~5.9.3",
 | 
			
		||||
        "typescript-eslint": "^8.46.1",
 | 
			
		||||
        "vite": "^7.1.10",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
@@ -147,17 +147,17 @@
 | 
			
		||||
 | 
			
		||||
    "@eslint/config-array": ["@eslint/config-array@0.21.0", "", { "dependencies": { "@eslint/object-schema": "^2.1.6", "debug": "^4.3.1", "minimatch": "^3.1.2" } }, "sha512-ENIdc4iLu0d93HeYirvKmrzshzofPw6VkZRKQGe9Nv46ZnWUzcF1xV01dcvEg/1wXUR61OmmlSfyeyO7EvjLxQ=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/config-helpers": ["@eslint/config-helpers@0.3.1", "", {}, "sha512-xR93k9WhrDYpXHORXpxVL5oHj3Era7wo6k/Wd8/IsQNnZUTzkGS29lyn3nAT05v6ltUuTFVCCYDEGfy2Or/sPA=="],
 | 
			
		||||
    "@eslint/config-helpers": ["@eslint/config-helpers@0.4.0", "", { "dependencies": { "@eslint/core": "^0.16.0" } }, "sha512-WUFvV4WoIwW8Bv0KeKCIIEgdSiFOsulyN0xrMu+7z43q/hkOLXjvb5u7UC9jDxvRzcrbEmuZBX5yJZz1741jog=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/core": ["@eslint/core@0.15.2", "", { "dependencies": { "@types/json-schema": "^7.0.15" } }, "sha512-78Md3/Rrxh83gCxoUc0EiciuOHsIITzLy53m3d9UyiW8y9Dj2D29FeETqyKA+BRK76tnTp6RXWb3pCay8Oyomg=="],
 | 
			
		||||
    "@eslint/core": ["@eslint/core@0.16.0", "", { "dependencies": { "@types/json-schema": "^7.0.15" } }, "sha512-nmC8/totwobIiFcGkDza3GIKfAw1+hLiYVrh3I1nIomQ8PEr5cxg34jnkmGawul/ep52wGRAcyeDCNtWKSOj4Q=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/eslintrc": ["@eslint/eslintrc@3.3.1", "", { "dependencies": { "ajv": "^6.12.4", "debug": "^4.3.2", "espree": "^10.0.1", "globals": "^14.0.0", "ignore": "^5.2.0", "import-fresh": "^3.2.1", "js-yaml": "^4.1.0", "minimatch": "^3.1.2", "strip-json-comments": "^3.1.1" } }, "sha512-gtF186CXhIl1p4pJNGZw8Yc6RlshoePRvE0X91oPGb3vZ8pM3qOS9W9NGPat9LziaBV7XrJWGylNQXkGcnM3IQ=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/js": ["@eslint/js@9.35.0", "", {}, "sha512-30iXE9whjlILfWobBkNerJo+TXYsgVM5ERQwMcMKCHckHflCmf7wXDAHlARoWnh0s1U72WqlbeyE7iAcCzuCPw=="],
 | 
			
		||||
    "@eslint/js": ["@eslint/js@9.37.0", "", {}, "sha512-jaS+NJ+hximswBG6pjNX0uEJZkrT0zwpVi3BA3vX22aFGjJjmgSTSmPpZCRKmoBL5VY/M6p0xsSJx7rk7sy5gg=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/object-schema": ["@eslint/object-schema@2.1.6", "", {}, "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/plugin-kit": ["@eslint/plugin-kit@0.3.5", "", { "dependencies": { "@eslint/core": "^0.15.2", "levn": "^0.4.1" } }, "sha512-Z5kJ+wU3oA7MMIqVR9tyZRtjYPr4OC004Q4Rw7pgOKUOKkJfZ3O24nz3WYfGRpMDNmcOi3TwQOmgm7B7Tpii0w=="],
 | 
			
		||||
    "@eslint/plugin-kit": ["@eslint/plugin-kit@0.4.0", "", { "dependencies": { "@eslint/core": "^0.16.0", "levn": "^0.4.1" } }, "sha512-sB5uyeq+dwCWyPi31B2gQlVlo+j5brPlWx4yZBrEaRo/nhdDE8Xke1gsGgtiBdaBTxuTkceLVuVt/pclrasb0A=="],
 | 
			
		||||
 | 
			
		||||
    "@floating-ui/core": ["@floating-ui/core@1.7.0", "", { "dependencies": { "@floating-ui/utils": "^0.2.9" } }, "sha512-FRdBLykrPPA6P76GGGqlex/e7fbe0F1ykgxHYNXQsH/iTEtjMj/f9bpY5oQqbjt5VgZvgz/uKXbGuROijh3VLA=="],
 | 
			
		||||
 | 
			
		||||
@@ -253,7 +253,7 @@
 | 
			
		||||
 | 
			
		||||
    "@radix-ui/rect": ["@radix-ui/rect@1.1.1", "", {}, "sha512-HPwpGIzkl28mWyZqG52jiqDJ12waP11Pa1lGoiyUkIEuMLBP0oeK/C89esbXrxsky5we7dfd8U58nm0SgAWpVw=="],
 | 
			
		||||
 | 
			
		||||
    "@rolldown/pluginutils": ["@rolldown/pluginutils@1.0.0-beta.35", "", {}, "sha512-slYrCpoxJUqzFDDNlvrOYRazQUNRvWPjXA17dAOISY3rDMxX6k8K4cj2H+hEYMHF81HO3uNd5rHVigAWRM5dSg=="],
 | 
			
		||||
    "@rolldown/pluginutils": ["@rolldown/pluginutils@1.0.0-beta.38", "", {}, "sha512-N/ICGKleNhA5nc9XXQG/kkKHJ7S55u0x0XUJbbkmdCnFuoRkM1Il12q9q0eX19+M7KKUEPw/daUPIRnxhcxAIw=="],
 | 
			
		||||
 | 
			
		||||
    "@rollup/rollup-android-arm-eabi": ["@rollup/rollup-android-arm-eabi@4.46.2", "", { "os": "android", "cpu": "arm" }, "sha512-Zj3Hl6sN34xJtMv7Anwb5Gu01yujyE/cLBDB2gnHTAHaWS1Z38L7kuSG+oAh0giZMqG060f/YBStXtMH6FvPMA=="],
 | 
			
		||||
 | 
			
		||||
@@ -297,41 +297,41 @@
 | 
			
		||||
 | 
			
		||||
    "@standard-schema/utils": ["@standard-schema/utils@0.3.0", "", {}, "sha512-e7Mew686owMaPJVNNLs55PUvgz371nKgwsc4vxE49zsODpJEnxgxRo2y/OKrqueavXgZNMDVj3DdHFlaSAeU8g=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node": ["@tailwindcss/node@4.1.13", "", { "dependencies": { "@jridgewell/remapping": "^2.3.4", "enhanced-resolve": "^5.18.3", "jiti": "^2.5.1", "lightningcss": "1.30.1", "magic-string": "^0.30.18", "source-map-js": "^1.2.1", "tailwindcss": "4.1.13" } }, "sha512-eq3ouolC1oEFOAvOMOBAmfCIqZBJuvWvvYWh5h5iOYfe1HFC6+GZ6EIL0JdM3/niGRJmnrOc+8gl9/HGUaaptw=="],
 | 
			
		||||
    "@tailwindcss/node": ["@tailwindcss/node@4.1.14", "", { "dependencies": { "@jridgewell/remapping": "^2.3.4", "enhanced-resolve": "^5.18.3", "jiti": "^2.6.0", "lightningcss": "1.30.1", "magic-string": "^0.30.19", "source-map-js": "^1.2.1", "tailwindcss": "4.1.14" } }, "sha512-hpz+8vFk3Ic2xssIA3e01R6jkmsAhvkQdXlEbRTk6S10xDAtiQiM3FyvZVGsucefq764euO/b8WUW9ysLdThHw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide": ["@tailwindcss/oxide@4.1.13", "", { "dependencies": { "detect-libc": "^2.0.4", "tar": "^7.4.3" }, "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.1.13", "@tailwindcss/oxide-darwin-arm64": "4.1.13", "@tailwindcss/oxide-darwin-x64": "4.1.13", "@tailwindcss/oxide-freebsd-x64": "4.1.13", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.1.13", "@tailwindcss/oxide-linux-arm64-gnu": "4.1.13", "@tailwindcss/oxide-linux-arm64-musl": "4.1.13", "@tailwindcss/oxide-linux-x64-gnu": "4.1.13", "@tailwindcss/oxide-linux-x64-musl": "4.1.13", "@tailwindcss/oxide-wasm32-wasi": "4.1.13", "@tailwindcss/oxide-win32-arm64-msvc": "4.1.13", "@tailwindcss/oxide-win32-x64-msvc": "4.1.13" } }, "sha512-CPgsM1IpGRa880sMbYmG1s4xhAy3xEt1QULgTJGQmZUeNgXFR7s1YxYygmJyBGtou4SyEosGAGEeYqY7R53bIA=="],
 | 
			
		||||
    "@tailwindcss/oxide": ["@tailwindcss/oxide@4.1.14", "", { "dependencies": { "detect-libc": "^2.0.4", "tar": "^7.5.1" }, "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.1.14", "@tailwindcss/oxide-darwin-arm64": "4.1.14", "@tailwindcss/oxide-darwin-x64": "4.1.14", "@tailwindcss/oxide-freebsd-x64": "4.1.14", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.1.14", "@tailwindcss/oxide-linux-arm64-gnu": "4.1.14", "@tailwindcss/oxide-linux-arm64-musl": "4.1.14", "@tailwindcss/oxide-linux-x64-gnu": "4.1.14", "@tailwindcss/oxide-linux-x64-musl": "4.1.14", "@tailwindcss/oxide-wasm32-wasi": "4.1.14", "@tailwindcss/oxide-win32-arm64-msvc": "4.1.14", "@tailwindcss/oxide-win32-x64-msvc": "4.1.14" } }, "sha512-23yx+VUbBwCg2x5XWdB8+1lkPajzLmALEfMb51zZUBYaYVPDQvBSD/WYDqiVyBIo2BZFa3yw1Rpy3G2Jp+K0dw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.1.13", "", { "os": "android", "cpu": "arm64" }, "sha512-BrpTrVYyejbgGo57yc8ieE+D6VT9GOgnNdmh5Sac6+t0m+v+sKQevpFVpwX3pBrM2qKrQwJ0c5eDbtjouY/+ew=="],
 | 
			
		||||
    "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.1.14", "", { "os": "android", "cpu": "arm64" }, "sha512-a94ifZrGwMvbdeAxWoSuGcIl6/DOP5cdxagid7xJv6bwFp3oebp7y2ImYsnZBMTwjn5Ev5xESvS3FFYUGgPODQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.1.13", "", { "os": "darwin", "cpu": "arm64" }, "sha512-YP+Jksc4U0KHcu76UhRDHq9bx4qtBftp9ShK/7UGfq0wpaP96YVnnjFnj3ZFrUAjc5iECzODl/Ts0AN7ZPOANQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.1.14", "", { "os": "darwin", "cpu": "arm64" }, "sha512-HkFP/CqfSh09xCnrPJA7jud7hij5ahKyWomrC3oiO2U9i0UjP17o9pJbxUN0IJ471GTQQmzwhp0DEcpbp4MZTA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.1.13", "", { "os": "darwin", "cpu": "x64" }, "sha512-aAJ3bbwrn/PQHDxCto9sxwQfT30PzyYJFG0u/BWZGeVXi5Hx6uuUOQEI2Fa43qvmUjTRQNZnGqe9t0Zntexeuw=="],
 | 
			
		||||
    "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.1.14", "", { "os": "darwin", "cpu": "x64" }, "sha512-eVNaWmCgdLf5iv6Qd3s7JI5SEFBFRtfm6W0mphJYXgvnDEAZ5sZzqmI06bK6xo0IErDHdTA5/t7d4eTfWbWOFw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.1.13", "", { "os": "freebsd", "cpu": "x64" }, "sha512-Wt8KvASHwSXhKE/dJLCCWcTSVmBj3xhVhp/aF3RpAhGeZ3sVo7+NTfgiN8Vey/Fi8prRClDs6/f0KXPDTZE6nQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.1.14", "", { "os": "freebsd", "cpu": "x64" }, "sha512-QWLoRXNikEuqtNb0dhQN6wsSVVjX6dmUFzuuiL09ZeXju25dsei2uIPl71y2Ic6QbNBsB4scwBoFnlBfabHkEw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.1.13", "", { "os": "linux", "cpu": "arm" }, "sha512-mbVbcAsW3Gkm2MGwA93eLtWrwajz91aXZCNSkGTx/R5eb6KpKD5q8Ueckkh9YNboU8RH7jiv+ol/I7ZyQ9H7Bw=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.1.14", "", { "os": "linux", "cpu": "arm" }, "sha512-VB4gjQni9+F0VCASU+L8zSIyjrLLsy03sjcR3bM0V2g4SNamo0FakZFKyUQ96ZVwGK4CaJsc9zd/obQy74o0Fw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.1.13", "", { "os": "linux", "cpu": "arm64" }, "sha512-wdtfkmpXiwej/yoAkrCP2DNzRXCALq9NVLgLELgLim1QpSfhQM5+ZxQQF8fkOiEpuNoKLp4nKZ6RC4kmeFH0HQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.1.14", "", { "os": "linux", "cpu": "arm64" }, "sha512-qaEy0dIZ6d9vyLnmeg24yzA8XuEAD9WjpM5nIM1sUgQ/Zv7cVkharPDQcmm/t/TvXoKo/0knI3me3AGfdx6w1w=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.1.13", "", { "os": "linux", "cpu": "arm64" }, "sha512-hZQrmtLdhyqzXHB7mkXfq0IYbxegaqTmfa1p9MBj72WPoDD3oNOh1Lnxf6xZLY9C3OV6qiCYkO1i/LrzEdW2mg=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.1.14", "", { "os": "linux", "cpu": "arm64" }, "sha512-ISZjT44s59O8xKsPEIesiIydMG/sCXoMBCqsphDm/WcbnuWLxxb+GcvSIIA5NjUw6F8Tex7s5/LM2yDy8RqYBQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.1.13", "", { "os": "linux", "cpu": "x64" }, "sha512-uaZTYWxSXyMWDJZNY1Ul7XkJTCBRFZ5Fo6wtjrgBKzZLoJNrG+WderJwAjPzuNZOnmdrVg260DKwXCFtJ/hWRQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.1.14", "", { "os": "linux", "cpu": "x64" }, "sha512-02c6JhLPJj10L2caH4U0zF8Hji4dOeahmuMl23stk0MU1wfd1OraE7rOloidSF8W5JTHkFdVo/O7uRUJJnUAJg=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.1.13", "", { "os": "linux", "cpu": "x64" }, "sha512-oXiPj5mi4Hdn50v5RdnuuIms0PVPI/EG4fxAfFiIKQh5TgQgX7oSuDWntHW7WNIi/yVLAiS+CRGW4RkoGSSgVQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.1.14", "", { "os": "linux", "cpu": "x64" }, "sha512-TNGeLiN1XS66kQhxHG/7wMeQDOoL0S33x9BgmydbrWAb9Qw0KYdd8o1ifx4HOGDWhVmJ+Ul+JQ7lyknQFilO3Q=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.1.13", "", { "dependencies": { "@emnapi/core": "^1.4.5", "@emnapi/runtime": "^1.4.5", "@emnapi/wasi-threads": "^1.0.4", "@napi-rs/wasm-runtime": "^0.2.12", "@tybys/wasm-util": "^0.10.0", "tslib": "^2.8.0" }, "cpu": "none" }, "sha512-+LC2nNtPovtrDwBc/nqnIKYh/W2+R69FA0hgoeOn64BdCX522u19ryLh3Vf3F8W49XBcMIxSe665kwy21FkhvA=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.1.14", "", { "dependencies": { "@emnapi/core": "^1.5.0", "@emnapi/runtime": "^1.5.0", "@emnapi/wasi-threads": "^1.1.0", "@napi-rs/wasm-runtime": "^1.0.5", "@tybys/wasm-util": "^0.10.1", "tslib": "^2.4.0" }, "cpu": "none" }, "sha512-uZYAsaW/jS/IYkd6EWPJKW/NlPNSkWkBlaeVBi/WsFQNP05/bzkebUL8FH1pdsqx4f2fH/bWFcUABOM9nfiJkQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.1.13", "", { "os": "win32", "cpu": "arm64" }, "sha512-dziTNeQXtoQ2KBXmrjCxsuPk3F3CQ/yb7ZNZNA+UkNTeiTGgfeh+gH5Pi7mRncVgcPD2xgHvkFCh/MhZWSgyQg=="],
 | 
			
		||||
    "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.1.14", "", { "os": "win32", "cpu": "arm64" }, "sha512-Az0RnnkcvRqsuoLH2Z4n3JfAef0wElgzHD5Aky/e+0tBUxUhIeIqFBTMNQvmMRSP15fWwmvjBxZ3Q8RhsDnxAA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.1.13", "", { "os": "win32", "cpu": "x64" }, "sha512-3+LKesjXydTkHk5zXX01b5KMzLV1xl2mcktBJkje7rhFUpUlYJy7IMOLqjIRQncLTa1WZZiFY/foAeB5nmaiTw=="],
 | 
			
		||||
    "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.1.14", "", { "os": "win32", "cpu": "x64" }, "sha512-ttblVGHgf68kEE4om1n/n44I0yGPkCPbLsqzjvybhpwa6mKKtgFfAzy6btc3HRmuW7nHe0OOrSeNP9sQmmH9XA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/vite": ["@tailwindcss/vite@4.1.13", "", { "dependencies": { "@tailwindcss/node": "4.1.13", "@tailwindcss/oxide": "4.1.13", "tailwindcss": "4.1.13" }, "peerDependencies": { "vite": "^5.2.0 || ^6 || ^7" } }, "sha512-0PmqLQ010N58SbMTJ7BVJ4I2xopiQn/5i6nlb4JmxzQf8zcS5+m2Cv6tqh+sfDwtIdjoEnOvwsGQ1hkUi8QEHQ=="],
 | 
			
		||||
    "@tailwindcss/vite": ["@tailwindcss/vite@4.1.14", "", { "dependencies": { "@tailwindcss/node": "4.1.14", "@tailwindcss/oxide": "4.1.14", "tailwindcss": "4.1.14" }, "peerDependencies": { "vite": "^5.2.0 || ^6 || ^7" } }, "sha512-BoFUoU0XqgCUS1UXWhmDJroKKhNXeDzD7/XwabjkDIAbMnc4ULn5e2FuEuBbhZ6ENZoSYzKlzvZ44Yr6EUDUSA=="],
 | 
			
		||||
 | 
			
		||||
    "@tanstack/eslint-plugin-query": ["@tanstack/eslint-plugin-query@5.89.0", "", { "dependencies": { "@typescript-eslint/utils": "^8.37.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0" } }, "sha512-vz8TEuw9GO0xXIdreMpcofvOY17T3cjgob9bSFln8yQsKsbsUvtpvV3F8pVC3tZEDq0IwO++3/e0/+7YKEarNA=="],
 | 
			
		||||
    "@tanstack/eslint-plugin-query": ["@tanstack/eslint-plugin-query@5.91.0", "", { "dependencies": { "@typescript-eslint/utils": "^8.44.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0" } }, "sha512-Kn6yWyRe3dIPf7NqyDMhcsTBz2Oh8jPSOpBdlnLQhGBJ6iTMBFYA4B1UreGJ/WdfzQskSMh5imcyWF+wqa/Q5g=="],
 | 
			
		||||
 | 
			
		||||
    "@tanstack/query-core": ["@tanstack/query-core@5.89.0", "", {}, "sha512-joFV1MuPhSLsKfTzwjmPDrp8ENfZ9N23ymFu07nLfn3JCkSHy0CFgsyhHTJOmWaumC/WiNIKM0EJyduCF/Ih/Q=="],
 | 
			
		||||
    "@tanstack/query-core": ["@tanstack/query-core@5.90.3", "", {}, "sha512-HtPOnCwmx4dd35PfXU8jjkhwYrsHfuqgC8RCJIwWglmhIUIlzPP0ZcEkDAc+UtAWCiLm7T8rxeEfHZlz3hYMCA=="],
 | 
			
		||||
 | 
			
		||||
    "@tanstack/react-query": ["@tanstack/react-query@5.89.0", "", { "dependencies": { "@tanstack/query-core": "5.89.0" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-SXbtWSTSRXyBOe80mszPxpEbaN4XPRUp/i0EfQK1uyj3KCk/c8FuPJNIRwzOVe/OU3rzxrYtiNabsAmk1l714A=="],
 | 
			
		||||
    "@tanstack/react-query": ["@tanstack/react-query@5.90.3", "", { "dependencies": { "@tanstack/query-core": "5.90.3" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-i/LRL6DtuhG6bjGzavIMIVuKKPWx2AnEBIsBfuMm3YoHne0a20nWmsatOCBcVSaT0/8/5YFjNkebHAPLVUSi0Q=="],
 | 
			
		||||
 | 
			
		||||
    "@types/babel__core": ["@types/babel__core@7.20.5", "", { "dependencies": { "@babel/parser": "^7.20.7", "@babel/types": "^7.20.7", "@types/babel__generator": "*", "@types/babel__template": "*", "@types/babel__traverse": "*" } }, "sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA=="],
 | 
			
		||||
 | 
			
		||||
@@ -355,37 +355,37 @@
 | 
			
		||||
 | 
			
		||||
    "@types/ms": ["@types/ms@2.1.0", "", {}, "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA=="],
 | 
			
		||||
 | 
			
		||||
    "@types/node": ["@types/node@24.5.2", "", { "dependencies": { "undici-types": "~7.12.0" } }, "sha512-FYxk1I7wPv3K2XBaoyH2cTnocQEu8AOZ60hPbsyukMPLv5/5qr7V1i8PLHdl6Zf87I+xZXFvPCXYjiTFq+YSDQ=="],
 | 
			
		||||
    "@types/node": ["@types/node@24.7.2", "", { "dependencies": { "undici-types": "~7.14.0" } }, "sha512-/NbVmcGTP+lj5oa4yiYxxeBjRivKQ5Ns1eSZeB99ExsEQ6rX5XYU1Zy/gGxY/ilqtD4Etx9mKyrPxZRetiahhA=="],
 | 
			
		||||
 | 
			
		||||
    "@types/react": ["@types/react@19.1.13", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-hHkbU/eoO3EG5/MZkuFSKmYqPbSVk5byPFa3e7y/8TybHiLMACgI8seVYlicwk7H5K/rI2px9xrQp/C+AUDTiQ=="],
 | 
			
		||||
    "@types/react": ["@types/react@19.2.2", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-6mDvHUFSjyT2B2yeNx2nUgMxh9LtOWvkhIU3uePn2I2oyNymUAX1NIsdgviM4CH+JSrp2D2hsMvJOkxY+0wNRA=="],
 | 
			
		||||
 | 
			
		||||
    "@types/react-dom": ["@types/react-dom@19.1.9", "", { "peerDependencies": { "@types/react": "^19.0.0" } }, "sha512-qXRuZaOsAdXKFyOhRBg6Lqqc0yay13vN7KrIg4L7N4aaHN68ma9OK3NE1BoDFgFOTfM7zg+3/8+2n8rLUH3OKQ=="],
 | 
			
		||||
    "@types/react-dom": ["@types/react-dom@19.2.2", "", { "peerDependencies": { "@types/react": "^19.2.0" } }, "sha512-9KQPoO6mZCi7jcIStSnlOWn2nEF3mNmyr3rIAsGnAbQKYbRLyqmeSc39EVgtxXVia+LMT8j3knZLAZAh+xLmrw=="],
 | 
			
		||||
 | 
			
		||||
    "@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.44.0", "", { "dependencies": { "@eslint-community/regexpp": "^4.10.0", "@typescript-eslint/scope-manager": "8.44.0", "@typescript-eslint/type-utils": "8.44.0", "@typescript-eslint/utils": "8.44.0", "@typescript-eslint/visitor-keys": "8.44.0", "graphemer": "^1.4.0", "ignore": "^7.0.0", "natural-compare": "^1.4.0", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "@typescript-eslint/parser": "^8.44.0", "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-EGDAOGX+uwwekcS0iyxVDmRV9HX6FLSM5kzrAToLTsr9OWCIKG/y3lQheCq18yZ5Xh78rRKJiEpP0ZaCs4ryOQ=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.46.1", "", { "dependencies": { "@eslint-community/regexpp": "^4.10.0", "@typescript-eslint/scope-manager": "8.46.1", "@typescript-eslint/type-utils": "8.46.1", "@typescript-eslint/utils": "8.46.1", "@typescript-eslint/visitor-keys": "8.46.1", "graphemer": "^1.4.0", "ignore": "^7.0.0", "natural-compare": "^1.4.0", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "@typescript-eslint/parser": "^8.46.1", "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-rUsLh8PXmBjdiPY+Emjz9NX2yHvhS11v0SR6xNJkm5GM1MO9ea/1GoDKlHHZGrOJclL/cZ2i/vRUYVtjRhrHVQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser": ["@typescript-eslint/parser@8.44.0", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.44.0", "@typescript-eslint/types": "8.44.0", "@typescript-eslint/typescript-estree": "8.44.0", "@typescript-eslint/visitor-keys": "8.44.0", "debug": "^4.3.4" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-VGMpFQGUQWYT9LfnPcX8ouFojyrZ/2w3K5BucvxL/spdNehccKhB4jUyB1yBCXpr2XFm0jkECxgrpXBW2ipoAw=="],
 | 
			
		||||
    "@typescript-eslint/parser": ["@typescript-eslint/parser@8.46.1", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.46.1", "@typescript-eslint/types": "8.46.1", "@typescript-eslint/typescript-estree": "8.46.1", "@typescript-eslint/visitor-keys": "8.46.1", "debug": "^4.3.4" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-6JSSaBZmsKvEkbRUkf7Zj7dru/8ZCrJxAqArcLaVMee5907JdtEbKGsZ7zNiIm/UAkpGUkaSMZEXShnN2D1HZA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.44.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.44.0", "@typescript-eslint/types": "^8.44.0", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-ZeaGNraRsq10GuEohKTo4295Z/SuGcSq2LzfGlqiuEvfArzo/VRrT0ZaJsVPuKZ55lVbNk8U6FcL+ZMH8CoyVA=="],
 | 
			
		||||
    "@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.46.1", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.46.1", "@typescript-eslint/types": "^8.46.1", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-FOIaFVMHzRskXr5J4Jp8lFVV0gz5ngv3RHmn+E4HYxSJ3DgDzU7fVI1/M7Ijh1zf6S7HIoaIOtln1H5y8V+9Zg=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.43.0", "", { "dependencies": { "@typescript-eslint/types": "8.43.0", "@typescript-eslint/visitor-keys": "8.43.0" } }, "sha512-daSWlQ87ZhsjrbMLvpuuMAt3y4ba57AuvadcR7f3nl8eS3BjRc8L9VLxFLk92RL5xdXOg6IQ+qKjjqNEimGuAg=="],
 | 
			
		||||
    "@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.45.0", "", { "dependencies": { "@typescript-eslint/types": "8.45.0", "@typescript-eslint/visitor-keys": "8.45.0" } }, "sha512-clmm8XSNj/1dGvJeO6VGH7EUSeA0FMs+5au/u3lrA3KfG8iJ4u8ym9/j2tTEoacAffdW1TVUzXO30W1JTJS7dA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.44.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-x5Y0+AuEPqAInc6yd0n5DAcvtoQ/vyaGwuX5HE9n6qAefk1GaedqrLQF8kQGylLUb9pnZyLf+iEiL9fr8APDtQ=="],
 | 
			
		||||
    "@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.46.1", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-X88+J/CwFvlJB+mK09VFqx5FE4H5cXD+H/Bdza2aEWkSb8hnWIQorNcscRl4IEo1Cz9VI/+/r/jnGWkbWPx54g=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils": ["@typescript-eslint/type-utils@8.44.0", "", { "dependencies": { "@typescript-eslint/types": "8.44.0", "@typescript-eslint/typescript-estree": "8.44.0", "@typescript-eslint/utils": "8.44.0", "debug": "^4.3.4", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-9cwsoSxJ8Sak67Be/hD2RNt/fsqmWnNE1iHohG8lxqLSNY8xNfyY7wloo5zpW3Nu9hxVgURevqfcH6vvKCt6yg=="],
 | 
			
		||||
    "@typescript-eslint/type-utils": ["@typescript-eslint/type-utils@8.46.1", "", { "dependencies": { "@typescript-eslint/types": "8.46.1", "@typescript-eslint/typescript-estree": "8.46.1", "@typescript-eslint/utils": "8.46.1", "debug": "^4.3.4", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-+BlmiHIiqufBxkVnOtFwjah/vrkF4MtKKvpXrKSPLCkCtAp8H01/VV43sfqA98Od7nJpDcFnkwgyfQbOG0AMvw=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/types": ["@typescript-eslint/types@8.43.0", "", {}, "sha512-vQ2FZaxJpydjSZJKiSW/LJsabFFvV7KgLC5DiLhkBcykhQj8iK9BOaDmQt74nnKdLvceM5xmhaTF+pLekrxEkw=="],
 | 
			
		||||
    "@typescript-eslint/types": ["@typescript-eslint/types@8.45.0", "", {}, "sha512-WugXLuOIq67BMgQInIxxnsSyRLFxdkJEJu8r4ngLR56q/4Q5LrbfkFRH27vMTjxEK8Pyz7QfzuZe/G15qQnVRA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.44.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.44.0", "@typescript-eslint/tsconfig-utils": "8.44.0", "@typescript-eslint/types": "8.44.0", "@typescript-eslint/visitor-keys": "8.44.0", "debug": "^4.3.4", "fast-glob": "^3.3.2", "is-glob": "^4.0.3", "minimatch": "^9.0.4", "semver": "^7.6.0", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-lqNj6SgnGcQZwL4/SBJ3xdPEfcBuhCG8zdcwCPgYcmiPLgokiNDKlbPzCwEwu7m279J/lBYWtDYL+87OEfn8Jw=="],
 | 
			
		||||
    "@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.46.1", "", { "dependencies": { "@typescript-eslint/project-service": "8.46.1", "@typescript-eslint/tsconfig-utils": "8.46.1", "@typescript-eslint/types": "8.46.1", "@typescript-eslint/visitor-keys": "8.46.1", "debug": "^4.3.4", "fast-glob": "^3.3.2", "is-glob": "^4.0.3", "minimatch": "^9.0.4", "semver": "^7.6.0", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-uIifjT4s8cQKFQ8ZBXXyoUODtRoAd7F7+G8MKmtzj17+1UbdzFl52AzRyZRyKqPHhgzvXunnSckVu36flGy8cg=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils": ["@typescript-eslint/utils@8.43.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.43.0", "@typescript-eslint/types": "8.43.0", "@typescript-eslint/typescript-estree": "8.43.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-S1/tEmkUeeswxd0GGcnwuVQPFWo8NzZTOMxCvw8BX7OMxnNae+i8Tm7REQen/SwUIPoPqfKn7EaZ+YLpiB3k9g=="],
 | 
			
		||||
    "@typescript-eslint/utils": ["@typescript-eslint/utils@8.45.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.45.0", "@typescript-eslint/types": "8.45.0", "@typescript-eslint/typescript-estree": "8.45.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-bxi1ht+tLYg4+XV2knz/F7RVhU0k6VrSMc9sb8DQ6fyCTrGQLHfo7lDtN0QJjZjKkLA2ThrKuCdHEvLReqtIGg=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.44.0", "", { "dependencies": { "@typescript-eslint/types": "8.44.0", "eslint-visitor-keys": "^4.2.1" } }, "sha512-zaz9u8EJ4GBmnehlrpoKvj/E3dNbuQ7q0ucyZImm3cLqJ8INTc970B1qEqDX/Rzq65r3TvVTN7kHWPBoyW7DWw=="],
 | 
			
		||||
    "@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.46.1", "", { "dependencies": { "@typescript-eslint/types": "8.46.1", "eslint-visitor-keys": "^4.2.1" } }, "sha512-ptkmIf2iDkNUjdeu2bQqhFPV1m6qTnFFjg7PPDjxKWaMaP0Z6I9l30Jr3g5QqbZGdw8YdYvLp+XnqnWWZOg/NA=="],
 | 
			
		||||
 | 
			
		||||
    "@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="],
 | 
			
		||||
 | 
			
		||||
    "@vitejs/plugin-react": ["@vitejs/plugin-react@5.0.3", "", { "dependencies": { "@babel/core": "^7.28.4", "@babel/plugin-transform-react-jsx-self": "^7.27.1", "@babel/plugin-transform-react-jsx-source": "^7.27.1", "@rolldown/pluginutils": "1.0.0-beta.35", "@types/babel__core": "^7.20.5", "react-refresh": "^0.17.0" }, "peerDependencies": { "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0" } }, "sha512-PFVHhosKkofGH0Yzrw1BipSedTH68BFF8ZWy1kfUpCtJcouXXY0+racG8sExw7hw0HoX36813ga5o3LTWZ4FUg=="],
 | 
			
		||||
    "@vitejs/plugin-react": ["@vitejs/plugin-react@5.0.4", "", { "dependencies": { "@babel/core": "^7.28.4", "@babel/plugin-transform-react-jsx-self": "^7.27.1", "@babel/plugin-transform-react-jsx-source": "^7.27.1", "@rolldown/pluginutils": "1.0.0-beta.38", "@types/babel__core": "^7.20.5", "react-refresh": "^0.17.0" }, "peerDependencies": { "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0" } }, "sha512-La0KD0vGkVkSk6K+piWDKRUyg8Rl5iAIKRMH0vMJI0Eg47bq1eOxmoObAaQG37WMW9MSyk7Cs8EIWwJC1PtzKA=="],
 | 
			
		||||
 | 
			
		||||
    "acorn": ["acorn@8.15.0", "", { "bin": { "acorn": "bin/acorn" } }, "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg=="],
 | 
			
		||||
 | 
			
		||||
@@ -491,11 +491,11 @@
 | 
			
		||||
 | 
			
		||||
    "escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
 | 
			
		||||
 | 
			
		||||
    "eslint": ["eslint@9.35.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.8.0", "@eslint-community/regexpp": "^4.12.1", "@eslint/config-array": "^0.21.0", "@eslint/config-helpers": "^0.3.1", "@eslint/core": "^0.15.2", "@eslint/eslintrc": "^3.3.1", "@eslint/js": "9.35.0", "@eslint/plugin-kit": "^0.3.5", "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", "@humanwhocodes/retry": "^0.4.2", "@types/estree": "^1.0.6", "@types/json-schema": "^7.0.15", "ajv": "^6.12.4", "chalk": "^4.0.0", "cross-spawn": "^7.0.6", "debug": "^4.3.2", "escape-string-regexp": "^4.0.0", "eslint-scope": "^8.4.0", "eslint-visitor-keys": "^4.2.1", "espree": "^10.4.0", "esquery": "^1.5.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", "file-entry-cache": "^8.0.0", "find-up": "^5.0.0", "glob-parent": "^6.0.2", "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", "json-stable-stringify-without-jsonify": "^1.0.1", "lodash.merge": "^4.6.2", "minimatch": "^3.1.2", "natural-compare": "^1.4.0", "optionator": "^0.9.3" }, "peerDependencies": { "jiti": "*" }, "optionalPeers": ["jiti"], "bin": { "eslint": "bin/eslint.js" } }, "sha512-QePbBFMJFjgmlE+cXAlbHZbHpdFVS2E/6vzCy7aKlebddvl1vadiC4JFV5u/wqTkNUwEV8WrQi257jf5f06hrg=="],
 | 
			
		||||
    "eslint": ["eslint@9.37.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.8.0", "@eslint-community/regexpp": "^4.12.1", "@eslint/config-array": "^0.21.0", "@eslint/config-helpers": "^0.4.0", "@eslint/core": "^0.16.0", "@eslint/eslintrc": "^3.3.1", "@eslint/js": "9.37.0", "@eslint/plugin-kit": "^0.4.0", "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", "@humanwhocodes/retry": "^0.4.2", "@types/estree": "^1.0.6", "@types/json-schema": "^7.0.15", "ajv": "^6.12.4", "chalk": "^4.0.0", "cross-spawn": "^7.0.6", "debug": "^4.3.2", "escape-string-regexp": "^4.0.0", "eslint-scope": "^8.4.0", "eslint-visitor-keys": "^4.2.1", "espree": "^10.4.0", "esquery": "^1.5.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", "file-entry-cache": "^8.0.0", "find-up": "^5.0.0", "glob-parent": "^6.0.2", "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", "json-stable-stringify-without-jsonify": "^1.0.1", "lodash.merge": "^4.6.2", "minimatch": "^3.1.2", "natural-compare": "^1.4.0", "optionator": "^0.9.3" }, "peerDependencies": { "jiti": "*" }, "optionalPeers": ["jiti"], "bin": { "eslint": "bin/eslint.js" } }, "sha512-XyLmROnACWqSxiGYArdef1fItQd47weqB7iwtfr9JHwRrqIXZdcFMvvEcL9xHCmL0SNsOvF0c42lWyM1U5dgig=="],
 | 
			
		||||
 | 
			
		||||
    "eslint-plugin-react-hooks": ["eslint-plugin-react-hooks@5.2.0", "", { "peerDependencies": { "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0" } }, "sha512-+f15FfK64YQwZdJNELETdn5ibXEUQmW1DZL6KXhNnc2heoy/sg9VJJeT7n8TlMWouzWqSWavFkIhHyIbIAEapg=="],
 | 
			
		||||
    "eslint-plugin-react-hooks": ["eslint-plugin-react-hooks@7.0.0", "", { "dependencies": { "@babel/core": "^7.24.4", "@babel/parser": "^7.24.4", "hermes-parser": "^0.25.1", "zod": "^3.22.4 || ^4.0.0", "zod-validation-error": "^3.0.3 || ^4.0.0" }, "peerDependencies": { "eslint": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0 || ^7.0.0 || ^8.0.0-0 || ^9.0.0" } }, "sha512-fNXaOwvKwq2+pXiRpXc825Vd63+KM4DLL40Rtlycb8m7fYpp6efrTp1sa6ZbP/Ap58K2bEKFXRmhURE+CJAQWw=="],
 | 
			
		||||
 | 
			
		||||
    "eslint-plugin-react-refresh": ["eslint-plugin-react-refresh@0.4.20", "", { "peerDependencies": { "eslint": ">=8.40" } }, "sha512-XpbHQ2q5gUF8BGOX4dHe+71qoirYMhApEPZ7sfhF/dNnOF1UXnCMGZf79SFTBO7Bz5YEIT4TMieSlJBWhP9WBA=="],
 | 
			
		||||
    "eslint-plugin-react-refresh": ["eslint-plugin-react-refresh@0.4.23", "", { "peerDependencies": { "eslint": ">=8.40" } }, "sha512-G4j+rv0NmbIR45kni5xJOrYvCtyD3/7LjpVH8MPPcudXDcNu8gv+4ATTDXTtbRR8rTCM5HxECvCSsRmxKnWDsA=="],
 | 
			
		||||
 | 
			
		||||
    "eslint-scope": ["eslint-scope@8.4.0", "", { "dependencies": { "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } }, "sha512-sNXOfKCn74rt8RICKMvJS7XKV/Xk9kA7DyJr8mJik3S7Cwgy3qlkkmyS2uQB3jiJg6VNdZd/pDBJu0nvG2NlTg=="],
 | 
			
		||||
 | 
			
		||||
@@ -575,11 +575,15 @@
 | 
			
		||||
 | 
			
		||||
    "hast-util-whitespace": ["hast-util-whitespace@3.0.0", "", { "dependencies": { "@types/hast": "^3.0.0" } }, "sha512-88JUN06ipLwsnv+dVn+OIYOvAuvBMy/Qoi6O7mQHxdPXpjy+Cd6xRkWwux7DKO+4sYILtLBRIKgsdpS2gQc7qw=="],
 | 
			
		||||
 | 
			
		||||
    "hermes-estree": ["hermes-estree@0.25.1", "", {}, "sha512-0wUoCcLp+5Ev5pDW2OriHC2MJCbwLwuRx+gAqMTOkGKJJiBCLjtrvy4PWUGn6MIVefecRpzoOZ/UV6iGdOr+Cw=="],
 | 
			
		||||
 | 
			
		||||
    "hermes-parser": ["hermes-parser@0.25.1", "", { "dependencies": { "hermes-estree": "0.25.1" } }, "sha512-6pEjquH3rqaI6cYAXYPcz9MS4rY6R4ngRgrgfDshRptUZIc3lw0MCIJIGDj9++mfySOuPTHB4nrSW99BCvOPIA=="],
 | 
			
		||||
 | 
			
		||||
    "html-parse-stringify": ["html-parse-stringify@3.0.1", "", { "dependencies": { "void-elements": "3.1.0" } }, "sha512-KknJ50kTInJ7qIScF3jeaFRpMpE8/lfiTdzf/twXyPBLAGrLRTmkz3AdTnKeh40X8k9L2fdYwEp/42WGXIRGcg=="],
 | 
			
		||||
 | 
			
		||||
    "html-url-attributes": ["html-url-attributes@3.0.1", "", {}, "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ=="],
 | 
			
		||||
 | 
			
		||||
    "i18next": ["i18next@25.5.2", "", { "dependencies": { "@babel/runtime": "^7.27.6" }, "peerDependencies": { "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-lW8Zeh37i/o0zVr+NoCHfNnfvVw+M6FQbRp36ZZ/NyHDJ3NJVpp2HhAUyU9WafL5AssymNoOjMRB48mmx2P6Hw=="],
 | 
			
		||||
    "i18next": ["i18next@25.6.0", "", { "dependencies": { "@babel/runtime": "^7.27.6" }, "peerDependencies": { "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-tTn8fLrwBYtnclpL5aPXK/tAYBLWVvoHM1zdfXoRNLcI+RvtMsoZRV98ePlaW3khHYKuNh/Q65W/+NVFUeIwVw=="],
 | 
			
		||||
 | 
			
		||||
    "i18next-browser-languagedetector": ["i18next-browser-languagedetector@8.2.0", "", { "dependencies": { "@babel/runtime": "^7.23.2" } }, "sha512-P+3zEKLnOF0qmiesW383vsLdtQVyKtCNA9cjSoKCppTKPQVfKd2W8hbVo5ZhNJKDqeM7BOcvNoKJOjpHh4Js9g=="],
 | 
			
		||||
 | 
			
		||||
@@ -663,7 +667,7 @@
 | 
			
		||||
 | 
			
		||||
    "lru-cache": ["lru-cache@5.1.1", "", { "dependencies": { "yallist": "^3.0.2" } }, "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w=="],
 | 
			
		||||
 | 
			
		||||
    "lucide-react": ["lucide-react@0.544.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-t5tS44bqd825zAW45UQxpG2CvcC4urOwn2TrwSH8u+MjeE+1NnWl6QqeQ/6NdjMqdOygyiT9p3Ev0p1NJykxjw=="],
 | 
			
		||||
    "lucide-react": ["lucide-react@0.545.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-7r1/yUuflQDSt4f1bpn5ZAocyIxcTyVyBBChSVtBKn5M+392cPmI5YJMWOJKk/HUWGm5wg83chlAZtCcGbEZtw=="],
 | 
			
		||||
 | 
			
		||||
    "magic-string": ["magic-string@0.30.19", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-2N21sPY9Ws53PZvsEpVtNuSW+ScYbQdp4b9qUaL+9QkHUrGFKo56Lg9Emg5s9V/qrtNBmiR01sYhUOwu3H+VOw=="],
 | 
			
		||||
 | 
			
		||||
@@ -739,9 +743,7 @@
 | 
			
		||||
 | 
			
		||||
    "minipass": ["minipass@7.1.2", "", {}, "sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw=="],
 | 
			
		||||
 | 
			
		||||
    "minizlib": ["minizlib@3.0.2", "", { "dependencies": { "minipass": "^7.1.2" } }, "sha512-oG62iEk+CYt5Xj2YqI5Xi9xWUeZhDI8jjQmC5oThVH5JGCTgIjr7ciJDzC7MBzYd//WvR1OTmP5Q38Q8ShQtVA=="],
 | 
			
		||||
 | 
			
		||||
    "mkdirp": ["mkdirp@3.0.1", "", { "bin": { "mkdirp": "dist/cjs/src/bin.js" } }, "sha512-+NsyUUAZDmo6YVHzL/stxSu3t9YS1iljliy3BSDrXJ/dkn1KYdmtZODGGjLcc9XLgVVpH4KshHB8XmZgMhaBXg=="],
 | 
			
		||||
    "minizlib": ["minizlib@3.1.0", "", { "dependencies": { "minipass": "^7.1.2" } }, "sha512-KZxYo1BUkWD2TVFLr0MQoM8vUUigWD3LlD83a/75BqC+4qE0Hb1Vo5v1FgcfaNXvfXzr+5EhQ6ing/CaBijTlw=="],
 | 
			
		||||
 | 
			
		||||
    "ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="],
 | 
			
		||||
 | 
			
		||||
@@ -785,13 +787,13 @@
 | 
			
		||||
 | 
			
		||||
    "queue-microtask": ["queue-microtask@1.2.3", "", {}, "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A=="],
 | 
			
		||||
 | 
			
		||||
    "react": ["react@19.1.1", "", {}, "sha512-w8nqGImo45dmMIfljjMwOGtbmC/mk4CMYhWIicdSflH91J9TyCyczcPFXJzrZ/ZXcgGRFeP6BU0BEJTw6tZdfQ=="],
 | 
			
		||||
    "react": ["react@19.2.0", "", {}, "sha512-tmbWg6W31tQLeB5cdIBOicJDJRR2KzXsV7uSK9iNfLWQ5bIZfxuPEHp7M8wiHyHnn0DD1i7w3Zmin0FtkrwoCQ=="],
 | 
			
		||||
 | 
			
		||||
    "react-dom": ["react-dom@19.1.1", "", { "dependencies": { "scheduler": "^0.26.0" }, "peerDependencies": { "react": "^19.1.1" } }, "sha512-Dlq/5LAZgF0Gaz6yiqZCf6VCcZs1ghAJyrsu84Q/GT0gV+mCxbfmKNoGRKBYMJ8IEdGPqu49YWXD02GCknEDkw=="],
 | 
			
		||||
    "react-dom": ["react-dom@19.2.0", "", { "dependencies": { "scheduler": "^0.27.0" }, "peerDependencies": { "react": "^19.2.0" } }, "sha512-UlbRu4cAiGaIewkPyiRGJk0imDN2T3JjieT6spoL2UeSf5od4n5LB/mQ4ejmxhCFT1tYe8IvaFulzynWovsEFQ=="],
 | 
			
		||||
 | 
			
		||||
    "react-hook-form": ["react-hook-form@7.62.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17 || ^18 || ^19" } }, "sha512-7KWFejc98xqG/F4bAxpL41NB3o1nnvQO1RWZT3TqRZYL8RryQETGfEdVnJN2fy1crCiBLLjkRBVK05j24FxJGA=="],
 | 
			
		||||
    "react-hook-form": ["react-hook-form@7.65.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17 || ^18 || ^19" } }, "sha512-xtOzDz063WcXvGWaHgLNrNzlsdFgtUWcb32E6WFaGTd7kPZG3EeDusjdZfUsPwKCKVXy1ZlntifaHZ4l8pAsmw=="],
 | 
			
		||||
 | 
			
		||||
    "react-i18next": ["react-i18next@15.7.3", "", { "dependencies": { "@babel/runtime": "^7.27.6", "html-parse-stringify": "^3.0.1" }, "peerDependencies": { "i18next": ">= 25.4.1", "react": ">= 16.8.0", "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-AANws4tOE+QSq/IeMF/ncoHlMNZaVLxpa5uUGW1wjike68elVYr0018L9xYoqBr1OFO7G7boDPrbn0HpMCJxTw=="],
 | 
			
		||||
    "react-i18next": ["react-i18next@16.0.1", "", { "dependencies": { "@babel/runtime": "^7.27.6", "html-parse-stringify": "^3.0.1" }, "peerDependencies": { "i18next": ">= 25.5.2", "react": ">= 16.8.0", "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-0S//bpYEkCPjzuVmxDf9Z6+Y+ArNvpAUk7eDL4qNCZXjDh6Z9j6MZ+NThU7kMCOsmYmDCun3GYEwkiOjjZo9Ug=="],
 | 
			
		||||
 | 
			
		||||
    "react-markdown": ["react-markdown@10.1.0", "", { "dependencies": { "@types/hast": "^3.0.0", "@types/mdast": "^4.0.0", "devlop": "^1.0.0", "hast-util-to-jsx-runtime": "^2.0.0", "html-url-attributes": "^3.0.0", "mdast-util-to-hast": "^13.0.0", "remark-parse": "^11.0.0", "remark-rehype": "^11.0.0", "unified": "^11.0.0", "unist-util-visit": "^5.0.0", "vfile": "^6.0.0" }, "peerDependencies": { "@types/react": ">=18", "react": ">=18" } }, "sha512-qKxVopLT/TyA6BX3Ue5NwabOsAzm0Q7kAPwq6L+wWDwisYs7R8vZ0nRXqq6rkueboxpkjvLGU9fWifiX/ZZFxQ=="],
 | 
			
		||||
 | 
			
		||||
@@ -801,7 +803,7 @@
 | 
			
		||||
 | 
			
		||||
    "react-remove-scroll-bar": ["react-remove-scroll-bar@2.3.8", "", { "dependencies": { "react-style-singleton": "^2.2.2", "tslib": "^2.0.0" }, "peerDependencies": { "@types/react": "*", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" }, "optionalPeers": ["@types/react"] }, "sha512-9r+yi9+mgU33AKcj6IbT9oRCO78WriSj6t/cF8DWBZJ9aOGPOTEDvdUDz1FwKim7QXWwmHqtdHnRJfhAxEG46Q=="],
 | 
			
		||||
 | 
			
		||||
    "react-router": ["react-router@7.9.1", "", { "dependencies": { "cookie": "^1.0.1", "set-cookie-parser": "^2.6.0" }, "peerDependencies": { "react": ">=18", "react-dom": ">=18" }, "optionalPeers": ["react-dom"] }, "sha512-pfAByjcTpX55mqSDGwGnY9vDCpxqBLASg0BMNAuMmpSGESo/TaOUG6BllhAtAkCGx8Rnohik/XtaqiYUJtgW2g=="],
 | 
			
		||||
    "react-router": ["react-router@7.9.4", "", { "dependencies": { "cookie": "^1.0.1", "set-cookie-parser": "^2.6.0" }, "peerDependencies": { "react": ">=18", "react-dom": ">=18" }, "optionalPeers": ["react-dom"] }, "sha512-SD3G8HKviFHg9xj7dNODUKDFgpG4xqD5nhyd0mYoB5iISepuZAvzSr8ywxgxKJ52yRzf/HWtVHc9AWwoTbljvA=="],
 | 
			
		||||
 | 
			
		||||
    "react-style-singleton": ["react-style-singleton@2.2.3", "", { "dependencies": { "get-nonce": "^1.0.0", "tslib": "^2.0.0" }, "peerDependencies": { "@types/react": "*", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc" }, "optionalPeers": ["@types/react"] }, "sha512-b6jSvxvVnyptAiLjbkWLE/lOnR4lfTtDAl+eUC7RZy+QQWc6wRzIV2CE6xBuMmDxc2qIihtDCZD5NPOFl7fRBQ=="],
 | 
			
		||||
 | 
			
		||||
@@ -817,7 +819,7 @@
 | 
			
		||||
 | 
			
		||||
    "run-parallel": ["run-parallel@1.2.0", "", { "dependencies": { "queue-microtask": "^1.2.2" } }, "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA=="],
 | 
			
		||||
 | 
			
		||||
    "scheduler": ["scheduler@0.26.0", "", {}, "sha512-NlHwttCI/l5gCPR3D1nNXtWABUmBwvZpEQiD4IXSbIDq8BzLIK/7Ir5gTFSGZDUu37K5cMNp0hFtzO38sC7gWA=="],
 | 
			
		||||
    "scheduler": ["scheduler@0.27.0", "", {}, "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q=="],
 | 
			
		||||
 | 
			
		||||
    "semver": ["semver@6.3.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA=="],
 | 
			
		||||
 | 
			
		||||
@@ -845,11 +847,11 @@
 | 
			
		||||
 | 
			
		||||
    "tailwind-merge": ["tailwind-merge@3.3.1", "", {}, "sha512-gBXpgUm/3rp1lMZZrM/w7D8GKqshif0zAymAhbCyIt8KMe+0v9DQ7cdYLR4FHH/cKpdTXb+A/tKKU3eolfsI+g=="],
 | 
			
		||||
 | 
			
		||||
    "tailwindcss": ["tailwindcss@4.1.13", "", {}, "sha512-i+zidfmTqtwquj4hMEwdjshYYgMbOrPzb9a0M3ZgNa0JMoZeFC6bxZvO8yr8ozS6ix2SDz0+mvryPeBs2TFE+w=="],
 | 
			
		||||
    "tailwindcss": ["tailwindcss@4.1.14", "", {}, "sha512-b7pCxjGO98LnxVkKjaZSDeNuljC4ueKUddjENJOADtubtdo8llTaJy7HwBMeLNSSo2N5QIAgklslK1+Ir8r6CA=="],
 | 
			
		||||
 | 
			
		||||
    "tapable": ["tapable@2.2.1", "", {}, "sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ=="],
 | 
			
		||||
 | 
			
		||||
    "tar": ["tar@7.4.3", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.0.1", "mkdirp": "^3.0.1", "yallist": "^5.0.0" } }, "sha512-5S7Va8hKfV7W5U6g3aYxXmlPoZVAwUMy9AOKyF2fVuZa2UD3qZjg578OrLRt8PcNN1PleVaL/5/yYATNL0ICUw=="],
 | 
			
		||||
    "tar": ["tar@7.5.1", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.1.0", "yallist": "^5.0.0" } }, "sha512-nlGpxf+hv0v7GkWBK2V9spgactGOp0qvfWRxUMjqHyzrt3SgwE48DIv/FhqPHJYLHpgW1opq3nERbz5Anq7n1g=="],
 | 
			
		||||
 | 
			
		||||
    "tinyglobby": ["tinyglobby@0.2.15", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.3" } }, "sha512-j2Zq4NyQYG5XMST4cbs02Ak8iJUdxRM0XI5QyxXuZOzKOINmWurp3smXu3y5wDcJrptwpSjgXHzIQxR0omXljQ=="],
 | 
			
		||||
 | 
			
		||||
@@ -863,15 +865,15 @@
 | 
			
		||||
 | 
			
		||||
    "tslib": ["tslib@2.8.1", "", {}, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
 | 
			
		||||
 | 
			
		||||
    "tw-animate-css": ["tw-animate-css@1.3.8", "", {}, "sha512-Qrk3PZ7l7wUcGYhwZloqfkWCmaXZAoqjkdbIDvzfGshwGtexa/DAs9koXxIkrpEasyevandomzCBAV1Yyop5rw=="],
 | 
			
		||||
    "tw-animate-css": ["tw-animate-css@1.4.0", "", {}, "sha512-7bziOlRqH0hJx80h/3mbicLW7o8qLsH5+RaLR2t+OHM3D0JlWGODQKQ4cxbK7WlvmUxpcj6Kgu6EKqjrGFe3QQ=="],
 | 
			
		||||
 | 
			
		||||
    "type-check": ["type-check@0.4.0", "", { "dependencies": { "prelude-ls": "^1.2.1" } }, "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew=="],
 | 
			
		||||
 | 
			
		||||
    "typescript": ["typescript@5.9.2", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-CWBzXQrc/qOkhidw1OzBTQuYRbfyxDXJMVJ1XNwUHGROVmuaeiEm3OslpZ1RV96d7SKKjZKrSJu3+t/xlw3R9A=="],
 | 
			
		||||
    "typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint": ["typescript-eslint@8.44.0", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.44.0", "@typescript-eslint/parser": "8.44.0", "@typescript-eslint/typescript-estree": "8.44.0", "@typescript-eslint/utils": "8.44.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-ib7mCkYuIzYonCq9XWF5XNw+fkj2zg629PSa9KNIQ47RXFF763S5BIX4wqz1+FLPogTZoiw8KmCiRPRa8bL3qw=="],
 | 
			
		||||
    "typescript-eslint": ["typescript-eslint@8.46.1", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.46.1", "@typescript-eslint/parser": "8.46.1", "@typescript-eslint/typescript-estree": "8.46.1", "@typescript-eslint/utils": "8.46.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-VHgijW803JafdSsDO8I761r3SHrgk4T00IdyQ+/UsthtgPRsBWQLqoSxOolxTpxRKi1kGXK0bSz4CoAc9ObqJA=="],
 | 
			
		||||
 | 
			
		||||
    "undici-types": ["undici-types@7.12.0", "", {}, "sha512-goOacqME2GYyOZZfb5Lgtu+1IDmAlAEu5xnD3+xTzS10hT0vzpf0SPjkXwAw9Jm+4n/mQGDP3LO8CPbYROeBfQ=="],
 | 
			
		||||
    "undici-types": ["undici-types@7.14.0", "", {}, "sha512-QQiYxHuyZ9gQUIrmPo3IA+hUl4KYk8uSA7cHrcKd/l3p1OTpZcM0Tbp9x7FAtXdAYhlasd60ncPpgu6ihG6TOA=="],
 | 
			
		||||
 | 
			
		||||
    "unified": ["unified@11.0.5", "", { "dependencies": { "@types/unist": "^3.0.0", "bail": "^2.0.0", "devlop": "^1.0.0", "extend": "^3.0.0", "is-plain-obj": "^4.0.0", "trough": "^2.0.0", "vfile": "^6.0.0" } }, "sha512-xKvGhPWw3k84Qjh8bI3ZeJjqnyadK+GEFtazSfZv/rKeTkTjOJho6mFqh2SM96iIcZokxiOpg78GazTSg8+KHA=="],
 | 
			
		||||
 | 
			
		||||
@@ -897,7 +899,7 @@
 | 
			
		||||
 | 
			
		||||
    "vfile-message": ["vfile-message@4.0.2", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-stringify-position": "^4.0.0" } }, "sha512-jRDZ1IMLttGj41KcZvlrYAaI3CfqpLpfpf+Mfig13viT6NKvRzWZ+lXz0Y5D60w6uJIBAOGq9mSHf0gktF0duw=="],
 | 
			
		||||
 | 
			
		||||
    "vite": ["vite@7.1.6", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.5.0", "picomatch": "^4.0.3", "postcss": "^8.5.6", "rollup": "^4.43.0", "tinyglobby": "^0.2.15" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^20.19.0 || >=22.12.0", "jiti": ">=1.21.0", "less": "^4.0.0", "lightningcss": "^1.21.0", "sass": "^1.70.0", "sass-embedded": "^1.70.0", "stylus": ">=0.54.8", "sugarss": "^5.0.0", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-SRYIB8t/isTwNn8vMB3MR6E+EQZM/WG1aKmmIUCfDXfVvKfc20ZpamngWHKzAmmu9ppsgxsg4b2I7c90JZudIQ=="],
 | 
			
		||||
    "vite": ["vite@7.1.10", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.5.0", "picomatch": "^4.0.3", "postcss": "^8.5.6", "rollup": "^4.43.0", "tinyglobby": "^0.2.15" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^20.19.0 || >=22.12.0", "jiti": ">=1.21.0", "less": "^4.0.0", "lightningcss": "^1.21.0", "sass": "^1.70.0", "sass-embedded": "^1.70.0", "stylus": ">=0.54.8", "sugarss": "^5.0.0", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-CmuvUBzVJ/e3HGxhg6cYk88NGgTnBoOo7ogtfJJ0fefUWAxN/WDSUa50o+oVBxuIhO8FoEZW0j2eW7sfjs5EtA=="],
 | 
			
		||||
 | 
			
		||||
    "void-elements": ["void-elements@3.1.0", "", {}, "sha512-Dhxzh5HZuiHQhbvTW9AMetFfBHDMYpo23Uo9btPXgdYP+3T5S+p+jgNy7spra+veYhBP2dCSgxR/i2Y02h5/6w=="],
 | 
			
		||||
 | 
			
		||||
@@ -909,7 +911,9 @@
 | 
			
		||||
 | 
			
		||||
    "yocto-queue": ["yocto-queue@0.1.0", "", {}, "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q=="],
 | 
			
		||||
 | 
			
		||||
    "zod": ["zod@4.1.9", "", {}, "sha512-HI32jTq0AUAC125z30E8bQNz0RQ+9Uc+4J7V97gLYjZVKRjeydPgGt6dvQzFrav7MYOUGFqqOGiHpA/fdbd0cQ=="],
 | 
			
		||||
    "zod": ["zod@4.1.12", "", {}, "sha512-JInaHOamG8pt5+Ey8kGmdcAcg3OL9reK8ltczgHTAwNhMys/6ThXHityHxVV2p3fkw/c+MAvBHFVYHFZDmjMCQ=="],
 | 
			
		||||
 | 
			
		||||
    "zod-validation-error": ["zod-validation-error@4.0.2", "", { "peerDependencies": { "zod": "^3.25.0 || ^4.0.0" } }, "sha512-Q6/nZLe6jxuU80qb/4uJ4t5v2VEZ44lzQjPDhYJNztRQ4wyWc6VF3D3Kb/fAuPetZQnhS3hnajCf9CsWesghLQ=="],
 | 
			
		||||
 | 
			
		||||
    "zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="],
 | 
			
		||||
 | 
			
		||||
@@ -939,17 +943,17 @@
 | 
			
		||||
 | 
			
		||||
    "@jridgewell/trace-mapping/@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.0", "", {}, "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/jiti": ["jiti@2.5.1", "", { "bin": { "jiti": "lib/jiti-cli.mjs" } }, "sha512-twQoecYPiVA5K/h6SxtORw/Bs3ar+mLUtoPSc7iMXzQzK8d7eJ/R09wmTwAjiamETn1cXYPGfNnu7DMoHgu12w=="],
 | 
			
		||||
    "@tailwindcss/node/jiti": ["jiti@2.6.1", "", { "bin": { "jiti": "lib/jiti-cli.mjs" } }, "sha512-ekilCSN1jwRvIbgeg/57YFh8qQDNbwDb9xT/qu2DAHbFFZUicIl4ygVaAvzveMhMVr3LnpSKTNnwt8PoOfmKhQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.4.5", "", { "dependencies": { "@emnapi/wasi-threads": "1.0.4", "tslib": "^2.4.0" }, "bundled": true }, "sha512-XsLw1dEOpkSX/WucdqUhPWP7hDxSvZiY+fsUC14h+FtQ2Ifni4znbBt8punRX+Uj2JG/uDb8nEHVKvrVlvdZ5Q=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.5.0", "", { "dependencies": { "@emnapi/wasi-threads": "1.1.0", "tslib": "^2.4.0" }, "bundled": true }, "sha512-sbP8GzB1WDzacS8fgNPpHlp6C9VZe+SJP3F90W9rLemaQj2PzIuTEl1qDOYQf58YIpyjViI24y9aPWCjEzY2cg=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.4.5", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-++LApOtY0pEEz1zrd9vy1/zXVaVJJ/EbAF3u0fXIzPJEDtnITsBGbbK0EkM72amhl/R5b+5xx0Y/QhcVOpuulg=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.5.0", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-97/BJ3iXHww3djw6hYIfErCZFee7qCtrneuLa20UXFCOTCfBM2cvQHjWJ2EG0s0MtdNwInarqCTz35i4wWXHsQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.0.4", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-PJR+bOmMOPH8AtcTGAyYNiuJ3/Fcoj2XN/gBEWzDIKh254XO+mM9XoXHk5GNEhodxeMznbg7BlRojVbKN+gC6g=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.1.0", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-WI0DdZ8xFSbgMjR1sFsKABJ/C5OnRrjT06JXbZKexJGrDuPTzZdDYfFlsgcCXCyf+suG5QU2e/y1Wo2V/OapLQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@0.2.12", "", { "dependencies": { "@emnapi/core": "^1.4.3", "@emnapi/runtime": "^1.4.3", "@tybys/wasm-util": "^0.10.0" }, "bundled": true }, "sha512-ZVWUcfwY4E/yPitQJl481FjFo3K22D6qF0DuFH6Y/nbnE11GY5uguDxZMGXPQ8WQ0128MXQD7TnfHyK4oWoIJQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@1.0.5", "", { "dependencies": { "@emnapi/core": "^1.5.0", "@emnapi/runtime": "^1.5.0", "@tybys/wasm-util": "^0.10.1" }, "bundled": true }, "sha512-TBr9Cf9onSAS2LQ2+QHx6XcC6h9+RIzJgbqG3++9TUZSH204AwEy5jg3BTQ0VATsyoGj4ee49tN/y6rvaOOtcg=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@tybys/wasm-util": ["@tybys/wasm-util@0.10.0", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-VyyPYFlOMNylG45GoAe0xDoLwWuowvf92F9kySqzYh8vmYm7D2u4iUJKa1tOUpS70Ku13ASrOkS4ScXFsTaCNQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@tybys/wasm-util": ["@tybys/wasm-util@0.10.1", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-9tTaPJLSiejZKx+Bmog4uSubteqTvFrVrURwkmHixBo0G4seD0zUxp98E1DzUBJxLQ3NPwXrGKDiVjwx/DpPsg=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
 | 
			
		||||
 | 
			
		||||
@@ -967,33 +971,33 @@
 | 
			
		||||
 | 
			
		||||
    "@types/estree-jsx/@types/estree": ["@types/estree@1.0.7", "", {}, "sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.44.0", "", { "dependencies": { "@typescript-eslint/types": "8.44.0", "@typescript-eslint/visitor-keys": "8.44.0" } }, "sha512-87Jv3E+al8wpD+rIdVJm/ItDBe/Im09zXIjFoipOjr5gHUhJmTzfFLuTJ/nPTMc2Srsroy4IBXwcTCHyRR7KzA=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.46.1", "", { "dependencies": { "@typescript-eslint/types": "8.46.1", "@typescript-eslint/visitor-keys": "8.46.1" } }, "sha512-weL9Gg3/5F0pVQKiF8eOXFZp8emqWzZsOJuWRUNtHT+UNV2xSJegmpCNQHy37aEQIbToTq7RHKhWvOsmbM680A=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils": ["@typescript-eslint/utils@8.44.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.44.0", "@typescript-eslint/types": "8.44.0", "@typescript-eslint/typescript-estree": "8.44.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-nktOlVcg3ALo0mYlV+L7sWUD58KG4CMj1rb2HUVOO4aL3K/6wcD+NERqd0rrA5Vg06b42YhF6cFxeixsp9Riqg=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils": ["@typescript-eslint/utils@8.46.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.46.1", "@typescript-eslint/types": "8.46.1", "@typescript-eslint/typescript-estree": "8.46.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-vkYUy6LdZS7q1v/Gxb2Zs7zziuXN0wxqsetJdeZdRe/f5dwJFglmuvZBfTUivCtjH725C1jWCDfpadadD95EDQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/ignore": ["ignore@7.0.4", "", {}, "sha512-gJzzk+PQNznz8ysRrC0aOkBNVRBDtE1n53IqyqEf3PXrYwomFs5q4pGMizBMJF+ykh03insJ27hB8gSrD2Hn8A=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.44.0", "", { "dependencies": { "@typescript-eslint/types": "8.44.0", "@typescript-eslint/visitor-keys": "8.44.0" } }, "sha512-87Jv3E+al8wpD+rIdVJm/ItDBe/Im09zXIjFoipOjr5gHUhJmTzfFLuTJ/nPTMc2Srsroy4IBXwcTCHyRR7KzA=="],
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.46.1", "", { "dependencies": { "@typescript-eslint/types": "8.46.1", "@typescript-eslint/visitor-keys": "8.46.1" } }, "sha512-weL9Gg3/5F0pVQKiF8eOXFZp8emqWzZsOJuWRUNtHT+UNV2xSJegmpCNQHy37aEQIbToTq7RHKhWvOsmbM680A=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/project-service/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "@typescript-eslint/project-service/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/scope-manager/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.43.0", "", { "dependencies": { "@typescript-eslint/types": "8.43.0", "eslint-visitor-keys": "^4.2.1" } }, "sha512-T+S1KqRD4sg/bHfLwrpF/K3gQLBM1n7Rp7OjjikjTEssI2YJzQpi5WXoynOaQ93ERIuq3O8RBTOUYDKszUCEHw=="],
 | 
			
		||||
    "@typescript-eslint/scope-manager/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.45.0", "", { "dependencies": { "@typescript-eslint/types": "8.45.0", "eslint-visitor-keys": "^4.2.1" } }, "sha512-qsaFBA3e09MIDAGFUrTk+dzqtfv1XPVz8t8d1f0ybTzrCY7BKiMC5cjrl1O/P7UmHsNyW90EYSkU/ZWpmXelag=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils": ["@typescript-eslint/utils@8.44.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.44.0", "@typescript-eslint/types": "8.44.0", "@typescript-eslint/typescript-estree": "8.44.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-nktOlVcg3ALo0mYlV+L7sWUD58KG4CMj1rb2HUVOO4aL3K/6wcD+NERqd0rrA5Vg06b42YhF6cFxeixsp9Riqg=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils": ["@typescript-eslint/utils@8.46.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.46.1", "@typescript-eslint/types": "8.46.1", "@typescript-eslint/typescript-estree": "8.46.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-vkYUy6LdZS7q1v/Gxb2Zs7zziuXN0wxqsetJdeZdRe/f5dwJFglmuvZBfTUivCtjH725C1jWCDfpadadD95EDQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "@typescript-eslint/typescript-estree/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree/minimatch": ["minimatch@9.0.5", "", { "dependencies": { "brace-expansion": "^2.0.1" } }, "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree/semver": ["semver@7.7.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.43.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.43.0", "@typescript-eslint/tsconfig-utils": "8.43.0", "@typescript-eslint/types": "8.43.0", "@typescript-eslint/visitor-keys": "8.43.0", "debug": "^4.3.4", "fast-glob": "^3.3.2", "is-glob": "^4.0.3", "minimatch": "^9.0.4", "semver": "^7.6.0", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-7Vv6zlAhPb+cvEpP06WXXy/ZByph9iL6BQRBDj4kmBsW98AqEeQHlj/13X+sZOrKSo9/rNKH4Ul4f6EICREFdw=="],
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.45.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.45.0", "@typescript-eslint/tsconfig-utils": "8.45.0", "@typescript-eslint/types": "8.45.0", "@typescript-eslint/visitor-keys": "8.45.0", "debug": "^4.3.4", "fast-glob": "^3.3.2", "is-glob": "^4.0.3", "minimatch": "^9.0.4", "semver": "^7.6.0", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-GfE1NfVbLam6XQ0LcERKwdTTPlLvHvXXhOeUGC1OXi4eQBoyy1iVsW+uzJ/J9jtCz6/7GCQ9MtrQ0fml/jWCnA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/visitor-keys/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "@typescript-eslint/visitor-keys/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="],
 | 
			
		||||
 | 
			
		||||
@@ -1009,7 +1013,7 @@
 | 
			
		||||
 | 
			
		||||
    "parse-entities/@types/unist": ["@types/unist@2.0.11", "", {}, "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils": ["@typescript-eslint/utils@8.44.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.44.0", "@typescript-eslint/types": "8.44.0", "@typescript-eslint/typescript-estree": "8.44.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-nktOlVcg3ALo0mYlV+L7sWUD58KG4CMj1rb2HUVOO4aL3K/6wcD+NERqd0rrA5Vg06b42YhF6cFxeixsp9Riqg=="],
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils": ["@typescript-eslint/utils@8.46.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.46.1", "@typescript-eslint/types": "8.46.1", "@typescript-eslint/typescript-estree": "8.46.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-vkYUy6LdZS7q1v/Gxb2Zs7zziuXN0wxqsetJdeZdRe/f5dwJFglmuvZBfTUivCtjH725C1jWCDfpadadD95EDQ=="],
 | 
			
		||||
 | 
			
		||||
    "@babel/helper-module-imports/@babel/traverse/@babel/generator": ["@babel/generator@7.27.1", "", { "dependencies": { "@babel/parser": "^7.27.1", "@babel/types": "^7.27.1", "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.25", "jsesc": "^3.0.2" } }, "sha512-UnJfnIpc/+JO0/+KRVQNGU+y5taA5vCbwN8+azkX6beii/ZF+enZJSOKo11ZSzGJjlNfJHfQtmQT8H+9TXPG2w=="],
 | 
			
		||||
 | 
			
		||||
@@ -1025,27 +1029,27 @@
 | 
			
		||||
 | 
			
		||||
    "@eslint/eslintrc/espree/eslint-visitor-keys": ["eslint-visitor-keys@4.2.0", "", {}, "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.44.0", "", { "dependencies": { "@typescript-eslint/types": "8.44.0", "@typescript-eslint/visitor-keys": "8.44.0" } }, "sha512-87Jv3E+al8wpD+rIdVJm/ItDBe/Im09zXIjFoipOjr5gHUhJmTzfFLuTJ/nPTMc2Srsroy4IBXwcTCHyRR7KzA=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.46.1", "", { "dependencies": { "@typescript-eslint/types": "8.46.1", "@typescript-eslint/visitor-keys": "8.46.1" } }, "sha512-weL9Gg3/5F0pVQKiF8eOXFZp8emqWzZsOJuWRUNtHT+UNV2xSJegmpCNQHy37aEQIbToTq7RHKhWvOsmbM680A=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree/minimatch/brace-expansion": ["brace-expansion@2.0.1", "", { "dependencies": { "balanced-match": "^1.0.0" } }, "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.43.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.43.0", "@typescript-eslint/types": "^8.43.0", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-htB/+D/BIGoNTQYffZw4uM4NzzuolCoaA/BusuSIcC8YjmBYQioew5VUZAYdAETPjeed0hqCaW7EHg+Robq8uw=="],
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.45.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.45.0", "@typescript-eslint/types": "^8.45.0", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-3pcVHwMG/iA8afdGLMuTibGR7pDsn9RjDev6CCB+naRsSYs2pns5QbinF4Xqw6YC/Sj3lMrm/Im0eMfaa61WUg=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.43.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-ALC2prjZcj2YqqL5X/bwWQmHA2em6/94GcbB/KKu5SX3EBDOsqztmmX1kMkvAJHzxk7TazKzJfFiEIagNV3qEA=="],
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.45.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-aFdr+c37sc+jqNMGhH+ajxPXwjv9UtFZk79k8pLoJ6p4y0snmYpPA52GuWHgt2ZF4gRRW6odsEj41uZLojDt5w=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.43.0", "", { "dependencies": { "@typescript-eslint/types": "8.43.0", "eslint-visitor-keys": "^4.2.1" } }, "sha512-T+S1KqRD4sg/bHfLwrpF/K3gQLBM1n7Rp7OjjikjTEssI2YJzQpi5WXoynOaQ93ERIuq3O8RBTOUYDKszUCEHw=="],
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.45.0", "", { "dependencies": { "@typescript-eslint/types": "8.45.0", "eslint-visitor-keys": "^4.2.1" } }, "sha512-qsaFBA3e09MIDAGFUrTk+dzqtfv1XPVz8t8d1f0ybTzrCY7BKiMC5cjrl1O/P7UmHsNyW90EYSkU/ZWpmXelag=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/minimatch": ["minimatch@9.0.5", "", { "dependencies": { "brace-expansion": "^2.0.1" } }, "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils/@typescript-eslint/typescript-estree/semver": ["semver@7.7.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.44.0", "", { "dependencies": { "@typescript-eslint/types": "8.44.0", "@typescript-eslint/visitor-keys": "8.44.0" } }, "sha512-87Jv3E+al8wpD+rIdVJm/ItDBe/Im09zXIjFoipOjr5gHUhJmTzfFLuTJ/nPTMc2Srsroy4IBXwcTCHyRR7KzA=="],
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.46.1", "", { "dependencies": { "@typescript-eslint/types": "8.46.1", "@typescript-eslint/visitor-keys": "8.46.1" } }, "sha512-weL9Gg3/5F0pVQKiF8eOXFZp8emqWzZsOJuWRUNtHT+UNV2xSJegmpCNQHy37aEQIbToTq7RHKhWvOsmbM680A=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.44.0", "", {}, "sha512-ZSl2efn44VsYM0MfDQe68RKzBz75NPgLQXuGypmym6QVOWL5kegTZuZ02xRAT9T+onqvM6T8CdQk0OwYMB6ZvA=="],
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="],
 | 
			
		||||
 | 
			
		||||
    "@babel/helper-module-imports/@babel/traverse/@babel/generator/@jridgewell/gen-mapping": ["@jridgewell/gen-mapping@0.3.8", "", { "dependencies": { "@jridgewell/set-array": "^1.2.1", "@jridgewell/sourcemap-codec": "^1.4.10", "@jridgewell/trace-mapping": "^0.3.24" } }, "sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA=="],
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -8,6 +8,7 @@
 | 
			
		||||
    <link rel="shortcut icon" href="/favicon.ico" />
 | 
			
		||||
    <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" />
 | 
			
		||||
    <meta name="apple-mobile-web-app-title" content="Tinyauth" />
 | 
			
		||||
    <meta name="robots" content="nofollow, noindex" />
 | 
			
		||||
    <link rel="manifest" href="/site.webmanifest" />
 | 
			
		||||
    <title>Tinyauth</title>
 | 
			
		||||
  </head>
 | 
			
		||||
 
 | 
			
		||||
@@ -15,43 +15,43 @@
 | 
			
		||||
    "@radix-ui/react-select": "^2.2.6",
 | 
			
		||||
    "@radix-ui/react-separator": "^1.1.7",
 | 
			
		||||
    "@radix-ui/react-slot": "^1.2.3",
 | 
			
		||||
    "@tailwindcss/vite": "^4.1.13",
 | 
			
		||||
    "@tanstack/react-query": "^5.89.0",
 | 
			
		||||
    "@tailwindcss/vite": "^4.1.14",
 | 
			
		||||
    "@tanstack/react-query": "^5.90.3",
 | 
			
		||||
    "axios": "^1.12.2",
 | 
			
		||||
    "class-variance-authority": "^0.7.1",
 | 
			
		||||
    "clsx": "^2.1.1",
 | 
			
		||||
    "i18next": "^25.5.2",
 | 
			
		||||
    "i18next": "^25.6.0",
 | 
			
		||||
    "i18next-browser-languagedetector": "^8.2.0",
 | 
			
		||||
    "i18next-resources-to-backend": "^1.2.1",
 | 
			
		||||
    "input-otp": "^1.4.2",
 | 
			
		||||
    "lucide-react": "^0.544.0",
 | 
			
		||||
    "lucide-react": "^0.545.0",
 | 
			
		||||
    "next-themes": "^0.4.6",
 | 
			
		||||
    "react": "^19.1.1",
 | 
			
		||||
    "react-dom": "^19.1.1",
 | 
			
		||||
    "react-hook-form": "^7.62.0",
 | 
			
		||||
    "react-i18next": "^15.7.3",
 | 
			
		||||
    "react": "^19.2.0",
 | 
			
		||||
    "react-dom": "^19.2.0",
 | 
			
		||||
    "react-hook-form": "^7.65.0",
 | 
			
		||||
    "react-i18next": "^16.0.1",
 | 
			
		||||
    "react-markdown": "^10.1.0",
 | 
			
		||||
    "react-router": "^7.9.1",
 | 
			
		||||
    "react-router": "^7.9.4",
 | 
			
		||||
    "sonner": "^2.0.7",
 | 
			
		||||
    "tailwind-merge": "^3.3.1",
 | 
			
		||||
    "tailwindcss": "^4.1.13",
 | 
			
		||||
    "zod": "^4.1.9"
 | 
			
		||||
    "tailwindcss": "^4.1.14",
 | 
			
		||||
    "zod": "^4.1.12"
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "@eslint/js": "^9.35.0",
 | 
			
		||||
    "@tanstack/eslint-plugin-query": "^5.89.0",
 | 
			
		||||
    "@types/node": "^24.5.2",
 | 
			
		||||
    "@types/react": "^19.1.13",
 | 
			
		||||
    "@types/react-dom": "^19.1.9",
 | 
			
		||||
    "@vitejs/plugin-react": "^5.0.3",
 | 
			
		||||
    "eslint": "^9.35.0",
 | 
			
		||||
    "eslint-plugin-react-hooks": "^5.2.0",
 | 
			
		||||
    "eslint-plugin-react-refresh": "^0.4.19",
 | 
			
		||||
    "@eslint/js": "^9.37.0",
 | 
			
		||||
    "@tanstack/eslint-plugin-query": "^5.91.0",
 | 
			
		||||
    "@types/node": "^24.7.2",
 | 
			
		||||
    "@types/react": "^19.2.2",
 | 
			
		||||
    "@types/react-dom": "^19.2.2",
 | 
			
		||||
    "@vitejs/plugin-react": "^5.0.4",
 | 
			
		||||
    "eslint": "^9.37.0",
 | 
			
		||||
    "eslint-plugin-react-hooks": "^7.0.0",
 | 
			
		||||
    "eslint-plugin-react-refresh": "^0.4.23",
 | 
			
		||||
    "globals": "^16.4.0",
 | 
			
		||||
    "prettier": "3.6.2",
 | 
			
		||||
    "tw-animate-css": "^1.3.8",
 | 
			
		||||
    "typescript": "~5.9.2",
 | 
			
		||||
    "typescript-eslint": "^8.44.0",
 | 
			
		||||
    "vite": "^7.1.6"
 | 
			
		||||
    "tw-animate-css": "^1.4.0",
 | 
			
		||||
    "typescript": "~5.9.3",
 | 
			
		||||
    "typescript-eslint": "^8.46.1",
 | 
			
		||||
    "vite": "^7.1.10"
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -44,6 +44,7 @@ export const TotpForm = (props: Props) => {
 | 
			
		||||
                  disabled={loading}
 | 
			
		||||
                  {...field}
 | 
			
		||||
                  autoComplete="one-time-code"
 | 
			
		||||
                  autoFocus
 | 
			
		||||
                >
 | 
			
		||||
                  <InputOTPGroup>
 | 
			
		||||
                    <InputOTPSlot index={0} />
 | 
			
		||||
 
 | 
			
		||||
@@ -1,11 +1,15 @@
 | 
			
		||||
import { useAppContext } from "@/context/app-context";
 | 
			
		||||
import { LanguageSelector } from "../language/language";
 | 
			
		||||
import { Outlet } from "react-router";
 | 
			
		||||
import { useCallback, useState } from "react";
 | 
			
		||||
import { useCallback, useEffect, useState } from "react";
 | 
			
		||||
import { DomainWarning } from "../domain-warning/domain-warning";
 | 
			
		||||
 | 
			
		||||
const BaseLayout = ({ children }: { children: React.ReactNode }) => {
 | 
			
		||||
  const { backgroundImage } = useAppContext();
 | 
			
		||||
  const { backgroundImage, title } = useAppContext();
 | 
			
		||||
 | 
			
		||||
  useEffect(() => {
 | 
			
		||||
    document.title = title;
 | 
			
		||||
  }, [title]);
 | 
			
		||||
 | 
			
		||||
  return (
 | 
			
		||||
    <div
 | 
			
		||||
 
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "أخفق الحصول على رابط OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "إعادة توجيه",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "إعادة توجيه إلى مزود OAuth الخاص بك",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "متابعة",
 | 
			
		||||
    "continueRedirectingTitle": "إعادة توجيه...",
 | 
			
		||||
    "continueRedirectingSubtitle": "يجب إعادة توجيهك إلى التطبيق قريبا",
 | 
			
		||||
    "continueInvalidRedirectTitle": "إعادة توجيه غير صالحة",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "رابط إعادة التوجيه غير صالح",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "إعادة توجيه غير آمنة",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "أنت تحاول إعادة التوجيه من <code>https</code> إلى <code>http</code>، هل أنت متأكد أنك تريد المتابعة؟",
 | 
			
		||||
    "continueTitle": "متابعة",
 | 
			
		||||
    "continueSubtitle": "انقر الزر للمتابعة إلى التطبيق الخاص بك.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "فشل تسجيل الخروج",
 | 
			
		||||
    "logoutFailSubtitle": "يرجى إعادة المحاولة",
 | 
			
		||||
    "logoutSuccessTitle": "تم تسجيل الخروج",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "حاول مجددا",
 | 
			
		||||
    "untrustedRedirectTitle": "إعادة توجيه غير موثوقة",
 | 
			
		||||
    "untrustedRedirectSubtitle": "أنت تحاول إعادة التوجيه إلى نطاق لا يتطابق مع النطاق المكون الخاص بك (<code>{{domain}}</code>). هل أنت متأكد من أنك تريد المتابعة؟",
 | 
			
		||||
    "cancelTitle": "إلغاء",
 | 
			
		||||
    "forgotPasswordTitle": "نسيت كلمة المرور؟",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "تجاهل",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -1,57 +1,62 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "Welcome back, login with",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "Or",
 | 
			
		||||
    "loginUsername": "Username",
 | 
			
		||||
    "loginPassword": "Password",
 | 
			
		||||
    "loginSubmit": "Login",
 | 
			
		||||
    "loginFailTitle": "Failed to log in",
 | 
			
		||||
    "loginFailSubtitle": "Please check your username and password",
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "Logged in",
 | 
			
		||||
    "loginSuccessSubtitle": "Welcome back!",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
    "logoutSuccessSubtitle": "You have been logged out",
 | 
			
		||||
    "logoutTitle": "Logout",
 | 
			
		||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
			
		||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
			
		||||
    "notFoundTitle": "Page not found",
 | 
			
		||||
    "notFoundSubtitle": "The page you are looking for does not exist.",
 | 
			
		||||
    "notFoundButton": "Go home",
 | 
			
		||||
    "totpFailTitle": "Failed to verify code",
 | 
			
		||||
    "totpFailSubtitle": "Please check your code and try again",
 | 
			
		||||
    "totpSuccessTitle": "Verified",
 | 
			
		||||
    "totpSuccessSubtitle": "Redirecting to your app",
 | 
			
		||||
    "totpTitle": "Enter your TOTP code",
 | 
			
		||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
			
		||||
    "unauthorizedTitle": "Unauthorized",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
    "errorTitle": "An error occurred",
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "loginTitle": "Vítejte zpět, přihlaste se pomocí",
 | 
			
		||||
    "loginTitleSimple": "Vítejte zpět, přihlaste se prosím",
 | 
			
		||||
    "loginDivider": "Nebo",
 | 
			
		||||
    "loginUsername": "Uživatelské jméno",
 | 
			
		||||
    "loginPassword": "Heslo",
 | 
			
		||||
    "loginSubmit": "Přihlásit",
 | 
			
		||||
    "loginFailTitle": "Přihlášení se nezdařilo",
 | 
			
		||||
    "loginFailSubtitle": "Zkontrolujte prosím své uživatelské jméno a heslo",
 | 
			
		||||
    "loginFailRateLimit": "Přiliš mnoho neúspěšných pokusů přihlášení. Zkuste to prosím později",
 | 
			
		||||
    "loginSuccessTitle": "Přihlášen",
 | 
			
		||||
    "loginSuccessSubtitle": "Vítejte zpět!",
 | 
			
		||||
    "loginOauthFailTitle": "Došlo k chybě",
 | 
			
		||||
    "loginOauthFailSubtitle": "Nepodařilo se získat OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Přesměrování",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Přesměrování k poskytovateli OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Pokračovat",
 | 
			
		||||
    "continueRedirectingTitle": "Přesměrování...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Brzy budete přesměrováni do aplikace",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Nezabezpečené přesměrování",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Pokoušíte se přesměrovat z <code>https</code> na <code>http</code>, které není bezpečné. Opravdu chcete pokračovat?",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Odhlášení se nezdařilo",
 | 
			
		||||
    "logoutFailSubtitle": "Zkuste to prosím znovu",
 | 
			
		||||
    "logoutSuccessTitle": "Odhlášen",
 | 
			
		||||
    "logoutSuccessSubtitle": "Byl jste odhlášen",
 | 
			
		||||
    "logoutTitle": "Odhlásit",
 | 
			
		||||
    "logoutUsernameSubtitle": "Jste přihlášen jako <code>{{username}}</code>. Pro odhlášení klikněte na tlačítko níže.",
 | 
			
		||||
    "logoutOauthSubtitle": "Jste přihlášen jako <code>{{username}}</code> pomocí {{provider}} poskytovatele OAuth. Pro odhlášení klikněte na tlačítko níže.",
 | 
			
		||||
    "notFoundTitle": "Stránka nenalezena",
 | 
			
		||||
    "notFoundSubtitle": "Stránka, kterou hledáte, neexistuje.",
 | 
			
		||||
    "notFoundButton": "Jít domů",
 | 
			
		||||
    "totpFailTitle": "Nepodařilo se ověřit kód",
 | 
			
		||||
    "totpFailSubtitle": "Zkontrolujte prosím kód a zkuste to znovu",
 | 
			
		||||
    "totpSuccessTitle": "Ověřeno",
 | 
			
		||||
    "totpSuccessSubtitle": "Přesměrování do aplikace",
 | 
			
		||||
    "totpTitle": "Zadejte TOTP kód",
 | 
			
		||||
    "totpSubtitle": "Zadejte prosím kód z ověřovací aplikace.",
 | 
			
		||||
    "unauthorizedTitle": "Nepovoleno",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Uživatel s uživatelským jménem <code>{{username}}</code> není oprávněn k přístupu ke zdroji <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Uživatel s uživatelským jménem <code>{{username}}</code> není oprávněn k přihlášení.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Uživatel s uživatelským jménem <code>{{username}}</code> není ve skupině potřebné k přístupu ke zdroji <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Vaše IP adresa <code>{{ip}}</code> není oprávněna k přístupu ke zdroji <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Zkusit znovu",
 | 
			
		||||
    "cancelTitle": "Zrušit",
 | 
			
		||||
    "forgotPasswordTitle": "Zapomněli jste heslo?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Nepodařilo se načíst poskytovatele ověřování. Zkontrolujte prosím konfiguraci.",
 | 
			
		||||
    "errorTitle": "Došlo k chybě",
 | 
			
		||||
    "errorSubtitle": "Nastala chyba při pokusu o provedení této akce. Pro více informací prosím zkontrolujte konzolu.",
 | 
			
		||||
    "forgotPasswordMessage": "Heslo můžete obnovit změnou proměnné `USERS`.",
 | 
			
		||||
    "fieldRequired": "Toto pole je povinné",
 | 
			
		||||
    "invalidInput": "Neplatný údaj",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Kunne ikke hente OAuth-URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Omdirigerer",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Omdirigerer til din OAuth-udbyder",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Fortsæt",
 | 
			
		||||
    "continueRedirectingTitle": "Omdirigerer...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Du bør blive omdirigeret til appen snart",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Ugyldig omdirigering",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "Omdirigerings-URL'en er ugyldig",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Usikker omdirigering",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Du forsøger at omdirigere fra <code>https</code> til <code>http</code>, som ikke er sikker. Er du sikker på, at du vil fortsætte?",
 | 
			
		||||
    "continueTitle": "Fortsæt",
 | 
			
		||||
    "continueSubtitle": "Klik på knappen for at fortsætte til din app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Log ud mislykkedes",
 | 
			
		||||
    "logoutFailSubtitle": "Prøv venligst igen",
 | 
			
		||||
    "logoutSuccessTitle": "Logget ud",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Brugeren med brugernavnet <code>{{username}}</code> er ikke i de grupper, som ressourcen <code>{{resource}}</code> kræver.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Din IP adresse <code>{{ip}}</code> er ikke autoriseret til at tilgå ressourcen <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Prøv igen",
 | 
			
		||||
    "untrustedRedirectTitle": "Usikker omdirigering",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Du forsøger at omdirigere til et domæne, der ikke matcher dit konfigurerede domæne (<code>{{domain}}</code>). Er du sikker på, at du vil fortsætte?",
 | 
			
		||||
    "cancelTitle": "Annuller",
 | 
			
		||||
    "forgotPasswordTitle": "Glemt din adgangskode?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Kunne ikke indlæse godkendelsesudbydere. Tjek venligst din konfiguration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "Der opstod en fejl under forsøget på at udføre denne handling. Tjek venligst konsollen for mere information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Fehler beim Abrufen der OAuth-URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Leite weiter",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Weiterleitung zu Ihrem OAuth-Provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Weiter",
 | 
			
		||||
    "continueRedirectingTitle": "Leite weiter...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Sie sollten in Kürze zur App weitergeleitet werden",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Ungültige Weiterleitung",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "Die Weiterleitungs-URL ist ungültig",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Unsichere Weiterleitung",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Sie versuchen von <code>https</code> auf <code>http</code> weiterzuleiten, was unsicher ist. Sind Sie sicher, dass Sie fortfahren möchten?",
 | 
			
		||||
    "continueTitle": "Weiter",
 | 
			
		||||
    "continueSubtitle": "Klicken Sie auf den Button, um zur App zu gelangen.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Abmelden fehlgeschlagen",
 | 
			
		||||
    "logoutFailSubtitle": "Bitte versuchen Sie es erneut",
 | 
			
		||||
    "logoutSuccessTitle": "Abgemeldet",
 | 
			
		||||
@@ -31,7 +34,7 @@
 | 
			
		||||
    "logoutOauthSubtitle": "Sie sind derzeit als <code>{{username}}</code> über den OAuth-Anbieter {{provider}} angemeldet. Klicken Sie auf den Button unten, um sich abzumelden.",
 | 
			
		||||
    "notFoundTitle": "Seite nicht gefunden",
 | 
			
		||||
    "notFoundSubtitle": "Die gesuchte Seite existiert nicht.",
 | 
			
		||||
    "notFoundButton": "Nach Hause",
 | 
			
		||||
    "notFoundButton": "Zurück",
 | 
			
		||||
    "totpFailTitle": "Fehler beim Verifizieren des Codes",
 | 
			
		||||
    "totpFailSubtitle": "Bitte überprüfen Sie Ihren Code und versuchen Sie es erneut",
 | 
			
		||||
    "totpSuccessTitle": "Verifiziert",
 | 
			
		||||
@@ -44,14 +47,16 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Der Benutzer mit Benutzername <code>{{username}}</code> ist nicht in den Gruppen, die von der Ressource <code>{{resource}}</code> benötigt werden.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Ihre IP-Adresse <code>{{ip}}</code> ist nicht berechtigt, auf die Ressource <code>{{resource}}</code> zuzugreifen.",
 | 
			
		||||
    "unauthorizedButton": "Erneut versuchen",
 | 
			
		||||
    "untrustedRedirectTitle": "Nicht vertrauenswürdige Weiterleitung",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Sie versuchen auf eine Domain umzuleiten, die nicht mit Ihrer konfigurierten Domain übereinstimmt (<code>{{domain}}</code>). Sind Sie sicher, dass Sie fortfahren möchten?",
 | 
			
		||||
    "cancelTitle": "Abbrechen",
 | 
			
		||||
    "forgotPasswordTitle": "Passwort vergessen?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Fehler beim Laden der Authentifizierungsanbieter. Bitte überprüfen Sie Ihre Konfiguration.",
 | 
			
		||||
    "errorTitle": "Ein Fehler ist aufgetreten",
 | 
			
		||||
    "errorSubtitle": "Beim Versuch, diese Aktion auszuführen, ist ein Fehler aufgetreten. Bitte überprüfen Sie die Konsole für weitere Informationen.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "forgotPasswordMessage": "Das Passwort kann durch Änderung der 'USERS' Variable zurückgesetzt werden.",
 | 
			
		||||
    "fieldRequired": "Dieses Feld ist notwendig",
 | 
			
		||||
    "invalidInput": "Ungültige Eingabe",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Αποτυχία λήψης OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Ανακατεύθυνση",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Ανακατεύθυνση στον πάροχο OAuth σας",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "Αυτόματη Ανακατεύθυνση OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "Θα ανακατευθυνθείτε αυτόματα στον πάροχο OAuth σας για να επαληθευτείτε.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Ανακατεύθυνση τώρα",
 | 
			
		||||
    "continueTitle": "Συνέχεια",
 | 
			
		||||
    "continueRedirectingTitle": "Ανακατεύθυνση...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Θα πρέπει να μεταφερθείτε σύντομα στην εφαρμογή σας",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Μη έγκυρη ανακατεύθυνση",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "Το URL ανακατεύθυνσης δεν είναι έγκυρο",
 | 
			
		||||
    "continueRedirectManually": "Χειροκίνητη ανακατεύθυνση",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Μη ασφαλής ανακατεύθυνση",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Προσπαθείτε να ανακατευθύνετε από <code>https</code> σε <code>http</code> το οποίο δεν είναι ασφαλές. Είστε σίγουροι ότι θέλετε να συνεχίσετε;",
 | 
			
		||||
    "continueTitle": "Συνέχεια",
 | 
			
		||||
    "continueSubtitle": "Κάντε κλικ στο κουμπί για να συνεχίσετε στην εφαρμογή σας.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Μη έμπιστη ανακατεύθυνση",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "Προσπαθείτε να ανακατευθύνετε σε ένα domain που δεν ταιριάζει με το ρυθμισμένο domain σας (<code>{{cookieDomain}}</code>). Είστε βέβαιοι ότι θέλετε να συνεχίσετε;",
 | 
			
		||||
    "logoutFailTitle": "Αποτυχία αποσύνδεσης",
 | 
			
		||||
    "logoutFailSubtitle": "Παρακαλώ δοκιμάστε ξανά",
 | 
			
		||||
    "logoutSuccessTitle": "Αποσυνδεδεμένος",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Ο χρήστης με όνομα χρήστη <code>{{username}}</code> δεν είναι στις ομάδες που απαιτούνται από τον πόρο <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Η διεύθυνση IP σας <code>{{ip}}</code> δεν είναι εξουσιοδοτημένη να έχει πρόσβαση στον πόρο <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Προσπαθήστε ξανά",
 | 
			
		||||
    "untrustedRedirectTitle": "Μη έμπιστη ανακατεύθυνση",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Προσπαθείτε να ανακατευθύνετε σε ένα domain που δεν ταιριάζει με τον ρυθμισμένο domain σας (<code>{{domain}}</code>). Είστε βέβαιοι ότι θέλετε να συνεχίσετε;",
 | 
			
		||||
    "cancelTitle": "Ακύρωση",
 | 
			
		||||
    "forgotPasswordTitle": "Ξεχάσατε το συνθηματικό σας;",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Αποτυχία φόρτωσης παρόχων πιστοποίησης. Παρακαλώ ελέγξτε τις ρυθμίσεις σας.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "Παρουσιάστηκε σφάλμα κατά την προσπάθεια εκτέλεσης αυτής της ενέργειας. Ελέγξτε την κονσόλα για περισσότερες πληροφορίες.",
 | 
			
		||||
    "forgotPasswordMessage": "Μπορείτε να επαναφέρετε τον κωδικό πρόσβασής σας αλλάζοντας τη μεταβλητή περιβάλλοντος `USERS`.",
 | 
			
		||||
    "fieldRequired": "Αυτό το πεδίο είναι υποχρεωτικό",
 | 
			
		||||
    "invalidInput": "Μη έγκυρη καταχώρηση"
 | 
			
		||||
    "invalidInput": "Μη έγκυρη καταχώρηση",
 | 
			
		||||
    "domainWarningTitle": "Μη έγκυρο domain",
 | 
			
		||||
    "domainWarningSubtitle": "Αυτή η εφαρμογή έχει ρυθμιστεί για πρόσβαση από <code>{{appUrl}}</code>, αλλά <code>{{currentUrl}}</code> χρησιμοποιείται. Αν συνεχίσετε, μπορεί να αντιμετωπίσετε προβλήματα με την ταυτοποίηση.",
 | 
			
		||||
    "ignoreTitle": "Παράβλεψη",
 | 
			
		||||
    "goToCorrectDomainTitle": "Μεταβείτε στο σωστό domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Error al obtener la URL de OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redireccionando",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redireccionando a tu proveedor de OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continuar",
 | 
			
		||||
    "continueRedirectingTitle": "Redireccionando...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Pronto será redirigido a la aplicación",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Redirección inválida",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "La URL de redirección es inválida",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Redirección insegura",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Está intentando redirigir desde <code>https</code> a <code>http</code> lo cual no es seguro. ¿Está seguro que desea continuar?",
 | 
			
		||||
    "continueTitle": "Continuar",
 | 
			
		||||
    "continueSubtitle": "Haga clic en el botón para continuar hacia su aplicación.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Fallo al cerrar sesión",
 | 
			
		||||
    "logoutFailSubtitle": "Por favor intente nuevamente",
 | 
			
		||||
    "logoutSuccessTitle": "Sesión cerrada",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "El usuario con nombre de usuario <code>{{username}}</code> no está en los grupos requeridos por el recurso <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Inténtelo de nuevo",
 | 
			
		||||
    "untrustedRedirectTitle": "Redirección no confiable",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Está intentando redirigir a un dominio que no coincide con su dominio configurado (<code>{{domain}}</code>). ¿Está seguro que desea continuar?",
 | 
			
		||||
    "cancelTitle": "Cancelar",
 | 
			
		||||
    "forgotPasswordTitle": "¿Olvidó su contraseña?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Error al cargar los proveedores de autenticación. Por favor revise su configuración.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "Ocurrió un error mientras se trataba de realizar esta acción. Por favor, revise la consola para más información.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Impossible d'obtenir l'URL OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirection",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirection vers votre fournisseur OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "Redirection automatique OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "Vous allez être automatiquement redirigé vers votre fournisseur OAuth pour vous authentifier.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Rediriger",
 | 
			
		||||
    "continueTitle": "Continuer",
 | 
			
		||||
    "continueRedirectingTitle": "Redirection...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Vous devriez être redirigé vers l'application bientôt",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Redirection invalide",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "L'URL de redirection est invalide",
 | 
			
		||||
    "continueRedirectManually": "Redirection manuelle",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Redirection non sécurisée",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Vous tentez de rediriger de <code>https</code> vers <code>http</code>, ce qui n'est pas sécurisé. Êtes-vous sûr de vouloir continuer ?",
 | 
			
		||||
    "continueTitle": "Continuer",
 | 
			
		||||
    "continueSubtitle": "Cliquez sur le bouton pour continuer vers votre application.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Redirection non sécurisée",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "Vous essayez de rediriger vers un domaine qui ne correspond pas à votre domaine configuré (<code>{{cookieDomain}}</code>). Êtes-vous sûr de vouloir continuer ?",
 | 
			
		||||
    "logoutFailTitle": "Échec de la déconnexion",
 | 
			
		||||
    "logoutFailSubtitle": "Veuillez réessayer",
 | 
			
		||||
    "logoutSuccessTitle": "Déconnecté",
 | 
			
		||||
@@ -38,20 +41,22 @@
 | 
			
		||||
    "totpSuccessSubtitle": "Redirection vers votre application",
 | 
			
		||||
    "totpTitle": "Saisissez votre code TOTP",
 | 
			
		||||
    "totpSubtitle": "Veuillez saisir le code de votre application d'authentification.",
 | 
			
		||||
    "unauthorizedTitle": "Unauthorized",
 | 
			
		||||
    "unauthorizedTitle": "Non autorisé",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "L'utilisateur avec le nom d'utilisateur <code>{{username}}</code> n'est pas autorisé à accéder à la ressource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "L'utilisateur avec le nom d'utilisateur <code>{{username}}</code> n'est pas autorisé à se connecter.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "L'utilisateur avec le nom d'utilisateur <code>{{username}}</code> n'appartient pas aux groupes requis par la ressource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Votre adresse IP <code>{{ip}}</code> n'est pas autorisée à accéder à la ressource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Réessayer",
 | 
			
		||||
    "untrustedRedirectTitle": "Redirection non fiable",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Vous tentez de rediriger vers un domaine qui ne correspond pas à votre domaine configuré (<code>{{domain}}</code>). Êtes-vous sûr de vouloir continuer ?",
 | 
			
		||||
    "cancelTitle": "Annuler",
 | 
			
		||||
    "forgotPasswordTitle": "Mot de passe oublié ?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Échec du chargement des fournisseurs d'authentification. Veuillez vérifier votre configuration.",
 | 
			
		||||
    "errorTitle": "Une erreur est survenue",
 | 
			
		||||
    "errorSubtitle": "Une erreur est survenue lors de l'exécution de cette action. Veuillez consulter la console pour plus d'informations.",
 | 
			
		||||
    "forgotPasswordMessage": "Vous pouvez réinitialiser votre mot de passe en modifiant la variable d'environnement `USERS`.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "fieldRequired": "Ce champ est obligatoire",
 | 
			
		||||
    "invalidInput": "Saisie non valide",
 | 
			
		||||
    "domainWarningTitle": "Domaine invalide",
 | 
			
		||||
    "domainWarningSubtitle": "Cette instance est configurée pour être accédée depuis <code>{{appUrl}}</code>, mais <code>{{currentUrl}}</code> est utilisé. Si vous continuez, vous pourriez rencontrer des problèmes d'authentification.",
 | 
			
		||||
    "ignoreTitle": "Ignorer",
 | 
			
		||||
    "goToCorrectDomainTitle": "Aller au bon domaine"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Fout bij het ophalen van OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Omleiden",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Omleiden naar je OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Ga verder",
 | 
			
		||||
    "continueRedirectingTitle": "Omleiden...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Je wordt naar de app doorgestuurd",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Ongeldige omleiding",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "De omleidings-URL is ongeldig",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Onveilige doorverwijzing",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Ga verder",
 | 
			
		||||
    "continueSubtitle": "Klik op de knop om door te gaan naar de app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Afmelden mislukt",
 | 
			
		||||
    "logoutFailSubtitle": "Probeer het opnieuw",
 | 
			
		||||
    "logoutSuccessTitle": "Afgemeld",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Opnieuw proberen",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Nie udało się uzyskać adresu URL OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "Przekierowywanie",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Przekierowywanie do Twojego dostawcy OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "Automatyczne przekierowanie OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "Nastąpi automatyczne przekierowanie do dostawcy OAuth w celu uwierzytelnienia.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Przekieruj teraz",
 | 
			
		||||
    "continueTitle": "Kontynuuj",
 | 
			
		||||
    "continueRedirectingTitle": "Przekierowywanie...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Wkrótce powinieneś zostać przekierowany do aplikacji",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Nieprawidłowe przekierowanie",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "Adres przekierowania jest nieprawidłowy",
 | 
			
		||||
    "continueRedirectManually": "Przekieruj mnie ręcznie",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Niezabezpieczone przekierowanie",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Próbujesz przekierować z <code>https</code> do <code>http</code>, co nie jest bezpieczne. Czy na pewno chcesz kontynuować?",
 | 
			
		||||
    "continueTitle": "Kontynuuj",
 | 
			
		||||
    "continueSubtitle": "Kliknij przycisk, aby przejść do aplikacji.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Niezaufane przekierowanie",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "Próbujesz przekierować do domeny, która nie pasuje do skonfigurowanej domeny (<code>{{cookieDomain}}</code>). Czy na pewno chcesz kontynuować?",
 | 
			
		||||
    "logoutFailTitle": "Nie udało się wylogować",
 | 
			
		||||
    "logoutFailSubtitle": "Spróbuj ponownie",
 | 
			
		||||
    "logoutSuccessTitle": "Wylogowano",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Użytkownik o nazwie <code>{{username}}</code> nie należy do grup wymaganych przez zasób <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Twój adres IP <code>{{ip}}</code> nie ma autoryzacji do dostępu do zasobu <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Spróbuj ponownie",
 | 
			
		||||
    "untrustedRedirectTitle": "Niezaufane przekierowanie",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Próbujesz przekierować do domeny, która nie pasuje do Twojej skonfigurowanej domeny (<code>{{domain}}</code>). Czy na pewno chcesz kontynuować?",
 | 
			
		||||
    "cancelTitle": "Anuluj",
 | 
			
		||||
    "forgotPasswordTitle": "Nie pamiętasz hasła?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Nie udało się załadować dostawców uwierzytelniania. Sprawdź swoją konfigurację.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "Wystąpił błąd podczas próby wykonania tej czynności. Sprawdź konsolę, aby uzyskać więcej informacji.",
 | 
			
		||||
    "forgotPasswordMessage": "Możesz zresetować hasło, zmieniając zmienną środowiskową `USERS`.",
 | 
			
		||||
    "fieldRequired": "To pole jest wymagane",
 | 
			
		||||
    "invalidInput": "Nieprawidłowe dane wejściowe"
 | 
			
		||||
    "invalidInput": "Nieprawidłowe dane wejściowe",
 | 
			
		||||
    "domainWarningTitle": "Nieprawidłowa domena",
 | 
			
		||||
    "domainWarningSubtitle": "Ta instancja jest skonfigurowana do uzyskania dostępu z <code>{{appUrl}}</code>, ale <code>{{currentUrl}}</code> jest w użyciu. Jeśli będziesz kontynuować, mogą wystąpić problemy z uwierzytelnianiem.",
 | 
			
		||||
    "ignoreTitle": "Zignoruj",
 | 
			
		||||
    "goToCorrectDomainTitle": "Przejdź do prawidłowej domeny"
 | 
			
		||||
}
 | 
			
		||||
@@ -1,34 +1,37 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "Bem-vindo de volta, acesse com",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "Or",
 | 
			
		||||
    "loginTitleSimple": "Bem-vindo de volta, faça o login",
 | 
			
		||||
    "loginDivider": "Ou",
 | 
			
		||||
    "loginUsername": "Nome de usuário",
 | 
			
		||||
    "loginPassword": "Senha",
 | 
			
		||||
    "loginSubmit": "Entrar",
 | 
			
		||||
    "loginFailTitle": "Falha ao iniciar sessão",
 | 
			
		||||
    "loginFailSubtitle": "Por favor, verifique seu usuário e senha",
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginFailRateLimit": "Você falhou em iniciar sessão muitas vezes, por favor tente novamente mais tarde",
 | 
			
		||||
    "loginSuccessTitle": "Sessão Iniciada",
 | 
			
		||||
    "loginSuccessSubtitle": "Bem-vindo de volta!",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailTitle": "Ocorreu um erro",
 | 
			
		||||
    "loginOauthFailSubtitle": "Falha ao obter URL de OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecionando",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecionando para seu provedor OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "Redirecionamento automático do OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "Você será automaticamente redirecionado para seu provedor OAuth para autenticar.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirecionar agora",
 | 
			
		||||
    "continueTitle": "Continuar",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecionando...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Você deve ser redirecionado para o aplicativo em breve",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Redirecionamento inválido",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "O endereço de redirecionamento é inválido",
 | 
			
		||||
    "continueRedirectManually": "Redirecionar-me manualmente",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Redirecionamento inseguro",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continuar",
 | 
			
		||||
    "continueSubtitle": "Clique no botão para continuar para o seu aplicativo.",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Você está tentando redirecionar de <Code>https</Code> para <Code>http</Code>, você tem certeza que deseja continuar?",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Redirecionamento não confiável",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "Você está tentando redirecionar para um domínio que não corresponde ao seu domínio configurado (<code>{{cookieDomain}}</code>). Tem certeza que deseja continuar?",
 | 
			
		||||
    "logoutFailTitle": "Falha ao encerrar sessão",
 | 
			
		||||
    "logoutFailSubtitle": "Por favor, tente novamente",
 | 
			
		||||
    "logoutSuccessTitle": "Sessão encerrada",
 | 
			
		||||
    "logoutSuccessSubtitle": "Você foi desconectado",
 | 
			
		||||
    "logoutTitle": "Sair",
 | 
			
		||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
			
		||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
			
		||||
    "logoutUsernameSubtitle": "Você está atualmente logado como <Code>{{username}}</Code>, clique no botão abaixo para sair.",
 | 
			
		||||
    "logoutOauthSubtitle": "Você está atualmente logado como <Code>{{username}}</Code> usando o provedor {{provider}} OAuth, clique no botão abaixo para sair.",
 | 
			
		||||
    "notFoundTitle": "Página não encontrada",
 | 
			
		||||
    "notFoundSubtitle": "A página que você está procurando não existe.",
 | 
			
		||||
    "notFoundButton": "Voltar para a tela inicial",
 | 
			
		||||
@@ -37,21 +40,23 @@
 | 
			
		||||
    "totpSuccessTitle": "Verificado",
 | 
			
		||||
    "totpSuccessSubtitle": "Redirecionando para o seu aplicativo",
 | 
			
		||||
    "totpTitle": "Insira o seu código TOTP",
 | 
			
		||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
			
		||||
    "totpSubtitle": "Por favor, insira o código do seu aplicativo de autenticação.",
 | 
			
		||||
    "unauthorizedTitle": "Não autorizado",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "O usuário com nome de usuário <code>{{username}}</code> não está autorizado a acessar o recurso <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "O usuário com o nome <code>{{username}}</code> não está autorizado a acessar.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "O usuário  <code>{{username}}</code> não está autorizado a acessar o recurso <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Seu endereço IP <code>{{ip}}</code> não está autorizado a acessar o recurso <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Tentar novamente",
 | 
			
		||||
    "untrustedRedirectTitle": "Redirecionamento não confiável",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancelar",
 | 
			
		||||
    "forgotPasswordTitle": "Esqueceu sua senha?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
    "errorTitle": "An error occurred",
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "failedToFetchProvidersTitle": "Falha ao carregar provedores de autenticação. Verifique sua configuração.",
 | 
			
		||||
    "errorTitle": "Ocorreu um erro",
 | 
			
		||||
    "errorSubtitle": "Ocorreu um erro ao tentar executar esta ação. Por favor, verifique o console para mais informações.",
 | 
			
		||||
    "forgotPasswordMessage": "Você pode redefinir sua senha alterando a variável de ambiente `USERS`.",
 | 
			
		||||
    "fieldRequired": "Este campo é obrigatório",
 | 
			
		||||
    "invalidInput": "Entrada Inválida",
 | 
			
		||||
    "domainWarningTitle": "Domínio inválido",
 | 
			
		||||
    "domainWarningSubtitle": "Esta instância está configurada para ser acessada de <code>{{appUrl}}</code>, mas <code>{{currentUrl}}</code> está sendo usado. Se você continuar, você pode encontrar problemas com a autenticação.",
 | 
			
		||||
    "ignoreTitle": "Ignorar",
 | 
			
		||||
    "goToCorrectDomainTitle": "Ir para o domínio correto"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -1,6 +1,6 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "С возвращением, войти с",
 | 
			
		||||
    "loginTitleSimple": "Вход",
 | 
			
		||||
    "loginTitleSimple": "С возвращением, пожалуйста войдите",
 | 
			
		||||
    "loginDivider": "Или",
 | 
			
		||||
    "loginUsername": "Имя пользователя",
 | 
			
		||||
    "loginPassword": "Пароль",
 | 
			
		||||
@@ -8,24 +8,27 @@
 | 
			
		||||
    "loginFailTitle": "Вход не удался",
 | 
			
		||||
    "loginFailSubtitle": "Проверьте имя пользователя и пароль",
 | 
			
		||||
    "loginFailRateLimit": "Слишком много ошибок входа. Попробуйте позже",
 | 
			
		||||
    "loginSuccessTitle": "Вы вошли",
 | 
			
		||||
    "loginSuccessTitle": "Вход выполнен",
 | 
			
		||||
    "loginSuccessSubtitle": "С возвращением!",
 | 
			
		||||
    "loginOauthFailTitle": "Произошла ошибка",
 | 
			
		||||
    "loginOauthFailSubtitle": "Не удалось получить OAuth URL",
 | 
			
		||||
    "loginOauthFailSubtitle": "Не удалось получить ссылку OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "Перенаправление",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Перенаправление к поставщику OAuth",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth автоматическое перенаправление",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "Вы будете автоматически перенаправлены для авторизации у вашего поставщика OAuth.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Перенаправить сейчас",
 | 
			
		||||
    "continueTitle": "Продолжить",
 | 
			
		||||
    "continueRedirectingTitle": "Перенаправление...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Скоро вы будете перенаправлены в приложение",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Неверное перенаправление",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "URL перенаправления недействителен",
 | 
			
		||||
    "continueRedirectManually": "Перенаправить вручную",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Небезопасное перенаправление",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Попытка перенаправления с <code>https</code> на <code>http</code>, уверены, что хотите продолжить?",
 | 
			
		||||
    "continueTitle": "Продолжить",
 | 
			
		||||
    "continueSubtitle": "Нажмите на кнопку, чтобы перейти к приложению.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Недоверенное перенаправление",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "Вы пытаетесь перенаправить на домен, который не соответствует вашему настроенному домену (<code>{{cookieDomain}}</code>). Вы уверены, что хотите продолжить?",
 | 
			
		||||
    "logoutFailTitle": "Не удалось выйти",
 | 
			
		||||
    "logoutFailSubtitle": "Попробуйте ещё раз",
 | 
			
		||||
    "logoutSuccessTitle": "Выход",
 | 
			
		||||
    "logoutSuccessSubtitle": "Вы вышли из системы",
 | 
			
		||||
    "logoutSuccessSubtitle": "Вы вышли",
 | 
			
		||||
    "logoutTitle": "Выйти",
 | 
			
		||||
    "logoutUsernameSubtitle": "Вход выполнен как <code>{{username}}</code>, нажмите на кнопку ниже, чтобы выйти.",
 | 
			
		||||
    "logoutOauthSubtitle": "Вход выполнен как <code>{{username}}</code> с использованием {{provider}} OAuth, нажмите кнопку ниже, чтобы выйти.",
 | 
			
		||||
@@ -37,21 +40,23 @@
 | 
			
		||||
    "totpSuccessTitle": "Подтверждён",
 | 
			
		||||
    "totpSuccessSubtitle": "Перенаправление в приложение",
 | 
			
		||||
    "totpTitle": "Введите код TOTP",
 | 
			
		||||
    "totpSubtitle": "Пожалуйста, введите код из вашего приложения — аутентификатора.",
 | 
			
		||||
    "unauthorizedTitle": "Доступ запрещен",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Пользователю <code>{{username}}</code> не разрешен доступ к <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Пользователю <code>{{username}}</code> не разрешен вход.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Пользователь <code>{{username}}</code> не состоит в группах, которым разрешен доступ к <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Ваш IP адрес <code>{{ip}}</code> не авторизован для доступа к ресурсу <code>{{resource}}</code>.",
 | 
			
		||||
    "totpSubtitle": "Пожалуйста, введите код из вашего приложения авторизации.",
 | 
			
		||||
    "unauthorizedTitle": "Доступ запрещён",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Пользователю <code>{{username}}</code> не разрешён доступ к <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Пользователю <code>{{username}}</code> не разрешён вход.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Пользователь <code>{{username}}</code> не состоит в группах, которым разрешён доступ к <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Вашему IP-адресу <code>{{ip}}</code> не разрешён доступ к ресурсу <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Повторить",
 | 
			
		||||
    "untrustedRedirectTitle": "Ненадежное перенаправление",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Попытка перенаправить на домен, который не соответствует вашему заданному домену (<code>{{domain}}</code>). Уверены, что хотите продолжить?",
 | 
			
		||||
    "cancelTitle": "Отмена",
 | 
			
		||||
    "forgotPasswordTitle": "Забыли пароль?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Не удалось загрузить провайдеров аутентификации. Пожалуйста, проверьте конфигурацию.",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Не удалось загрузить поставщика авторизации. Пожалуйста, проверьте конфигурацию.",
 | 
			
		||||
    "errorTitle": "Произошла ошибка",
 | 
			
		||||
    "errorSubtitle": "Произошла ошибка при попытке выполнить это действие. Проверьте консоль для дополнительной информации.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "forgotPasswordMessage": "Вы можете сбросить свой пароль, изменив переменную окружения `USERS`.",
 | 
			
		||||
    "fieldRequired": "Это поле является обязательным",
 | 
			
		||||
    "invalidInput": "Недопустимый ввод",
 | 
			
		||||
    "domainWarningTitle": "Неверный домен",
 | 
			
		||||
    "domainWarningSubtitle": "Этот экземпляр настроен на доступ к нему из <code>{{appUrl}}</code>, но <code>{{currentUrl}}</code> в настоящее время используется. Если вы продолжите, то могут возникнуть проблемы с авторизацией.",
 | 
			
		||||
    "ignoreTitle": "Игнорировать",
 | 
			
		||||
    "goToCorrectDomainTitle": "Перейти к правильному домену"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Неуспело преузимање OAuth адресе",
 | 
			
		||||
    "loginOauthSuccessTitle": "Преусмеравање",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Преусмеравање на вашег OAuth провајдера",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Настави",
 | 
			
		||||
    "continueRedirectingTitle": "Преусмеравање...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Требали би сте ускоро да будете преусмерени на апликацију",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Неисправно преусмеравање",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "Адреса за преусмеравање није исправна",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Небезбедно преусмеравање",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Покушавате да преусмерите са <code>https</code> на <code>http</code> што није безбедно. Да ли желите да наставите?",
 | 
			
		||||
    "continueTitle": "Настави",
 | 
			
		||||
    "continueSubtitle": "Кликните на дугме да би сте наставили на нашу апликацију.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Неуспешно одјављивање",
 | 
			
		||||
    "logoutFailSubtitle": "Молим вас покушајте поново",
 | 
			
		||||
    "logoutSuccessTitle": "Одјављени",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Корисник са корисничким именом <code>{{username}}</code> није у групама које захтева ресурс <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Ваша IP адреса <code>{{ip}}</code> није ауторизована да приступи ресурсу <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Покушајте поново",
 | 
			
		||||
    "untrustedRedirectTitle": "Преусмерење без поверења",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Покушавате да преусмерите на домен који се не поклапа са подешеним доменом (<code>{{domain}}</code>). Да ли желите да наставите?",
 | 
			
		||||
    "cancelTitle": "Поништи",
 | 
			
		||||
    "forgotPasswordTitle": "Заборавили сте лозинку?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Није успело учитавање провајдера аутентификације. Молим вас проверите ваша подешавања.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "Појавила се грешка при покушају извршавања ове радње. Молим вас проверите конзолу за додатне информације.",
 | 
			
		||||
    "forgotPasswordMessage": "Можете поништити вашу лозинку променом `USERS` променљиве окружења.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -1,32 +1,35 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "Welcome back, login with",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "Or",
 | 
			
		||||
    "loginUsername": "Username",
 | 
			
		||||
    "loginPassword": "Password",
 | 
			
		||||
    "loginSubmit": "Login",
 | 
			
		||||
    "loginFailTitle": "Failed to log in",
 | 
			
		||||
    "loginFailSubtitle": "Please check your username and password",
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "Logged in",
 | 
			
		||||
    "loginSuccessSubtitle": "Welcome back!",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
    "logoutSuccessSubtitle": "You have been logged out",
 | 
			
		||||
    "logoutTitle": "Logout",
 | 
			
		||||
    "loginTitle": "Välkommen tillbaka, logga in med",
 | 
			
		||||
    "loginTitleSimple": "Välkommen tillbaka, logga in",
 | 
			
		||||
    "loginDivider": "Eller",
 | 
			
		||||
    "loginUsername": "Användarnamn",
 | 
			
		||||
    "loginPassword": "Lösenord",
 | 
			
		||||
    "loginSubmit": "Logga in",
 | 
			
		||||
    "loginFailTitle": "Kunde inte logga in",
 | 
			
		||||
    "loginFailSubtitle": "Kontrollera ditt användarnamn och lösenord",
 | 
			
		||||
    "loginFailRateLimit": "Du misslyckades med att logga in för många gånger. Försök igen senare",
 | 
			
		||||
    "loginSuccessTitle": "Inloggad",
 | 
			
		||||
    "loginSuccessSubtitle": "Välkommen tillbaka!",
 | 
			
		||||
    "loginOauthFailTitle": "Ett fel har uppstått",
 | 
			
		||||
    "loginOauthFailSubtitle": "Kunde inte hämta OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Omdirigerar",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Omdirigera till din OAuth leverantör",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Fortsätt",
 | 
			
		||||
    "continueRedirectingTitle": "Omdirigerar...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Du bör omdirigeras till appen snart",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Osäker omdirigering",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Du försöker omdirigera från <code>https</code> till <code>http</code> som inte är säker. Är du säker på att du vill fortsätta?",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Kunde inte logga ut.",
 | 
			
		||||
    "logoutFailSubtitle": "Vänligen försök igen",
 | 
			
		||||
    "logoutSuccessTitle": "Utloggad",
 | 
			
		||||
    "logoutSuccessSubtitle": "Du har blivit utloggad",
 | 
			
		||||
    "logoutTitle": "Logga ut",
 | 
			
		||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
			
		||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
			
		||||
    "notFoundTitle": "Page not found",
 | 
			
		||||
@@ -38,14 +41,12 @@
 | 
			
		||||
    "totpSuccessSubtitle": "Redirecting to your app",
 | 
			
		||||
    "totpTitle": "Enter your TOTP code",
 | 
			
		||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
			
		||||
    "unauthorizedTitle": "Unauthorized",
 | 
			
		||||
    "unauthorizedTitle": "Obehörig",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Yönlendiriliyor",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Devam et",
 | 
			
		||||
    "continueRedirectingTitle": "Yönlendiriliyor...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Devam et",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Lütfen tekrar deneyin",
 | 
			
		||||
    "logoutSuccessTitle": "Çıkış yapıldı",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "İptal",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "Welcome back, login with",
 | 
			
		||||
    "loginTitle": "З поверненням, увійдіть через",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "Or",
 | 
			
		||||
    "loginUsername": "Username",
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information.",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "获取 OAuth URL 失败",
 | 
			
		||||
    "loginOauthSuccessTitle": "重定向中",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "重定向到您的 OAuth 提供商",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth自动重定向",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "您将被自动重定向到您的 OAuth 提供商进行身份验证。",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "立即跳转",
 | 
			
		||||
    "continueTitle": "继续",
 | 
			
		||||
    "continueRedirectingTitle": "正在重定向……",
 | 
			
		||||
    "continueRedirectingSubtitle": "您应该很快被重定向到应用",
 | 
			
		||||
    "continueInvalidRedirectTitle": "无效的重定向",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "重定向URL无效",
 | 
			
		||||
    "continueRedirectManually": "请手动跳转",
 | 
			
		||||
    "continueInsecureRedirectTitle": "不安全的重定向",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "您正在尝试从<code>https</code>重定向到<code>http</code>可能存在风险。您确定要继续吗?",
 | 
			
		||||
    "continueTitle": "继续",
 | 
			
		||||
    "continueSubtitle": "点击按钮以继续您的应用。",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "不可信的重定向",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "您尝试跳转的域名与配置的域名(<code>{{cookieDomain}}</code>)不匹配。是否继续?",
 | 
			
		||||
    "logoutFailTitle": "注销失败",
 | 
			
		||||
    "logoutFailSubtitle": "请重试",
 | 
			
		||||
    "logoutSuccessTitle": "已登出",
 | 
			
		||||
@@ -30,7 +33,7 @@
 | 
			
		||||
    "logoutUsernameSubtitle": "您当前登录用户为<code>{{username}}</code>。点击下方按钮注销。",
 | 
			
		||||
    "logoutOauthSubtitle": "您当前以<code>{{username}}</code>登录,使用的是{{provider}} OAuth 提供商。点击下方按钮注销。",
 | 
			
		||||
    "notFoundTitle": "无法找到页面",
 | 
			
		||||
    "notFoundSubtitle": "您正在查找的页面不存在。",
 | 
			
		||||
    "notFoundSubtitle": "您访问的页面不存在。",
 | 
			
		||||
    "notFoundButton": "回到主页",
 | 
			
		||||
    "totpFailTitle": "无法验证代码",
 | 
			
		||||
    "totpFailSubtitle": "请检查您的代码并重试",
 | 
			
		||||
@@ -42,16 +45,18 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "用户名为<code>{{username}}</code>的用户无权访问资源<code>{{resource}}</code>。",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "用户名为<code>{{username}}</code>的用户无权登录。",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "用户名为<code>{{username}}</code>的用户不在资源<code>{{resource}}</code>所需的组中。",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "用户 <code>{{ip}}</code> 无权访问资源 <code>{{resource}}</code>。",
 | 
			
		||||
    "unauthorizedButton": "重试",
 | 
			
		||||
    "untrustedRedirectTitle": "不可信的重定向",
 | 
			
		||||
    "untrustedRedirectSubtitle": "您正在尝试重定向到一个与您已配置的域名 (<code>{{domain}}</code>) 不匹配的域名。您确定要继续吗?",
 | 
			
		||||
    "cancelTitle": "取消",
 | 
			
		||||
    "forgotPasswordTitle": "忘记密码?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "加载身份验证提供程序失败,请检查您的配置。",
 | 
			
		||||
    "errorTitle": "发生了错误",
 | 
			
		||||
    "errorSubtitle": "执行此操作时发生错误,请检查控制台以获取更多信息。",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "forgotPasswordMessage": "您可以通过更改 `USERS ` 环境变量重置您的密码。",
 | 
			
		||||
    "fieldRequired": "必添字段",
 | 
			
		||||
    "invalidInput": "无效的输入",
 | 
			
		||||
    "domainWarningTitle": "无效域名",
 | 
			
		||||
    "domainWarningSubtitle": "当前实例配置的访问地址为 <code>{{appUrl}}</code>,但您正在使用 <code>{{currentUrl}}</code>。若继续操作,可能会遇到身份验证问题。",
 | 
			
		||||
    "ignoreTitle": "忽略",
 | 
			
		||||
    "goToCorrectDomainTitle": "转到正确的域名"
 | 
			
		||||
}
 | 
			
		||||
@@ -14,14 +14,17 @@
 | 
			
		||||
    "loginOauthFailSubtitle": "無法取得 OAuth 網址",
 | 
			
		||||
    "loginOauthSuccessTitle": "重新導向中",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "正在將您重新導向至 OAuth 供應商",
 | 
			
		||||
    "loginOauthAutoRedirectTitle": "OAuth Auto Redirect",
 | 
			
		||||
    "loginOauthAutoRedirectSubtitle": "You will be automatically redirected to your OAuth provider to authenticate.",
 | 
			
		||||
    "loginOauthAutoRedirectButton": "Redirect now",
 | 
			
		||||
    "continueTitle": "繼續",
 | 
			
		||||
    "continueRedirectingTitle": "重新導向中……",
 | 
			
		||||
    "continueRedirectingSubtitle": "您即將被重新導向至應用程式",
 | 
			
		||||
    "continueInvalidRedirectTitle": "無效的重新導向",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "重新導向的網址無效",
 | 
			
		||||
    "continueRedirectManually": "Redirect me manually",
 | 
			
		||||
    "continueInsecureRedirectTitle": "不安全的重新導向",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "您正嘗試從安全的 <code>https</code> 重新導向至不安全的 <code>http</code>。您確定要繼續嗎?",
 | 
			
		||||
    "continueTitle": "繼續",
 | 
			
		||||
    "continueSubtitle": "點擊按鈕以繼續前往您的應用程式。",
 | 
			
		||||
    "continueUntrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "continueUntrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{cookieDomain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "logoutFailTitle": "登出失敗",
 | 
			
		||||
    "logoutFailSubtitle": "請再試一次",
 | 
			
		||||
    "logoutSuccessTitle": "登出成功",
 | 
			
		||||
@@ -44,8 +47,6 @@
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "使用者 <code>{{username}}</code> 不在存取資源 <code>{{resource}}</code> 所需的群組中。",
 | 
			
		||||
    "unauthorizedIpSubtitle": "您的 IP 位址 <code>{{ip}}</code> 未被授權存取資源 <code>{{resource}}</code>。",
 | 
			
		||||
    "unauthorizedButton": "再試一次",
 | 
			
		||||
    "untrustedRedirectTitle": "不受信任的重新導向",
 | 
			
		||||
    "untrustedRedirectSubtitle": "您正嘗試重新導向至的網域與您設定的網域 (<code>{{domain}}</code>) 不符。您確定要繼續嗎?",
 | 
			
		||||
    "cancelTitle": "取消",
 | 
			
		||||
    "forgotPasswordTitle": "忘記密碼?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "載入驗證供應商失敗。請檢查您的設定。",
 | 
			
		||||
@@ -53,5 +54,9 @@
 | 
			
		||||
    "errorSubtitle": "執行此操作時發生錯誤。請檢查主控台以獲取更多資訊。",
 | 
			
		||||
    "forgotPasswordMessage": "You can reset your password by changing the `USERS` environment variable.",
 | 
			
		||||
    "fieldRequired": "This field is required",
 | 
			
		||||
    "invalidInput": "Invalid input"
 | 
			
		||||
    "invalidInput": "Invalid input",
 | 
			
		||||
    "domainWarningTitle": "Invalid Domain",
 | 
			
		||||
    "domainWarningSubtitle": "This instance is configured to be accessed from <code>{{appUrl}}</code>, but <code>{{currentUrl}}</code> is being used. If you proceed, you may encounter issues with authentication.",
 | 
			
		||||
    "ignoreTitle": "Ignore",
 | 
			
		||||
    "goToCorrectDomainTitle": "Go to correct domain"
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										35
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								go.mod
									
									
									
									
									
								
							@@ -8,7 +8,7 @@ require (
 | 
			
		||||
	github.com/cenkalti/backoff/v5 v5.0.3
 | 
			
		||||
	github.com/gin-gonic/gin v1.11.0
 | 
			
		||||
	github.com/glebarez/sqlite v1.11.0
 | 
			
		||||
	github.com/go-playground/validator/v10 v10.27.0
 | 
			
		||||
	github.com/go-playground/validator/v10 v10.28.0
 | 
			
		||||
	github.com/golang-migrate/migrate/v4 v4.19.0
 | 
			
		||||
	github.com/google/go-querystring v1.1.0
 | 
			
		||||
	github.com/google/uuid v1.6.0
 | 
			
		||||
@@ -18,7 +18,7 @@ require (
 | 
			
		||||
	github.com/spf13/viper v1.21.0
 | 
			
		||||
	github.com/traefik/paerser v0.2.2
 | 
			
		||||
	github.com/weppos/publicsuffix-go v0.50.0
 | 
			
		||||
	golang.org/x/crypto v0.42.0
 | 
			
		||||
	golang.org/x/crypto v0.43.0
 | 
			
		||||
	golang.org/x/exp v0.0.0-20250620022241-b7579e27df2b
 | 
			
		||||
	gorm.io/gorm v1.31.0
 | 
			
		||||
	gotest.tools/v3 v3.5.2
 | 
			
		||||
@@ -45,17 +45,18 @@ require (
 | 
			
		||||
	github.com/moby/term v0.5.2 // indirect
 | 
			
		||||
	github.com/ncruces/go-strftime v0.1.9 // indirect
 | 
			
		||||
	github.com/quic-go/qpack v0.5.1 // indirect
 | 
			
		||||
	github.com/quic-go/quic-go v0.54.0 // indirect
 | 
			
		||||
	github.com/quic-go/quic-go v0.54.1 // indirect
 | 
			
		||||
	github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec // indirect
 | 
			
		||||
	github.com/stoewer/go-strcase v1.3.1 // indirect
 | 
			
		||||
	github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect
 | 
			
		||||
	go.opentelemetry.io/auto/sdk v1.1.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.34.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/sdk v1.34.0 // indirect
 | 
			
		||||
	go.uber.org/mock v0.5.0 // indirect
 | 
			
		||||
	go.yaml.in/yaml/v3 v3.0.4 // indirect
 | 
			
		||||
	golang.org/x/mod v0.27.0 // indirect
 | 
			
		||||
	golang.org/x/term v0.35.0 // indirect
 | 
			
		||||
	golang.org/x/tools v0.36.0 // indirect
 | 
			
		||||
	golang.org/x/mod v0.28.0 // indirect
 | 
			
		||||
	golang.org/x/term v0.36.0 // indirect
 | 
			
		||||
	golang.org/x/tools v0.37.0 // indirect
 | 
			
		||||
	modernc.org/libc v1.66.3 // indirect
 | 
			
		||||
	modernc.org/mathutil v1.7.1 // indirect
 | 
			
		||||
	modernc.org/memory v1.11.0 // indirect
 | 
			
		||||
@@ -71,25 +72,25 @@ require (
 | 
			
		||||
	github.com/bytedance/sonic v1.14.0 // indirect
 | 
			
		||||
	github.com/bytedance/sonic/loader v0.3.0 // indirect
 | 
			
		||||
	github.com/catppuccin/go v0.3.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/bubbles v0.21.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/bubbletea v1.3.4 // indirect
 | 
			
		||||
	github.com/charmbracelet/huh v0.7.0
 | 
			
		||||
	github.com/charmbracelet/bubbles v0.21.1-0.20250623103423-23b8fd6302d7 // indirect
 | 
			
		||||
	github.com/charmbracelet/bubbletea v1.3.6 // indirect
 | 
			
		||||
	github.com/charmbracelet/huh v0.8.0
 | 
			
		||||
	github.com/charmbracelet/lipgloss v1.1.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/x/ansi v0.8.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/x/ansi v0.9.3 // indirect
 | 
			
		||||
	github.com/charmbracelet/x/exp/strings v0.0.0-20240722160745-212f7b056ed0 // indirect
 | 
			
		||||
	github.com/charmbracelet/x/term v0.2.1 // indirect
 | 
			
		||||
	github.com/cloudwego/base64x v0.1.6 // indirect
 | 
			
		||||
	github.com/distribution/reference v0.6.0 // indirect
 | 
			
		||||
	github.com/docker/docker v28.4.0+incompatible
 | 
			
		||||
	github.com/docker/docker v28.5.1+incompatible
 | 
			
		||||
	github.com/docker/go-connections v0.5.0 // indirect
 | 
			
		||||
	github.com/docker/go-units v0.5.0 // indirect
 | 
			
		||||
	github.com/dustin/go-humanize v1.0.1 // indirect
 | 
			
		||||
	github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect
 | 
			
		||||
	github.com/felixge/httpsnoop v1.0.4 // indirect
 | 
			
		||||
	github.com/fsnotify/fsnotify v1.9.0 // indirect
 | 
			
		||||
	github.com/gabriel-vasile/mimetype v1.4.8 // indirect
 | 
			
		||||
	github.com/gabriel-vasile/mimetype v1.4.10 // indirect
 | 
			
		||||
	github.com/gin-contrib/sse v1.1.0 // indirect
 | 
			
		||||
	github.com/go-ldap/ldap/v3 v3.4.11
 | 
			
		||||
	github.com/go-ldap/ldap/v3 v3.4.12
 | 
			
		||||
	github.com/go-logr/logr v1.4.3 // indirect
 | 
			
		||||
	github.com/go-logr/stdr v1.2.2 // indirect
 | 
			
		||||
	github.com/go-playground/locales v0.14.1 // indirect
 | 
			
		||||
@@ -130,10 +131,10 @@ require (
 | 
			
		||||
	go.opentelemetry.io/otel/metric v1.37.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/trace v1.37.0 // indirect
 | 
			
		||||
	golang.org/x/arch v0.20.0 // indirect
 | 
			
		||||
	golang.org/x/net v0.44.0 // indirect
 | 
			
		||||
	golang.org/x/oauth2 v0.31.0
 | 
			
		||||
	golang.org/x/net v0.45.0 // indirect
 | 
			
		||||
	golang.org/x/oauth2 v0.32.0
 | 
			
		||||
	golang.org/x/sync v0.17.0 // indirect
 | 
			
		||||
	golang.org/x/sys v0.36.0 // indirect
 | 
			
		||||
	golang.org/x/text v0.29.0 // indirect
 | 
			
		||||
	golang.org/x/sys v0.37.0 // indirect
 | 
			
		||||
	golang.org/x/text v0.30.0 // indirect
 | 
			
		||||
	google.golang.org/protobuf v1.36.9 // indirect
 | 
			
		||||
)
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										78
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										78
									
								
								go.sum
									
									
									
									
									
								
							@@ -6,14 +6,14 @@ github.com/MakeNowJust/heredoc v1.0.0 h1:cXCdzVdstXyiTqTvfqk9SDHpKNjxuom+DOlyEeQ
 | 
			
		||||
github.com/MakeNowJust/heredoc v1.0.0/go.mod h1:mG5amYoWBHf8vpLOuehzbGGw0EHxpZZ6lCpQ4fNJ8LE=
 | 
			
		||||
github.com/Microsoft/go-winio v0.6.2 h1:F2VQgta7ecxGYO8k3ZZz3RS8fVIXVxONVUPlNERoyfY=
 | 
			
		||||
github.com/Microsoft/go-winio v0.6.2/go.mod h1:yd8OoFMLzJbo9gZq8j5qaps8bJ9aShtEA8Ipt1oGCvU=
 | 
			
		||||
github.com/alexbrainman/sspi v0.0.0-20231016080023-1a75b4708caa h1:LHTHcTQiSGT7VVbI0o4wBRNQIgn917usHWOd6VAffYI=
 | 
			
		||||
github.com/alexbrainman/sspi v0.0.0-20231016080023-1a75b4708caa/go.mod h1:cEWa1LVoE5KvSD9ONXsZrj0z6KqySlCCNKHlLzbqAt4=
 | 
			
		||||
github.com/alexbrainman/sspi v0.0.0-20250919150558-7d374ff0d59e h1:4dAU9FXIyQktpoUAgOJK3OTFc/xug0PCXYCqU0FgDKI=
 | 
			
		||||
github.com/alexbrainman/sspi v0.0.0-20250919150558-7d374ff0d59e/go.mod h1:cEWa1LVoE5KvSD9ONXsZrj0z6KqySlCCNKHlLzbqAt4=
 | 
			
		||||
github.com/atotto/clipboard v0.1.4 h1:EH0zSVneZPSuFR11BlR9YppQTVDbh5+16AmcJi4g1z4=
 | 
			
		||||
github.com/atotto/clipboard v0.1.4/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI=
 | 
			
		||||
github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k=
 | 
			
		||||
github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8=
 | 
			
		||||
github.com/aymanbagabas/go-udiff v0.2.0 h1:TK0fH4MteXUDspT88n8CKzvK0X9O2xu9yQjWpi6yML8=
 | 
			
		||||
github.com/aymanbagabas/go-udiff v0.2.0/go.mod h1:RE4Ex0qsGkTAJoQdQQCA0uG+nAzJO/pI/QwceO5fgrA=
 | 
			
		||||
github.com/aymanbagabas/go-udiff v0.3.1 h1:LV+qyBQ2pqe0u42ZsUEtPiCaUoqgA9gYRDs3vj1nolY=
 | 
			
		||||
github.com/aymanbagabas/go-udiff v0.3.1/go.mod h1:G0fsKmG+P6ylD0r6N/KgQD/nWzgfnl8ZBcNLgcbrw8E=
 | 
			
		||||
github.com/boombuler/barcode v1.0.1-0.20190219062509-6c824513bacc/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8=
 | 
			
		||||
github.com/boombuler/barcode v1.0.2 h1:79yrbttoZrLGkL/oOI8hBrUKucwOL0oOjUgEguGMcJ4=
 | 
			
		||||
github.com/boombuler/barcode v1.0.2/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8=
 | 
			
		||||
@@ -27,18 +27,18 @@ github.com/cenkalti/backoff/v4 v4.3.0 h1:MyRJ/UdXutAwSAT+s3wNd7MfTIcy71VQueUuFK3
 | 
			
		||||
github.com/cenkalti/backoff/v4 v4.3.0/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE=
 | 
			
		||||
github.com/cenkalti/backoff/v5 v5.0.3 h1:ZN+IMa753KfX5hd8vVaMixjnqRZ3y8CuJKRKj1xcsSM=
 | 
			
		||||
github.com/cenkalti/backoff/v5 v5.0.3/go.mod h1:rkhZdG3JZukswDf7f0cwqPNk4K0sa+F97BxZthm/crw=
 | 
			
		||||
github.com/charmbracelet/bubbles v0.21.0 h1:9TdC97SdRVg/1aaXNVWfFH3nnLAwOXr8Fn6u6mfQdFs=
 | 
			
		||||
github.com/charmbracelet/bubbles v0.21.0/go.mod h1:HF+v6QUR4HkEpz62dx7ym2xc71/KBHg+zKwJtMw+qtg=
 | 
			
		||||
github.com/charmbracelet/bubbletea v1.3.4 h1:kCg7B+jSCFPLYRA52SDZjr51kG/fMUEoPoZrkaDHyoI=
 | 
			
		||||
github.com/charmbracelet/bubbletea v1.3.4/go.mod h1:dtcUCyCGEX3g9tosuYiut3MXgY/Jsv9nKVdibKKRRXo=
 | 
			
		||||
github.com/charmbracelet/bubbles v0.21.1-0.20250623103423-23b8fd6302d7 h1:JFgG/xnwFfbezlUnFMJy0nusZvytYysV4SCS2cYbvws=
 | 
			
		||||
github.com/charmbracelet/bubbles v0.21.1-0.20250623103423-23b8fd6302d7/go.mod h1:ISC1gtLcVilLOf23wvTfoQuYbW2q0JevFxPfUzZ9Ybw=
 | 
			
		||||
github.com/charmbracelet/bubbletea v1.3.6 h1:VkHIxPJQeDt0aFJIsVxw8BQdh/F/L2KKZGsK6et5taU=
 | 
			
		||||
github.com/charmbracelet/bubbletea v1.3.6/go.mod h1:oQD9VCRQFF8KplacJLo28/jofOI2ToOfGYeFgBBxHOc=
 | 
			
		||||
github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc h1:4pZI35227imm7yK2bGPcfpFEmuY1gc2YSTShr4iJBfs=
 | 
			
		||||
github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc/go.mod h1:X4/0JoqgTIPSFcRA/P6INZzIuyqdFY5rm8tb41s9okk=
 | 
			
		||||
github.com/charmbracelet/huh v0.7.0 h1:W8S1uyGETgj9Tuda3/JdVkc3x7DBLZYPZc4c+/rnRdc=
 | 
			
		||||
github.com/charmbracelet/huh v0.7.0/go.mod h1:UGC3DZHlgOKHvHC07a5vHag41zzhpPFj34U92sOmyuk=
 | 
			
		||||
github.com/charmbracelet/huh v0.8.0 h1:Xz/Pm2h64cXQZn/Jvele4J3r7DDiqFCNIVteYukxDvY=
 | 
			
		||||
github.com/charmbracelet/huh v0.8.0/go.mod h1:5YVc+SlZ1IhQALxRPpkGwwEKftN/+OlJlnJYlDRFqN4=
 | 
			
		||||
github.com/charmbracelet/lipgloss v1.1.0 h1:vYXsiLHVkK7fp74RkV7b2kq9+zDLoEU4MZoFqR/noCY=
 | 
			
		||||
github.com/charmbracelet/lipgloss v1.1.0/go.mod h1:/6Q8FR2o+kj8rz4Dq0zQc3vYf7X+B0binUUBwA0aL30=
 | 
			
		||||
github.com/charmbracelet/x/ansi v0.8.0 h1:9GTq3xq9caJW8ZrBTe0LIe2fvfLR/bYXKTx2llXn7xE=
 | 
			
		||||
github.com/charmbracelet/x/ansi v0.8.0/go.mod h1:wdYl/ONOLHLIVmQaxbIYEC/cRKOQyjTkowiI4blgS9Q=
 | 
			
		||||
github.com/charmbracelet/x/ansi v0.9.3 h1:BXt5DHS/MKF+LjuK4huWrC6NCvHtexww7dMayh6GXd0=
 | 
			
		||||
github.com/charmbracelet/x/ansi v0.9.3/go.mod h1:3RQDQ6lDnROptfpWuUVIUG64bD2g2BgntdxH0Ya5TeE=
 | 
			
		||||
github.com/charmbracelet/x/cellbuf v0.0.13 h1:/KBBKHuVRbq1lYx5BzEHBAFBP8VcQzJejZ/IA3iR28k=
 | 
			
		||||
github.com/charmbracelet/x/cellbuf v0.0.13/go.mod h1:xe0nKWGd3eJgtqZRaN9RjMtK7xUYchjzPr7q6kcvCCs=
 | 
			
		||||
github.com/charmbracelet/x/conpty v0.1.0 h1:4zc8KaIcbiL4mghEON8D72agYtSeIgq8FSThSPQIb+U=
 | 
			
		||||
@@ -72,8 +72,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
 | 
			
		||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/distribution/reference v0.6.0 h1:0IXCQ5g4/QMHHkarYzh5l+u8T3t73zM5QvfrDyIgxBk=
 | 
			
		||||
github.com/distribution/reference v0.6.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E=
 | 
			
		||||
github.com/docker/docker v28.4.0+incompatible h1:KVC7bz5zJY/4AZe/78BIvCnPsLaC9T/zh72xnlrTTOk=
 | 
			
		||||
github.com/docker/docker v28.4.0+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
 | 
			
		||||
github.com/docker/docker v28.5.1+incompatible h1:Bm8DchhSD2J6PsFzxC35TZo4TLGR2PdW/E69rU45NhM=
 | 
			
		||||
github.com/docker/docker v28.5.1+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
 | 
			
		||||
github.com/docker/go-connections v0.5.0 h1:USnMq7hx7gwdVZq1L49hLXaFtUdTADjXGp+uj1Br63c=
 | 
			
		||||
github.com/docker/go-connections v0.5.0/go.mod h1:ov60Kzw0kKElRwhNs9UlUHAE/F9Fe6GLaXnqyDdmEXc=
 | 
			
		||||
github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4=
 | 
			
		||||
@@ -88,8 +88,8 @@ github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHk
 | 
			
		||||
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
 | 
			
		||||
github.com/fsnotify/fsnotify v1.9.0 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k=
 | 
			
		||||
github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0=
 | 
			
		||||
github.com/gabriel-vasile/mimetype v1.4.8 h1:FfZ3gj38NjllZIeJAmMhr+qKL8Wu+nOoI3GqacKw1NM=
 | 
			
		||||
github.com/gabriel-vasile/mimetype v1.4.8/go.mod h1:ByKUIKGjh1ODkGM1asKUbQZOLGrPjydw3hYPU2YU9t8=
 | 
			
		||||
github.com/gabriel-vasile/mimetype v1.4.10 h1:zyueNbySn/z8mJZHLt6IPw0KoZsiQNszIpU+bX4+ZK0=
 | 
			
		||||
github.com/gabriel-vasile/mimetype v1.4.10/go.mod h1:d+9Oxyo1wTzWdyVUPMmXFvp4F9tea18J8ufA774AB3s=
 | 
			
		||||
github.com/gin-contrib/sse v1.1.0 h1:n0w2GMuUpWDVp7qSpvze6fAu9iRxJY4Hmj6AmBOU05w=
 | 
			
		||||
github.com/gin-contrib/sse v1.1.0/go.mod h1:hxRZ5gVpWMT7Z0B0gSNYqqsSCNIJMjzvm6fqCz9vjwM=
 | 
			
		||||
github.com/gin-gonic/gin v1.11.0 h1:OW/6PLjyusp2PPXtyxKHU0RbX6I/l28FTdDlae5ueWk=
 | 
			
		||||
@@ -100,8 +100,8 @@ github.com/glebarez/sqlite v1.11.0 h1:wSG0irqzP6VurnMEpFGer5Li19RpIRi2qvQz++w0GM
 | 
			
		||||
github.com/glebarez/sqlite v1.11.0/go.mod h1:h8/o8j5wiAsqSPoWELDUdJXhjAhsVliSn7bWZjOhrgQ=
 | 
			
		||||
github.com/go-asn1-ber/asn1-ber v1.5.8-0.20250403174932-29230038a667 h1:BP4M0CvQ4S3TGls2FvczZtj5Re/2ZzkV9VwqPHH/3Bo=
 | 
			
		||||
github.com/go-asn1-ber/asn1-ber v1.5.8-0.20250403174932-29230038a667/go.mod h1:hEBeB/ic+5LoWskz+yKT7vGhhPYkProFKoKdwZRWMe0=
 | 
			
		||||
github.com/go-ldap/ldap/v3 v3.4.11 h1:4k0Yxweg+a3OyBLjdYn5OKglv18JNvfDykSoI8bW0gU=
 | 
			
		||||
github.com/go-ldap/ldap/v3 v3.4.11/go.mod h1:bY7t0FLK8OAVpp/vV6sSlpz3EQDGcQwc8pF0ujLgKvM=
 | 
			
		||||
github.com/go-ldap/ldap/v3 v3.4.12 h1:1b81mv7MagXZ7+1r7cLTWmyuTqVqdwbtJSjC0DAp9s4=
 | 
			
		||||
github.com/go-ldap/ldap/v3 v3.4.12/go.mod h1:+SPAGcTtOfmGsCb3h1RFiq4xpp4N636G75OEace8lNo=
 | 
			
		||||
github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
 | 
			
		||||
github.com/go-logr/logr v1.4.3 h1:CjnDlHq8ikf6E492q6eKboGOC0T8CDaOvkHCIg8idEI=
 | 
			
		||||
github.com/go-logr/logr v1.4.3/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
 | 
			
		||||
@@ -113,8 +113,8 @@ github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/o
 | 
			
		||||
github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY=
 | 
			
		||||
github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY=
 | 
			
		||||
github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.27.0 h1:w8+XrWVMhGkxOaaowyKH35gFydVHOvC0/uWoy2Fzwn4=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.27.0/go.mod h1:I5QpIEbmr8On7W0TktmJAumgzX4CA1XNl4ZmDuVHKKo=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.28.0 h1:Q7ibns33JjyW48gHkuFT91qX48KG0ktULL6FgHdG688=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.28.0/go.mod h1:GoI6I1SjPBh9p7ykNE/yj3fFYbyDOpwMn5KXd+m2hUU=
 | 
			
		||||
github.com/go-viper/mapstructure/v2 v2.4.0 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs=
 | 
			
		||||
github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM=
 | 
			
		||||
github.com/goccy/go-json v0.10.4 h1:JSwxQzIqKfmFX1swYPpUThQZp/Ka4wzJdK0LWVytLPM=
 | 
			
		||||
@@ -229,8 +229,8 @@ github.com/pquerna/otp v1.5.0 h1:NMMR+WrmaqXU4EzdGJEE1aUUI0AMRzsp96fFFWNPwxs=
 | 
			
		||||
github.com/pquerna/otp v1.5.0/go.mod h1:dkJfzwRKNiegxyNb54X/3fLwhCynbMspSyWKnvi1AEg=
 | 
			
		||||
github.com/quic-go/qpack v0.5.1 h1:giqksBPnT/HDtZ6VhtFKgoLOWmlyo9Ei6u9PqzIMbhI=
 | 
			
		||||
github.com/quic-go/qpack v0.5.1/go.mod h1:+PC4XFrEskIVkcLzpEkbLqq1uCoxPhQuvK5rH1ZgaEg=
 | 
			
		||||
github.com/quic-go/quic-go v0.54.0 h1:6s1YB9QotYI6Ospeiguknbp2Znb/jZYjZLRXn9kMQBg=
 | 
			
		||||
github.com/quic-go/quic-go v0.54.0/go.mod h1:e68ZEaCdyviluZmy44P6Iey98v/Wfz6HCjQEm+l8zTY=
 | 
			
		||||
github.com/quic-go/quic-go v0.54.1 h1:4ZAWm0AhCb6+hE+l5Q1NAL0iRn/ZrMwqHRGQiFwj2eg=
 | 
			
		||||
github.com/quic-go/quic-go v0.54.1/go.mod h1:e68ZEaCdyviluZmy44P6Iey98v/Wfz6HCjQEm+l8zTY=
 | 
			
		||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE=
 | 
			
		||||
github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo=
 | 
			
		||||
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
 | 
			
		||||
@@ -259,6 +259,8 @@ github.com/spf13/pflag v1.0.10 h1:4EBh2KAYBwaONj6b2Ye1GiHfwjqyROoF4RwYO+vPwFk=
 | 
			
		||||
github.com/spf13/pflag v1.0.10/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
 | 
			
		||||
github.com/spf13/viper v1.21.0 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU=
 | 
			
		||||
github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY=
 | 
			
		||||
github.com/stoewer/go-strcase v1.3.1 h1:iS0MdW+kVTxgMoE1LAZyMiYJFKlOzLooE4MxjirtkAs=
 | 
			
		||||
github.com/stoewer/go-strcase v1.3.1/go.mod h1:fAH5hQ5pehh+j3nZfvwdk2RgEgQjAoM8wodgtPmh1xo=
 | 
			
		||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
 | 
			
		||||
github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw=
 | 
			
		||||
github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo=
 | 
			
		||||
@@ -304,32 +306,32 @@ go.yaml.in/yaml/v3 v3.0.4 h1:tfq32ie2Jv2UxXFdLJdh3jXuOzWiL1fo0bu/FbuKpbc=
 | 
			
		||||
go.yaml.in/yaml/v3 v3.0.4/go.mod h1:DhzuOOF2ATzADvBadXxruRBLzYTpT36CKvDb3+aBEFg=
 | 
			
		||||
golang.org/x/arch v0.20.0 h1:dx1zTU0MAE98U+TQ8BLl7XsJbgze2WnNKF/8tGp/Q6c=
 | 
			
		||||
golang.org/x/arch v0.20.0/go.mod h1:bdwinDaKcfZUGpH09BB7ZmOfhalA8lQdzl62l8gGWsk=
 | 
			
		||||
golang.org/x/crypto v0.42.0 h1:chiH31gIWm57EkTXpwnqf8qeuMUi0yekh6mT2AvFlqI=
 | 
			
		||||
golang.org/x/crypto v0.42.0/go.mod h1:4+rDnOTJhQCx2q7/j6rAN5XDw8kPjeaXEUR2eL94ix8=
 | 
			
		||||
golang.org/x/crypto v0.43.0 h1:dduJYIi3A3KOfdGOHX8AVZ/jGiyPa3IbBozJ5kNuE04=
 | 
			
		||||
golang.org/x/crypto v0.43.0/go.mod h1:BFbav4mRNlXJL4wNeejLpWxB7wMbc79PdRGhWKncxR0=
 | 
			
		||||
golang.org/x/exp v0.0.0-20250620022241-b7579e27df2b h1:M2rDM6z3Fhozi9O7NWsxAkg/yqS/lQJ6PmkyIV3YP+o=
 | 
			
		||||
golang.org/x/exp v0.0.0-20250620022241-b7579e27df2b/go.mod h1:3//PLf8L/X+8b4vuAfHzxeRUl04Adcb341+IGKfnqS8=
 | 
			
		||||
golang.org/x/mod v0.27.0 h1:kb+q2PyFnEADO2IEF935ehFUXlWiNjJWtRNgBLSfbxQ=
 | 
			
		||||
golang.org/x/mod v0.27.0/go.mod h1:rWI627Fq0DEoudcK+MBkNkCe0EetEaDSwJJkCcjpazc=
 | 
			
		||||
golang.org/x/net v0.44.0 h1:evd8IRDyfNBMBTTY5XRF1vaZlD+EmWx6x8PkhR04H/I=
 | 
			
		||||
golang.org/x/net v0.44.0/go.mod h1:ECOoLqd5U3Lhyeyo/QDCEVQ4sNgYsqvCZ722XogGieY=
 | 
			
		||||
golang.org/x/oauth2 v0.31.0 h1:8Fq0yVZLh4j4YA47vHKFTa9Ew5XIrCP8LC6UeNZnLxo=
 | 
			
		||||
golang.org/x/oauth2 v0.31.0/go.mod h1:lzm5WQJQwKZ3nwavOZ3IS5Aulzxi68dUSgRHujetwEA=
 | 
			
		||||
golang.org/x/mod v0.28.0 h1:gQBtGhjxykdjY9YhZpSlZIsbnaE2+PgjfLWUQTnoZ1U=
 | 
			
		||||
golang.org/x/mod v0.28.0/go.mod h1:yfB/L0NOf/kmEbXjzCPOx1iK1fRutOydrCMsqRhEBxI=
 | 
			
		||||
golang.org/x/net v0.45.0 h1:RLBg5JKixCy82FtLJpeNlVM0nrSqpCRYzVU1n8kj0tM=
 | 
			
		||||
golang.org/x/net v0.45.0/go.mod h1:ECOoLqd5U3Lhyeyo/QDCEVQ4sNgYsqvCZ722XogGieY=
 | 
			
		||||
golang.org/x/oauth2 v0.32.0 h1:jsCblLleRMDrxMN29H3z/k1KliIvpLgCkE6R8FXXNgY=
 | 
			
		||||
golang.org/x/oauth2 v0.32.0/go.mod h1:lzm5WQJQwKZ3nwavOZ3IS5Aulzxi68dUSgRHujetwEA=
 | 
			
		||||
golang.org/x/sync v0.17.0 h1:l60nONMj9l5drqw6jlhIELNv9I0A4OFgRsG9k2oT9Ug=
 | 
			
		||||
golang.org/x/sync v0.17.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
 | 
			
		||||
golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.36.0 h1:KVRy2GtZBrk1cBYA7MKu5bEZFxQk4NIDV6RLVcC8o0k=
 | 
			
		||||
golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
 | 
			
		||||
golang.org/x/term v0.35.0 h1:bZBVKBudEyhRcajGcNc3jIfWPqV4y/Kt2XcoigOWtDQ=
 | 
			
		||||
golang.org/x/term v0.35.0/go.mod h1:TPGtkTLesOwf2DE8CgVYiZinHAOuy5AYUYT1lENIZnA=
 | 
			
		||||
golang.org/x/text v0.29.0 h1:1neNs90w9YzJ9BocxfsQNHKuAT4pkghyXc4nhZ6sJvk=
 | 
			
		||||
golang.org/x/text v0.29.0/go.mod h1:7MhJOA9CD2qZyOKYazxdYMF85OwPdEr9jTtBpO7ydH4=
 | 
			
		||||
golang.org/x/sys v0.37.0 h1:fdNQudmxPjkdUTPnLn5mdQv7Zwvbvpaxqs831goi9kQ=
 | 
			
		||||
golang.org/x/sys v0.37.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
 | 
			
		||||
golang.org/x/term v0.36.0 h1:zMPR+aF8gfksFprF/Nc/rd1wRS1EI6nDBGyWAvDzx2Q=
 | 
			
		||||
golang.org/x/term v0.36.0/go.mod h1:Qu394IJq6V6dCBRgwqshf3mPF85AqzYEzofzRdZkWss=
 | 
			
		||||
golang.org/x/text v0.30.0 h1:yznKA/E9zq54KzlzBEAWn1NXSQ8DIp/NYMy88xJjl4k=
 | 
			
		||||
golang.org/x/text v0.30.0/go.mod h1:yDdHFIX9t+tORqspjENWgzaCVXgk0yYnYuSZ8UzzBVM=
 | 
			
		||||
golang.org/x/time v0.5.0 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk=
 | 
			
		||||
golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM=
 | 
			
		||||
golang.org/x/tools v0.36.0 h1:kWS0uv/zsvHEle1LbV5LE8QujrxB3wfQyxHfhOk0Qkg=
 | 
			
		||||
golang.org/x/tools v0.36.0/go.mod h1:WBDiHKJK8YgLHlcQPYQzNCkUxUypCaa5ZegCVutKm+s=
 | 
			
		||||
golang.org/x/tools v0.37.0 h1:DVSRzp7FwePZW356yEAChSdNcQo6Nsp+fex1SUW09lE=
 | 
			
		||||
golang.org/x/tools v0.37.0/go.mod h1:MBN5QPQtLMHVdvsbtarmTNukZDdgwdwlO5qGacAzF0w=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
google.golang.org/genproto v0.0.0-20240213162025-012b6fc9bca9 h1:9+tzLLstTlPTRyJTh+ah5wIMsBW5c4tQwGTN3thOW9Y=
 | 
			
		||||
google.golang.org/genproto/googleapis/api v0.0.0-20250115164207-1a7da9e5054f h1:gap6+3Gk41EItBuyi4XX/bp4oqJ3UwuIMl25yGinuAA=
 | 
			
		||||
 
 | 
			
		||||
@@ -2,21 +2,25 @@ package bootstrap
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"bytes"
 | 
			
		||||
	"context"
 | 
			
		||||
	"encoding/json"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"net/url"
 | 
			
		||||
	"os"
 | 
			
		||||
	"sort"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"time"
 | 
			
		||||
	"tinyauth/internal/config"
 | 
			
		||||
	"tinyauth/internal/controller"
 | 
			
		||||
	"tinyauth/internal/middleware"
 | 
			
		||||
	"tinyauth/internal/model"
 | 
			
		||||
	"tinyauth/internal/service"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
	"github.com/gin-gonic/gin"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"gorm.io/gorm"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type Controller interface {
 | 
			
		||||
@@ -74,6 +78,15 @@ func (app *BootstrapApp) Setup() error {
 | 
			
		||||
	csrfCookieName := fmt.Sprintf("%s-%s", config.CSRFCookieName, cookieId)
 | 
			
		||||
	redirectCookieName := fmt.Sprintf("%s-%s", config.RedirectCookieName, cookieId)
 | 
			
		||||
 | 
			
		||||
	// Dumps
 | 
			
		||||
	log.Trace().Interface("config", app.config).Msg("Config dump")
 | 
			
		||||
	log.Trace().Interface("users", users).Msg("Users dump")
 | 
			
		||||
	log.Trace().Interface("oauthProviders", oauthProviders).Msg("OAuth providers dump")
 | 
			
		||||
	log.Trace().Str("cookieDomain", cookieDomain).Msg("Cookie domain")
 | 
			
		||||
	log.Trace().Str("sessionCookieName", sessionCookieName).Msg("Session cookie name")
 | 
			
		||||
	log.Trace().Str("csrfCookieName", csrfCookieName).Msg("CSRF cookie name")
 | 
			
		||||
	log.Trace().Str("redirectCookieName", redirectCookieName).Msg("Redirect cookie name")
 | 
			
		||||
 | 
			
		||||
	// Create configs
 | 
			
		||||
	authConfig := service.AuthServiceConfig{
 | 
			
		||||
		Users:             users,
 | 
			
		||||
@@ -126,12 +139,14 @@ func (app *BootstrapApp) Setup() error {
 | 
			
		||||
 | 
			
		||||
	// Create services
 | 
			
		||||
	dockerService := service.NewDockerService()
 | 
			
		||||
	aclsService := service.NewAccessControlsService(dockerService)
 | 
			
		||||
	authService := service.NewAuthService(authConfig, dockerService, ldapService, database)
 | 
			
		||||
	oauthBrokerService := service.NewOAuthBrokerService(oauthProviders)
 | 
			
		||||
 | 
			
		||||
	// Initialize services
 | 
			
		||||
	// Initialize services (order matters)
 | 
			
		||||
	services := []Service{
 | 
			
		||||
		dockerService,
 | 
			
		||||
		aclsService,
 | 
			
		||||
		authService,
 | 
			
		||||
		oauthBrokerService,
 | 
			
		||||
	}
 | 
			
		||||
@@ -150,18 +165,6 @@ func (app *BootstrapApp) Setup() error {
 | 
			
		||||
	configuredProviders := make([]controller.Provider, 0)
 | 
			
		||||
 | 
			
		||||
	for id, provider := range oauthProviders {
 | 
			
		||||
		if id == "" {
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if provider.Name == "" {
 | 
			
		||||
			if name, ok := config.OverrideProviders[id]; ok {
 | 
			
		||||
				provider.Name = name
 | 
			
		||||
			} else {
 | 
			
		||||
				provider.Name = utils.Capitalize(id)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		configuredProviders = append(configuredProviders, controller.Provider{
 | 
			
		||||
			Name:  provider.Name,
 | 
			
		||||
			ID:    id,
 | 
			
		||||
@@ -169,6 +172,10 @@ func (app *BootstrapApp) Setup() error {
 | 
			
		||||
		})
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	sort.Slice(configuredProviders, func(i, j int) bool {
 | 
			
		||||
		return configuredProviders[i].Name < configuredProviders[j].Name
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	if authService.UserAuthConfigured() || ldapService != nil {
 | 
			
		||||
		configuredProviders = append(configuredProviders, controller.Provider{
 | 
			
		||||
			Name:  "Username",
 | 
			
		||||
@@ -184,11 +191,8 @@ func (app *BootstrapApp) Setup() error {
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create engine
 | 
			
		||||
	if config.Version != "development" {
 | 
			
		||||
		gin.SetMode(gin.ReleaseMode)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	engine := gin.New()
 | 
			
		||||
	engine.Use(gin.Recovery())
 | 
			
		||||
 | 
			
		||||
	if len(app.config.TrustedProxies) > 0 {
 | 
			
		||||
		err := engine.SetTrustedProxies(strings.Split(app.config.TrustedProxies, ","))
 | 
			
		||||
@@ -244,7 +248,7 @@ func (app *BootstrapApp) Setup() error {
 | 
			
		||||
 | 
			
		||||
	proxyController := controller.NewProxyController(controller.ProxyControllerConfig{
 | 
			
		||||
		AppURL: app.config.AppURL,
 | 
			
		||||
	}, apiRouter, dockerService, authService)
 | 
			
		||||
	}, apiRouter, aclsService, authService)
 | 
			
		||||
 | 
			
		||||
	userController := controller.NewUserController(controller.UserControllerConfig{
 | 
			
		||||
		CookieDomain: cookieDomain,
 | 
			
		||||
@@ -278,6 +282,10 @@ func (app *BootstrapApp) Setup() error {
 | 
			
		||||
		go app.heartbeat()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Start DB cleanup routine
 | 
			
		||||
	log.Debug().Msg("Starting database cleanup routine")
 | 
			
		||||
	go app.dbCleanup(database)
 | 
			
		||||
 | 
			
		||||
	// Start server
 | 
			
		||||
	address := fmt.Sprintf("%s:%d", app.config.Address, app.config.Port)
 | 
			
		||||
	log.Info().Msgf("Starting server on %s", address)
 | 
			
		||||
@@ -339,3 +347,17 @@ func (app *BootstrapApp) heartbeat() {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (app *BootstrapApp) dbCleanup(db *gorm.DB) {
 | 
			
		||||
	ticker := time.NewTicker(time.Duration(30) * time.Minute)
 | 
			
		||||
	defer ticker.Stop()
 | 
			
		||||
	ctx := context.Background()
 | 
			
		||||
 | 
			
		||||
	for ; true; <-ticker.C {
 | 
			
		||||
		log.Debug().Msg("Cleaning up old database sessions")
 | 
			
		||||
		_, err := gorm.G[model.Session](db).Where("expiry < ?", time.Now().UnixMilli()).Delete(ctx)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Error().Err(err).Msg("Failed to cleanup old sessions")
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -53,16 +53,16 @@ type Claims struct {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type OAuthServiceConfig struct {
 | 
			
		||||
	ClientID           string   `key:"client-id"`
 | 
			
		||||
	ClientSecret       string   `key:"client-secret"`
 | 
			
		||||
	ClientSecretFile   string   `key:"client-secret-file"`
 | 
			
		||||
	Scopes             []string `key:"scopes"`
 | 
			
		||||
	RedirectURL        string   `key:"redirect-url"`
 | 
			
		||||
	AuthURL            string   `key:"auth-url"`
 | 
			
		||||
	TokenURL           string   `key:"token-url"`
 | 
			
		||||
	UserinfoURL        string   `key:"user-info-url"`
 | 
			
		||||
	InsecureSkipVerify bool     `key:"insecure-skip-verify"`
 | 
			
		||||
	Name               string   `key:"name"`
 | 
			
		||||
	ClientID           string `field:"client-id"`
 | 
			
		||||
	ClientSecret       string
 | 
			
		||||
	ClientSecretFile   string
 | 
			
		||||
	Scopes             []string
 | 
			
		||||
	RedirectURL        string `field:"redirect-url"`
 | 
			
		||||
	AuthURL            string `field:"auth-url"`
 | 
			
		||||
	TokenURL           string `field:"token-url"`
 | 
			
		||||
	UserinfoURL        string `field:"user-info-url"`
 | 
			
		||||
	InsecureSkipVerify bool
 | 
			
		||||
	Name               string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
var OverrideProviders = map[string]string{
 | 
			
		||||
 
 | 
			
		||||
@@ -13,8 +13,8 @@ func NewHealthController(router *gin.RouterGroup) *HealthController {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (controller *HealthController) SetupRoutes() {
 | 
			
		||||
	controller.router.GET("/health", controller.healthHandler)
 | 
			
		||||
	controller.router.HEAD("/health", controller.healthHandler)
 | 
			
		||||
	controller.router.GET("/healthz", controller.healthHandler)
 | 
			
		||||
	controller.router.HEAD("/healthz", controller.healthHandler)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (controller *HealthController) healthHandler(c *gin.Context) {
 | 
			
		||||
 
 | 
			
		||||
@@ -72,6 +72,7 @@ func (controller *OAuthController) oauthURLHandler(c *gin.Context) {
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	service.GenerateVerifier()
 | 
			
		||||
	state := service.GenerateState()
 | 
			
		||||
	authURL := service.GetAuthURL(state)
 | 
			
		||||
	c.SetCookie(controller.config.CSRFCookieName, state, int(time.Hour.Seconds()), "/", fmt.Sprintf(".%s", controller.config.CookieDomain), controller.config.SecureCookie, true)
 | 
			
		||||
@@ -162,7 +163,7 @@ func (controller *OAuthController) oauthCallbackHandler(c *gin.Context) {
 | 
			
		||||
 | 
			
		||||
	var name string
 | 
			
		||||
 | 
			
		||||
	if user.Name != "" {
 | 
			
		||||
	if strings.TrimSpace(user.Name) != "" {
 | 
			
		||||
		log.Debug().Msg("Using name from OAuth provider")
 | 
			
		||||
		name = user.Name
 | 
			
		||||
	} else {
 | 
			
		||||
@@ -172,7 +173,7 @@ func (controller *OAuthController) oauthCallbackHandler(c *gin.Context) {
 | 
			
		||||
 | 
			
		||||
	var username string
 | 
			
		||||
 | 
			
		||||
	if user.PreferredUsername != "" {
 | 
			
		||||
	if strings.TrimSpace(user.PreferredUsername) != "" {
 | 
			
		||||
		log.Debug().Msg("Using preferred username from OAuth provider")
 | 
			
		||||
		username = user.PreferredUsername
 | 
			
		||||
	} else {
 | 
			
		||||
 
 | 
			
		||||
@@ -24,15 +24,15 @@ type ProxyControllerConfig struct {
 | 
			
		||||
type ProxyController struct {
 | 
			
		||||
	config ProxyControllerConfig
 | 
			
		||||
	router *gin.RouterGroup
 | 
			
		||||
	docker *service.DockerService
 | 
			
		||||
	acls   *service.AccessControlsService
 | 
			
		||||
	auth   *service.AuthService
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewProxyController(config ProxyControllerConfig, router *gin.RouterGroup, docker *service.DockerService, auth *service.AuthService) *ProxyController {
 | 
			
		||||
func NewProxyController(config ProxyControllerConfig, router *gin.RouterGroup, acls *service.AccessControlsService, auth *service.AuthService) *ProxyController {
 | 
			
		||||
	return &ProxyController{
 | 
			
		||||
		config: config,
 | 
			
		||||
		router: router,
 | 
			
		||||
		docker: docker,
 | 
			
		||||
		acls:   acls,
 | 
			
		||||
		auth:   auth,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
@@ -76,18 +76,21 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
	proto := c.Request.Header.Get("X-Forwarded-Proto")
 | 
			
		||||
	host := c.Request.Header.Get("X-Forwarded-Host")
 | 
			
		||||
 | 
			
		||||
	labels, err := controller.docker.GetLabels(host)
 | 
			
		||||
	// Get acls
 | 
			
		||||
	acls, err := controller.acls.GetAccessControls(host)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Error().Err(err).Msg("Failed to get labels from Docker")
 | 
			
		||||
		log.Error().Err(err).Msg("Failed to get access controls for resource")
 | 
			
		||||
		controller.handleError(c, req, isBrowser)
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Trace().Interface("acls", acls).Msg("ACLs for resource")
 | 
			
		||||
 | 
			
		||||
	clientIP := c.ClientIP()
 | 
			
		||||
 | 
			
		||||
	if controller.auth.IsBypassedIP(labels.IP, clientIP) {
 | 
			
		||||
		controller.setHeaders(c, labels)
 | 
			
		||||
	if controller.auth.IsBypassedIP(acls.IP, clientIP) {
 | 
			
		||||
		controller.setHeaders(c, acls)
 | 
			
		||||
		c.JSON(200, gin.H{
 | 
			
		||||
			"status":  200,
 | 
			
		||||
			"message": "Authenticated",
 | 
			
		||||
@@ -95,7 +98,7 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	authEnabled, err := controller.auth.IsAuthEnabled(uri, labels.Path)
 | 
			
		||||
	authEnabled, err := controller.auth.IsAuthEnabled(uri, acls.Path)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Error().Err(err).Msg("Failed to check if auth is enabled for resource")
 | 
			
		||||
@@ -105,7 +108,7 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
 | 
			
		||||
	if !authEnabled {
 | 
			
		||||
		log.Debug().Msg("Authentication disabled for resource, allowing access")
 | 
			
		||||
		controller.setHeaders(c, labels)
 | 
			
		||||
		controller.setHeaders(c, acls)
 | 
			
		||||
		c.JSON(200, gin.H{
 | 
			
		||||
			"status":  200,
 | 
			
		||||
			"message": "Authenticated",
 | 
			
		||||
@@ -113,7 +116,7 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if !controller.auth.CheckIP(labels.IP, clientIP) {
 | 
			
		||||
	if !controller.auth.CheckIP(acls.IP, clientIP) {
 | 
			
		||||
		if req.Proxy == "nginx" || !isBrowser {
 | 
			
		||||
			c.JSON(401, gin.H{
 | 
			
		||||
				"status":  401,
 | 
			
		||||
@@ -150,13 +153,15 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
		userContext = context
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Trace().Interface("context", userContext).Msg("User context from request")
 | 
			
		||||
 | 
			
		||||
	if userContext.Provider == "basic" && userContext.TotpEnabled {
 | 
			
		||||
		log.Debug().Msg("User has TOTP enabled, denying basic auth access")
 | 
			
		||||
		userContext.IsLoggedIn = false
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if userContext.IsLoggedIn {
 | 
			
		||||
		appAllowed := controller.auth.IsResourceAllowed(c, userContext, labels)
 | 
			
		||||
		appAllowed := controller.auth.IsResourceAllowed(c, userContext, acls)
 | 
			
		||||
 | 
			
		||||
		if !appAllowed {
 | 
			
		||||
			log.Warn().Str("user", userContext.Username).Str("resource", strings.Split(host, ".")[0]).Msg("User not allowed to access resource")
 | 
			
		||||
@@ -190,7 +195,7 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if userContext.OAuth {
 | 
			
		||||
			groupOK := controller.auth.IsInOAuthGroup(c, userContext, labels.OAuth.Groups)
 | 
			
		||||
			groupOK := controller.auth.IsInOAuthGroup(c, userContext, acls.OAuth.Groups)
 | 
			
		||||
 | 
			
		||||
			if !groupOK {
 | 
			
		||||
				log.Warn().Str("user", userContext.Username).Str("resource", strings.Split(host, ".")[0]).Msg("User OAuth groups do not match resource requirements")
 | 
			
		||||
@@ -230,7 +235,7 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
		c.Header("Remote-Email", utils.SanitizeHeader(userContext.Email))
 | 
			
		||||
		c.Header("Remote-Groups", utils.SanitizeHeader(userContext.OAuthGroups))
 | 
			
		||||
 | 
			
		||||
		controller.setHeaders(c, labels)
 | 
			
		||||
		controller.setHeaders(c, acls)
 | 
			
		||||
 | 
			
		||||
		c.JSON(200, gin.H{
 | 
			
		||||
			"status":  200,
 | 
			
		||||
@@ -260,21 +265,21 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) {
 | 
			
		||||
	c.Redirect(http.StatusTemporaryRedirect, fmt.Sprintf("%s/login?%s", controller.config.AppURL, queries.Encode()))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (controller *ProxyController) setHeaders(c *gin.Context, labels config.App) {
 | 
			
		||||
func (controller *ProxyController) setHeaders(c *gin.Context, acls config.App) {
 | 
			
		||||
	c.Header("Authorization", c.Request.Header.Get("Authorization"))
 | 
			
		||||
 | 
			
		||||
	headers := utils.ParseHeaders(labels.Response.Headers)
 | 
			
		||||
	headers := utils.ParseHeaders(acls.Response.Headers)
 | 
			
		||||
 | 
			
		||||
	for key, value := range headers {
 | 
			
		||||
		log.Debug().Str("header", key).Msg("Setting header")
 | 
			
		||||
		c.Header(key, value)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	basicPassword := utils.GetSecret(labels.Response.BasicAuth.Password, labels.Response.BasicAuth.PasswordFile)
 | 
			
		||||
	basicPassword := utils.GetSecret(acls.Response.BasicAuth.Password, acls.Response.BasicAuth.PasswordFile)
 | 
			
		||||
 | 
			
		||||
	if labels.Response.BasicAuth.Username != "" && basicPassword != "" {
 | 
			
		||||
		log.Debug().Str("username", labels.Response.BasicAuth.Username).Msg("Setting basic auth header")
 | 
			
		||||
		c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Response.BasicAuth.Username, basicPassword)))
 | 
			
		||||
	if acls.Response.BasicAuth.Username != "" && basicPassword != "" {
 | 
			
		||||
		log.Debug().Str("username", acls.Response.BasicAuth.Username).Msg("Setting basic auth header")
 | 
			
		||||
		c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(acls.Response.BasicAuth.Username, basicPassword)))
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -39,6 +39,11 @@ func setupProxyController(t *testing.T, middlewares *[]gin.HandlerFunc) (*gin.En
 | 
			
		||||
 | 
			
		||||
	assert.NilError(t, dockerService.Init())
 | 
			
		||||
 | 
			
		||||
	// Access controls
 | 
			
		||||
	accessControlsService := service.NewAccessControlsService(dockerService)
 | 
			
		||||
 | 
			
		||||
	assert.NilError(t, accessControlsService.Init())
 | 
			
		||||
 | 
			
		||||
	// Auth service
 | 
			
		||||
	authService := service.NewAuthService(service.AuthServiceConfig{
 | 
			
		||||
		Users: []config.User{
 | 
			
		||||
@@ -59,7 +64,7 @@ func setupProxyController(t *testing.T, middlewares *[]gin.HandlerFunc) (*gin.En
 | 
			
		||||
	// Controller
 | 
			
		||||
	ctrl := controller.NewProxyController(controller.ProxyControllerConfig{
 | 
			
		||||
		AppURL: "http://localhost:8080",
 | 
			
		||||
	}, group, dockerService, authService)
 | 
			
		||||
	}, group, accessControlsService, authService)
 | 
			
		||||
	ctrl.SetupRoutes()
 | 
			
		||||
 | 
			
		||||
	return router, recorder, authService
 | 
			
		||||
 
 | 
			
		||||
@@ -1,10 +1,12 @@
 | 
			
		||||
package middleware
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"io/fs"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"time"
 | 
			
		||||
	"tinyauth/internal/assets"
 | 
			
		||||
 | 
			
		||||
	"github.com/gin-gonic/gin"
 | 
			
		||||
@@ -27,14 +29,16 @@ func (m *UIMiddleware) Init() error {
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	m.uiFs = ui
 | 
			
		||||
	m.uiFileServer = http.FileServer(http.FS(ui))
 | 
			
		||||
	m.uiFileServer = http.FileServerFS(ui)
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (m *UIMiddleware) Middleware() gin.HandlerFunc {
 | 
			
		||||
	return func(c *gin.Context) {
 | 
			
		||||
		switch strings.Split(c.Request.URL.Path, "/")[1] {
 | 
			
		||||
		path := strings.TrimPrefix(c.Request.URL.Path, "/")
 | 
			
		||||
 | 
			
		||||
		switch strings.SplitN(path, "/", 2)[0] {
 | 
			
		||||
		case "api":
 | 
			
		||||
			c.Next()
 | 
			
		||||
			return
 | 
			
		||||
@@ -42,12 +46,19 @@ func (m *UIMiddleware) Middleware() gin.HandlerFunc {
 | 
			
		||||
			c.Next()
 | 
			
		||||
			return
 | 
			
		||||
		default:
 | 
			
		||||
			_, err := fs.Stat(m.uiFs, strings.TrimPrefix(c.Request.URL.Path, "/"))
 | 
			
		||||
			_, err := fs.Stat(m.uiFs, path)
 | 
			
		||||
 | 
			
		||||
			// Enough for one authentication flow
 | 
			
		||||
			maxAge := 15 * time.Minute
 | 
			
		||||
 | 
			
		||||
			if os.IsNotExist(err) {
 | 
			
		||||
				c.Request.URL.Path = "/"
 | 
			
		||||
			} else if strings.HasPrefix(path, "assets/") {
 | 
			
		||||
				// assets are named with a hash and can be cached for a long time
 | 
			
		||||
				maxAge = 30 * 24 * time.Hour
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			c.Writer.Header().Set("Cache-Control", fmt.Sprintf("public, max-age=%d", int(maxAge.Seconds())))
 | 
			
		||||
			m.uiFileServer.ServeHTTP(c.Writer, c.Request)
 | 
			
		||||
			c.Abort()
 | 
			
		||||
			return
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										103
									
								
								internal/service/access_controls_service.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								internal/service/access_controls_service.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,103 @@
 | 
			
		||||
package service
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"tinyauth/internal/config"
 | 
			
		||||
	"tinyauth/internal/utils/decoders"
 | 
			
		||||
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type AccessControlsService struct {
 | 
			
		||||
	docker  *DockerService
 | 
			
		||||
	envACLs config.Apps
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewAccessControlsService(docker *DockerService) *AccessControlsService {
 | 
			
		||||
	return &AccessControlsService{
 | 
			
		||||
		docker: docker,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (acls *AccessControlsService) Init() error {
 | 
			
		||||
	acls.envACLs = config.Apps{}
 | 
			
		||||
	env := os.Environ()
 | 
			
		||||
	appEnvVars := []string{}
 | 
			
		||||
 | 
			
		||||
	for _, e := range env {
 | 
			
		||||
		if strings.HasPrefix(e, "TINYAUTH_APPS_") {
 | 
			
		||||
			appEnvVars = append(appEnvVars, e)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err := acls.loadEnvACLs(appEnvVars)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (acls *AccessControlsService) loadEnvACLs(appEnvVars []string) error {
 | 
			
		||||
	if len(appEnvVars) == 0 {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	envAcls := map[string]string{}
 | 
			
		||||
 | 
			
		||||
	for _, e := range appEnvVars {
 | 
			
		||||
		parts := strings.SplitN(e, "=", 2)
 | 
			
		||||
		if len(parts) != 2 {
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Normalize key, this should use the same normalization logic as in utils/decoders/decoders.go
 | 
			
		||||
		key := parts[0]
 | 
			
		||||
		key = strings.ToLower(key)
 | 
			
		||||
		key = strings.ReplaceAll(key, "_", ".")
 | 
			
		||||
		value := parts[1]
 | 
			
		||||
		envAcls[key] = value
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	apps, err := decoders.DecodeLabels(envAcls)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	acls.envACLs = apps
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (acls *AccessControlsService) lookupEnvACLs(appDomain string) *config.App {
 | 
			
		||||
	if len(acls.envACLs.Apps) == 0 {
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for appName, appACLs := range acls.envACLs.Apps {
 | 
			
		||||
		if appACLs.Config.Domain == appDomain {
 | 
			
		||||
			return &appACLs
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if strings.SplitN(appDomain, ".", 2)[0] == appName {
 | 
			
		||||
			return &appACLs
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (acls *AccessControlsService) GetAccessControls(appDomain string) (config.App, error) {
 | 
			
		||||
	// First check environment variables
 | 
			
		||||
	envACLs := acls.lookupEnvACLs(appDomain)
 | 
			
		||||
 | 
			
		||||
	if envACLs != nil {
 | 
			
		||||
		log.Debug().Str("domain", appDomain).Msg("Found matching access controls in environment variables")
 | 
			
		||||
		return *envACLs, nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Fallback to Docker labels
 | 
			
		||||
	return acls.docker.GetLabels(appDomain)
 | 
			
		||||
}
 | 
			
		||||
@@ -1,6 +1,8 @@
 | 
			
		||||
package service
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"regexp"
 | 
			
		||||
	"strings"
 | 
			
		||||
@@ -41,6 +43,7 @@ type AuthService struct {
 | 
			
		||||
	loginMutex    sync.RWMutex
 | 
			
		||||
	ldap          *LdapService
 | 
			
		||||
	database      *gorm.DB
 | 
			
		||||
	ctx           context.Context
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewAuthService(config AuthServiceConfig, docker *DockerService, ldap *LdapService, database *gorm.DB) *AuthService {
 | 
			
		||||
@@ -54,6 +57,7 @@ func NewAuthService(config AuthServiceConfig, docker *DockerService, ldap *LdapS
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *AuthService) Init() error {
 | 
			
		||||
	auth.ctx = context.Background()
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -213,7 +217,7 @@ func (auth *AuthService) CreateSessionCookie(c *gin.Context, data *config.Sessio
 | 
			
		||||
		OAuthName:   data.OAuthName,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	err = auth.database.Create(&session).Error
 | 
			
		||||
	err = gorm.G[model.Session](auth.database).Create(auth.ctx, &session)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
@@ -231,10 +235,10 @@ func (auth *AuthService) DeleteSessionCookie(c *gin.Context) error {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	res := auth.database.Unscoped().Where("uuid = ?", cookie).Delete(&model.Session{})
 | 
			
		||||
	_, err = gorm.G[model.Session](auth.database).Where("uuid = ?", cookie).Delete(auth.ctx)
 | 
			
		||||
 | 
			
		||||
	if res.Error != nil {
 | 
			
		||||
		return res.Error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	c.SetCookie(auth.config.SessionCookieName, "", -1, "/", fmt.Sprintf(".%s", auth.config.CookieDomain), auth.config.SecureCookie, true)
 | 
			
		||||
@@ -249,15 +253,13 @@ func (auth *AuthService) GetSessionCookie(c *gin.Context) (config.SessionCookie,
 | 
			
		||||
		return config.SessionCookie{}, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var session model.Session
 | 
			
		||||
	session, err := gorm.G[model.Session](auth.database).Where("uuid = ?", cookie).First(auth.ctx)
 | 
			
		||||
 | 
			
		||||
	res := auth.database.Unscoped().Where("uuid = ?", cookie).First(&session)
 | 
			
		||||
 | 
			
		||||
	if res.Error != nil {
 | 
			
		||||
		return config.SessionCookie{}, res.Error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return config.SessionCookie{}, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if res.RowsAffected == 0 {
 | 
			
		||||
	if errors.Is(err, gorm.ErrRecordNotFound) {
 | 
			
		||||
		return config.SessionCookie{}, fmt.Errorf("session not found")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
@@ -287,21 +289,21 @@ func (auth *AuthService) UserAuthConfigured() bool {
 | 
			
		||||
	return len(auth.config.Users) > 0 || auth.ldap != nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *AuthService) IsResourceAllowed(c *gin.Context, context config.UserContext, labels config.App) bool {
 | 
			
		||||
func (auth *AuthService) IsResourceAllowed(c *gin.Context, context config.UserContext, acls config.App) bool {
 | 
			
		||||
	if context.OAuth {
 | 
			
		||||
		log.Debug().Msg("Checking OAuth whitelist")
 | 
			
		||||
		return utils.CheckFilter(labels.OAuth.Whitelist, context.Email)
 | 
			
		||||
		return utils.CheckFilter(acls.OAuth.Whitelist, context.Email)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if labels.Users.Block != "" {
 | 
			
		||||
	if acls.Users.Block != "" {
 | 
			
		||||
		log.Debug().Msg("Checking blocked users")
 | 
			
		||||
		if utils.CheckFilter(labels.Users.Block, context.Username) {
 | 
			
		||||
		if utils.CheckFilter(acls.Users.Block, context.Username) {
 | 
			
		||||
			return false
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Debug().Msg("Checking users")
 | 
			
		||||
	return utils.CheckFilter(labels.Users.Allow, context.Username)
 | 
			
		||||
	return utils.CheckFilter(acls.Users.Allow, context.Username)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *AuthService) IsInOAuthGroup(c *gin.Context, context config.UserContext, requiredGroups string) bool {
 | 
			
		||||
@@ -318,6 +320,7 @@ func (auth *AuthService) IsInOAuthGroup(c *gin.Context, context config.UserConte
 | 
			
		||||
 | 
			
		||||
	for userGroup := range strings.SplitSeq(context.OAuthGroups, ",") {
 | 
			
		||||
		if utils.CheckFilter(requiredGroups, strings.TrimSpace(userGroup)) {
 | 
			
		||||
			log.Trace().Str("group", userGroup).Str("required", requiredGroups).Msg("User group matched")
 | 
			
		||||
			return true
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
@@ -368,8 +371,8 @@ func (auth *AuthService) GetBasicAuth(c *gin.Context) *config.User {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *AuthService) CheckIP(labels config.AppIP, ip string) bool {
 | 
			
		||||
	for _, blocked := range labels.Block {
 | 
			
		||||
func (auth *AuthService) CheckIP(acls config.AppIP, ip string) bool {
 | 
			
		||||
	for _, blocked := range acls.Block {
 | 
			
		||||
		res, err := utils.FilterIP(blocked, ip)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Warn().Err(err).Str("item", blocked).Msg("Invalid IP/CIDR in block list")
 | 
			
		||||
@@ -381,7 +384,7 @@ func (auth *AuthService) CheckIP(labels config.AppIP, ip string) bool {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for _, allowed := range labels.Allow {
 | 
			
		||||
	for _, allowed := range acls.Allow {
 | 
			
		||||
		res, err := utils.FilterIP(allowed, ip)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Warn().Err(err).Str("item", allowed).Msg("Invalid IP/CIDR in allow list")
 | 
			
		||||
@@ -393,7 +396,7 @@ func (auth *AuthService) CheckIP(labels config.AppIP, ip string) bool {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if len(labels.Allow) > 0 {
 | 
			
		||||
	if len(acls.Allow) > 0 {
 | 
			
		||||
		log.Debug().Str("ip", ip).Msg("IP not in allow list, denying access")
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
@@ -402,8 +405,8 @@ func (auth *AuthService) CheckIP(labels config.AppIP, ip string) bool {
 | 
			
		||||
	return true
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *AuthService) IsBypassedIP(labels config.AppIP, ip string) bool {
 | 
			
		||||
	for _, bypassed := range labels.Bypass {
 | 
			
		||||
func (auth *AuthService) IsBypassedIP(acls config.AppIP, ip string) bool {
 | 
			
		||||
	for _, bypassed := range acls.Bypass {
 | 
			
		||||
		res, err := utils.FilterIP(bypassed, ip)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Warn().Err(err).Str("item", bypassed).Msg("Invalid IP/CIDR in bypass list")
 | 
			
		||||
 
 | 
			
		||||
@@ -14,6 +14,7 @@ import (
 | 
			
		||||
type DockerService struct {
 | 
			
		||||
	client      *client.Client
 | 
			
		||||
	context     context.Context
 | 
			
		||||
	isConnected bool
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewDockerService() *DockerService {
 | 
			
		||||
@@ -31,10 +32,24 @@ func (docker *DockerService) Init() error {
 | 
			
		||||
 | 
			
		||||
	docker.client = client
 | 
			
		||||
	docker.context = ctx
 | 
			
		||||
 | 
			
		||||
	_, err = docker.client.Ping(docker.context)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Debug().Err(err).Msg("Docker not connected")
 | 
			
		||||
		docker.isConnected = false
 | 
			
		||||
		docker.client = nil
 | 
			
		||||
		docker.context = nil
 | 
			
		||||
		return nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	docker.isConnected = true
 | 
			
		||||
	log.Debug().Msg("Docker connected")
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (docker *DockerService) GetContainers() ([]container.Summary, error) {
 | 
			
		||||
func (docker *DockerService) getContainers() ([]container.Summary, error) {
 | 
			
		||||
	containers, err := docker.client.ContainerList(docker.context, container.ListOptions{})
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
@@ -42,7 +57,7 @@ func (docker *DockerService) GetContainers() ([]container.Summary, error) {
 | 
			
		||||
	return containers, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (docker *DockerService) InspectContainer(containerId string) (container.InspectResponse, error) {
 | 
			
		||||
func (docker *DockerService) inspectContainer(containerId string) (container.InspectResponse, error) {
 | 
			
		||||
	inspect, err := docker.client.ContainerInspect(docker.context, containerId)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return container.InspectResponse{}, err
 | 
			
		||||
@@ -50,45 +65,36 @@ func (docker *DockerService) InspectContainer(containerId string) (container.Ins
 | 
			
		||||
	return inspect, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (docker *DockerService) DockerConnected() bool {
 | 
			
		||||
	_, err := docker.client.Ping(docker.context)
 | 
			
		||||
	return err == nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (docker *DockerService) GetLabels(appDomain string) (config.App, error) {
 | 
			
		||||
	isConnected := docker.DockerConnected()
 | 
			
		||||
 | 
			
		||||
	if !isConnected {
 | 
			
		||||
	if !docker.isConnected {
 | 
			
		||||
		log.Debug().Msg("Docker not connected, returning empty labels")
 | 
			
		||||
		return config.App{}, nil
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	containers, err := docker.GetContainers()
 | 
			
		||||
	containers, err := docker.getContainers()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return config.App{}, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for _, ctr := range containers {
 | 
			
		||||
		inspect, err := docker.InspectContainer(ctr.ID)
 | 
			
		||||
		inspect, err := docker.inspectContainer(ctr.ID)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Warn().Str("id", ctr.ID).Err(err).Msg("Error inspecting container, skipping")
 | 
			
		||||
			continue
 | 
			
		||||
			return config.App{}, err
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		labels, err := decoders.DecodeLabels(inspect.Config.Labels)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Warn().Str("id", ctr.ID).Err(err).Msg("Error getting container labels, skipping")
 | 
			
		||||
			continue
 | 
			
		||||
			return config.App{}, err
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		for appName, appLabels := range labels.Apps {
 | 
			
		||||
			if appLabels.Config.Domain == appDomain {
 | 
			
		||||
				log.Debug().Str("id", inspect.ID).Msg("Found matching container by domain")
 | 
			
		||||
				log.Debug().Str("id", inspect.ID).Str("name", inspect.Name).Msg("Found matching container by domain")
 | 
			
		||||
				return appLabels, nil
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			if strings.TrimPrefix(inspect.Name, "/") == appName {
 | 
			
		||||
				log.Debug().Str("id", inspect.ID).Msg("Found matching container by app name")
 | 
			
		||||
			if strings.SplitN(appDomain, ".", 2)[0] == appName {
 | 
			
		||||
				log.Debug().Str("id", inspect.ID).Str("name", inspect.Name).Msg("Found matching container by app name")
 | 
			
		||||
				return appLabels, nil
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 
 | 
			
		||||
@@ -12,6 +12,7 @@ import (
 | 
			
		||||
	"time"
 | 
			
		||||
	"tinyauth/internal/config"
 | 
			
		||||
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"golang.org/x/oauth2"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
@@ -58,10 +59,8 @@ func (generic *GenericOAuthService) Init() error {
 | 
			
		||||
	ctx := context.Background()
 | 
			
		||||
 | 
			
		||||
	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient)
 | 
			
		||||
	verifier := oauth2.GenerateVerifier()
 | 
			
		||||
 | 
			
		||||
	generic.context = ctx
 | 
			
		||||
	generic.verifier = verifier
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -75,6 +74,12 @@ func (generic *GenericOAuthService) GenerateState() string {
 | 
			
		||||
	return state
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (generic *GenericOAuthService) GenerateVerifier() string {
 | 
			
		||||
	verifier := oauth2.GenerateVerifier()
 | 
			
		||||
	generic.verifier = verifier
 | 
			
		||||
	return verifier
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (generic *GenericOAuthService) GetAuthURL(state string) string {
 | 
			
		||||
	return generic.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(generic.verifier))
 | 
			
		||||
}
 | 
			
		||||
@@ -110,6 +115,8 @@ func (generic *GenericOAuthService) Userinfo() (config.Claims, error) {
 | 
			
		||||
		return user, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Trace().Str("body", string(body)).Msg("Userinfo response body")
 | 
			
		||||
 | 
			
		||||
	err = json.Unmarshal(body, &user)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return user, err
 | 
			
		||||
 
 | 
			
		||||
@@ -53,10 +53,7 @@ func (github *GithubOAuthService) Init() error {
 | 
			
		||||
	httpClient := &http.Client{}
 | 
			
		||||
	ctx := context.Background()
 | 
			
		||||
	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient)
 | 
			
		||||
	verifier := oauth2.GenerateVerifier()
 | 
			
		||||
 | 
			
		||||
	github.context = ctx
 | 
			
		||||
	github.verifier = verifier
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -70,6 +67,12 @@ func (github *GithubOAuthService) GenerateState() string {
 | 
			
		||||
	return state
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (github *GithubOAuthService) GenerateVerifier() string {
 | 
			
		||||
	verifier := oauth2.GenerateVerifier()
 | 
			
		||||
	github.verifier = verifier
 | 
			
		||||
	return verifier
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (github *GithubOAuthService) GetAuthURL(state string) string {
 | 
			
		||||
	return github.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(github.verifier))
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -48,10 +48,7 @@ func (google *GoogleOAuthService) Init() error {
 | 
			
		||||
	httpClient := &http.Client{}
 | 
			
		||||
	ctx := context.Background()
 | 
			
		||||
	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient)
 | 
			
		||||
	verifier := oauth2.GenerateVerifier()
 | 
			
		||||
 | 
			
		||||
	google.context = ctx
 | 
			
		||||
	google.verifier = verifier
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -65,6 +62,12 @@ func (oauth *GoogleOAuthService) GenerateState() string {
 | 
			
		||||
	return state
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (google *GoogleOAuthService) GenerateVerifier() string {
 | 
			
		||||
	verifier := oauth2.GenerateVerifier()
 | 
			
		||||
	google.verifier = verifier
 | 
			
		||||
	return verifier
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (google *GoogleOAuthService) GetAuthURL(state string) string {
 | 
			
		||||
	return google.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(google.verifier))
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -11,6 +11,7 @@ import (
 | 
			
		||||
type OAuthService interface {
 | 
			
		||||
	Init() error
 | 
			
		||||
	GenerateState() string
 | 
			
		||||
	GenerateVerifier() string
 | 
			
		||||
	GetAuthURL(state string) string
 | 
			
		||||
	VerifyCode(code string) error
 | 
			
		||||
	Userinfo() (config.Claims, error)
 | 
			
		||||
@@ -50,7 +51,7 @@ func (broker *OAuthBrokerService) Init() error {
 | 
			
		||||
			log.Error().Err(err).Msgf("Failed to initialize OAuth service: %T", name)
 | 
			
		||||
			return err
 | 
			
		||||
		}
 | 
			
		||||
		log.Info().Str("service", service.GetName()).Msg("Initialized OAuth service")
 | 
			
		||||
		log.Info().Str("service", name).Msg("Initialized OAuth service")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return nil
 | 
			
		||||
 
 | 
			
		||||
@@ -100,17 +100,17 @@ func IsRedirectSafe(redirectURL string, domain string) bool {
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cookieDomain, err := GetCookieDomain(redirectURL)
 | 
			
		||||
	host := parsedURL.Hostname()
 | 
			
		||||
	if host == domain {
 | 
			
		||||
		return true
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	cookieDomain, err := GetCookieDomain(redirectURL)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if cookieDomain != domain {
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return true
 | 
			
		||||
	return cookieDomain == domain
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func GetLogLevel(level string) zerolog.Level {
 | 
			
		||||
@@ -147,7 +147,7 @@ func GetOAuthProvidersConfig(env []string, args []string, appUrl string) (map[st
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	envProviders, err := decoders.DecodeEnv(envMap)
 | 
			
		||||
	envProviders, err := decoders.DecodeEnv[config.Providers, config.OAuthServiceConfig](envMap, "providers")
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
@@ -167,7 +167,7 @@ func GetOAuthProvidersConfig(env []string, args []string, appUrl string) (map[st
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	flagProviders, err := decoders.DecodeFlags(flagsMap)
 | 
			
		||||
	flagProviders, err := decoders.DecodeFlags[config.Providers, config.OAuthServiceConfig](flagsMap, "providers")
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
@@ -184,7 +184,6 @@ func GetOAuthProvidersConfig(env []string, args []string, appUrl string) (map[st
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If we have google/github providers and no redirect URL then set a default
 | 
			
		||||
 | 
			
		||||
	for id := range config.OverrideProviders {
 | 
			
		||||
		if provider, exists := providers[id]; exists {
 | 
			
		||||
			if provider.RedirectURL == "" {
 | 
			
		||||
@@ -194,6 +193,18 @@ func GetOAuthProvidersConfig(env []string, args []string, appUrl string) (map[st
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set names
 | 
			
		||||
	for id, provider := range providers {
 | 
			
		||||
		if provider.Name == "" {
 | 
			
		||||
			if name, ok := config.OverrideProviders[id]; ok {
 | 
			
		||||
				provider.Name = name
 | 
			
		||||
			} else {
 | 
			
		||||
				provider.Name = Capitalize(id)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		providers[id] = provider
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Return combined providers
 | 
			
		||||
	return providers, nil
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -164,7 +164,7 @@ func TestIsRedirectSafe(t *testing.T) {
 | 
			
		||||
	// Case with no subdomain
 | 
			
		||||
	redirectURL := "http://example.com/welcome"
 | 
			
		||||
	result := utils.IsRedirectSafe(redirectURL, domain)
 | 
			
		||||
	assert.Equal(t, false, result)
 | 
			
		||||
	assert.Equal(t, true, result)
 | 
			
		||||
 | 
			
		||||
	// Case with different domain
 | 
			
		||||
	redirectURL = "http://malicious.com/phishing"
 | 
			
		||||
@@ -202,6 +202,41 @@ func TestIsRedirectSafe(t *testing.T) {
 | 
			
		||||
	assert.Equal(t, false, result)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestIsRedirectSafeMultiLevel(t *testing.T) {
 | 
			
		||||
	// Setup
 | 
			
		||||
	cookieDomain := "tinyauth.example.com"
 | 
			
		||||
 | 
			
		||||
	// Case with 3rd level domain
 | 
			
		||||
	redirectURL := "http://tinyauth.example.com/welcome"
 | 
			
		||||
	result := utils.IsRedirectSafe(redirectURL, cookieDomain)
 | 
			
		||||
	assert.Equal(t, true, result)
 | 
			
		||||
 | 
			
		||||
	// Case with root domain
 | 
			
		||||
	redirectURL = "http://example.com/unsafe"
 | 
			
		||||
	result = utils.IsRedirectSafe(redirectURL, cookieDomain)
 | 
			
		||||
	assert.Equal(t, false, result)
 | 
			
		||||
 | 
			
		||||
	// Case with 4th level domain
 | 
			
		||||
	redirectURL = "http://auth.tinyauth.example.com/post-login"
 | 
			
		||||
	result = utils.IsRedirectSafe(redirectURL, cookieDomain)
 | 
			
		||||
	assert.Equal(t, true, result)
 | 
			
		||||
 | 
			
		||||
	// Case with 5th level domain (should be unsafe)
 | 
			
		||||
	redirectURL = "http://x.auth.tinyauth.example.com/deep"
 | 
			
		||||
	result = utils.IsRedirectSafe(redirectURL, cookieDomain)
 | 
			
		||||
	assert.Equal(t, false, result)
 | 
			
		||||
 | 
			
		||||
	// Case with different subdomain
 | 
			
		||||
	redirectURL = "http://auth.tinyauth.example.net/attack"
 | 
			
		||||
	result = utils.IsRedirectSafe(redirectURL, cookieDomain)
 | 
			
		||||
	assert.Equal(t, false, result)
 | 
			
		||||
 | 
			
		||||
	// Case with malformed URL
 | 
			
		||||
	redirectURL = "http://[::1]:namedport"
 | 
			
		||||
	result = utils.IsRedirectSafe(redirectURL, cookieDomain)
 | 
			
		||||
	assert.Equal(t, false, result)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestGetOAuthProvidersConfig(t *testing.T) {
 | 
			
		||||
	env := []string{"PROVIDERS_CLIENT1_CLIENT_ID=client1-id", "PROVIDERS_CLIENT1_CLIENT_SECRET=client1-secret"}
 | 
			
		||||
	args := []string{"/tinyauth/tinyauth", "--providers-client2-client-id=client2-id", "--providers-client2-client-secret=client2-secret"}
 | 
			
		||||
@@ -210,10 +245,12 @@ func TestGetOAuthProvidersConfig(t *testing.T) {
 | 
			
		||||
		"client1": {
 | 
			
		||||
			ClientID:     "client1-id",
 | 
			
		||||
			ClientSecret: "client1-secret",
 | 
			
		||||
			Name:         "Client1",
 | 
			
		||||
		},
 | 
			
		||||
		"client2": {
 | 
			
		||||
			ClientID:     "client2-id",
 | 
			
		||||
			ClientSecret: "client2-secret",
 | 
			
		||||
			Name:         "Client2",
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
@@ -247,6 +284,7 @@ func TestGetOAuthProvidersConfig(t *testing.T) {
 | 
			
		||||
		"client1": {
 | 
			
		||||
			ClientID:     "client1-id",
 | 
			
		||||
			ClientSecret: "file content",
 | 
			
		||||
			Name:         "Client1",
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
@@ -262,6 +300,7 @@ func TestGetOAuthProvidersConfig(t *testing.T) {
 | 
			
		||||
			ClientID:     "google-id",
 | 
			
		||||
			ClientSecret: "google-secret",
 | 
			
		||||
			RedirectURL:  "http://app.url/api/oauth/callback/google",
 | 
			
		||||
			Name:         "Google",
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -3,29 +3,24 @@ package decoders
 | 
			
		||||
import (
 | 
			
		||||
	"reflect"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"tinyauth/internal/config"
 | 
			
		||||
 | 
			
		||||
	"github.com/stoewer/go-strcase"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func NormalizeKeys(keys map[string]string, rootName string, sep string) map[string]string {
 | 
			
		||||
func normalizeKeys[T any](input map[string]string, root string, sep string) map[string]string {
 | 
			
		||||
	knownKeys := getKnownKeys[T]()
 | 
			
		||||
	normalized := make(map[string]string)
 | 
			
		||||
	knownKeys := getKnownKeys()
 | 
			
		||||
 | 
			
		||||
	for k, v := range keys {
 | 
			
		||||
		var finalKey []string
 | 
			
		||||
		var suffix string
 | 
			
		||||
		var camelClientName string
 | 
			
		||||
		var camelField string
 | 
			
		||||
	for k, v := range input {
 | 
			
		||||
		parts := []string{"tinyauth"}
 | 
			
		||||
 | 
			
		||||
		finalKey = append(finalKey, rootName)
 | 
			
		||||
		finalKey = append(finalKey, "providers")
 | 
			
		||||
		lowerKey := strings.ToLower(k)
 | 
			
		||||
		key := strings.ToLower(k)
 | 
			
		||||
		key = strings.ReplaceAll(key, sep, "-")
 | 
			
		||||
 | 
			
		||||
		if !strings.HasPrefix(lowerKey, "providers"+sep) {
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
		suffix := ""
 | 
			
		||||
 | 
			
		||||
		for _, known := range knownKeys {
 | 
			
		||||
			if strings.HasSuffix(lowerKey, strings.ReplaceAll(known, "-", sep)) {
 | 
			
		||||
			if strings.HasSuffix(key, known) {
 | 
			
		||||
				suffix = known
 | 
			
		||||
				break
 | 
			
		||||
			}
 | 
			
		||||
@@ -35,55 +30,47 @@ func NormalizeKeys(keys map[string]string, rootName string, sep string) map[stri
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if strings.TrimSpace(strings.TrimSuffix(strings.TrimPrefix(lowerKey, "providers"+sep), strings.ReplaceAll(suffix, "-", sep))) == "" {
 | 
			
		||||
		parts = append(parts, root)
 | 
			
		||||
 | 
			
		||||
		id := strings.TrimPrefix(key, root+"-")
 | 
			
		||||
		id = strings.TrimSuffix(id, "-"+suffix)
 | 
			
		||||
 | 
			
		||||
		if id == "" {
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		clientNameParts := strings.Split(strings.TrimPrefix(strings.TrimSuffix(lowerKey, sep+strings.ReplaceAll(suffix, "-", sep)), "providers"+sep), sep)
 | 
			
		||||
		parts = append(parts, id)
 | 
			
		||||
		parts = append(parts, suffix)
 | 
			
		||||
 | 
			
		||||
		for i, p := range clientNameParts {
 | 
			
		||||
			if i == 0 {
 | 
			
		||||
				camelClientName += p
 | 
			
		||||
				continue
 | 
			
		||||
		final := ""
 | 
			
		||||
 | 
			
		||||
		for i, part := range parts {
 | 
			
		||||
			if i > 0 {
 | 
			
		||||
				final += "."
 | 
			
		||||
			}
 | 
			
		||||
			if p == "" {
 | 
			
		||||
				continue
 | 
			
		||||
			}
 | 
			
		||||
			camelClientName += strings.ToUpper(string([]rune(p)[0])) + string([]rune(p)[1:])
 | 
			
		||||
			final += strcase.LowerCamelCase(part)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		finalKey = append(finalKey, camelClientName)
 | 
			
		||||
 | 
			
		||||
		fieldParts := strings.Split(suffix, "-")
 | 
			
		||||
 | 
			
		||||
		for i, p := range fieldParts {
 | 
			
		||||
			if i == 0 {
 | 
			
		||||
				camelField += p
 | 
			
		||||
				continue
 | 
			
		||||
			}
 | 
			
		||||
			if p == "" {
 | 
			
		||||
				continue
 | 
			
		||||
			}
 | 
			
		||||
			camelField += strings.ToUpper(string([]rune(p)[0])) + string([]rune(p)[1:])
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		finalKey = append(finalKey, camelField)
 | 
			
		||||
		normalized[strings.Join(finalKey, ".")] = v
 | 
			
		||||
		normalized[final] = v
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return normalized
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func getKnownKeys() []string {
 | 
			
		||||
	var known []string
 | 
			
		||||
func getKnownKeys[T any]() []string {
 | 
			
		||||
	var keys []string
 | 
			
		||||
	var t T
 | 
			
		||||
 | 
			
		||||
	p := config.OAuthServiceConfig{}
 | 
			
		||||
	v := reflect.ValueOf(p)
 | 
			
		||||
	typeOfP := v.Type()
 | 
			
		||||
	v := reflect.ValueOf(t)
 | 
			
		||||
	typeOfT := v.Type()
 | 
			
		||||
 | 
			
		||||
	for field := range typeOfP.NumField() {
 | 
			
		||||
		known = append(known, typeOfP.Field(field).Tag.Get("key"))
 | 
			
		||||
	for field := range typeOfT.NumField() {
 | 
			
		||||
		if typeOfT.Field(field).Tag.Get("field") != "" {
 | 
			
		||||
			keys = append(keys, typeOfT.Field(field).Tag.Get("field"))
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
		keys = append(keys, strcase.KebabCase(typeOfT.Field(field).Name))
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return known
 | 
			
		||||
	return keys
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,49 +0,0 @@
 | 
			
		||||
package decoders_test
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"testing"
 | 
			
		||||
	"tinyauth/internal/utils/decoders"
 | 
			
		||||
 | 
			
		||||
	"gotest.tools/v3/assert"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestNormalizeKeys(t *testing.T) {
 | 
			
		||||
	// Test with env
 | 
			
		||||
	test := map[string]string{
 | 
			
		||||
		"PROVIDERS_CLIENT1_CLIENT_ID":                    "my-client-id",
 | 
			
		||||
		"PROVIDERS_CLIENT1_CLIENT_SECRET":                "my-client-secret",
 | 
			
		||||
		"PROVIDERS_MY_AWESOME_CLIENT_CLIENT_ID":          "my-awesome-client-id",
 | 
			
		||||
		"PROVIDERS_MY_AWESOME_CLIENT_CLIENT_SECRET_FILE": "/path/to/secret",
 | 
			
		||||
		"I_LOOK_LIKE_A_KEY_CLIENT_ID":                    "should-not-appear",
 | 
			
		||||
		"PROVIDERS_CLIENT_ID":                            "should-not-appear",
 | 
			
		||||
	}
 | 
			
		||||
	expected := map[string]string{
 | 
			
		||||
		"tinyauth.providers.client1.clientId":                 "my-client-id",
 | 
			
		||||
		"tinyauth.providers.client1.clientSecret":             "my-client-secret",
 | 
			
		||||
		"tinyauth.providers.myAwesomeClient.clientId":         "my-awesome-client-id",
 | 
			
		||||
		"tinyauth.providers.myAwesomeClient.clientSecretFile": "/path/to/secret",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	normalized := decoders.NormalizeKeys(test, "tinyauth", "_")
 | 
			
		||||
	assert.DeepEqual(t, normalized, expected)
 | 
			
		||||
 | 
			
		||||
	// Test with flags (assume -- is already stripped)
 | 
			
		||||
	test = map[string]string{
 | 
			
		||||
		"providers-client1-client-id":                    "my-client-id",
 | 
			
		||||
		"providers-client1-client-secret":                "my-client-secret",
 | 
			
		||||
		"providers-my-awesome-client-client-id":          "my-awesome-client-id",
 | 
			
		||||
		"providers-my-awesome-client-client-secret-file": "/path/to/secret",
 | 
			
		||||
		"providers-should-not-appear-client":             "should-not-appear",
 | 
			
		||||
		"i-look-like-a-key-client-id":                    "should-not-appear",
 | 
			
		||||
		"providers-client-id":                            "should-not-appear",
 | 
			
		||||
	}
 | 
			
		||||
	expected = map[string]string{
 | 
			
		||||
		"tinyauth.providers.client1.clientId":                 "my-client-id",
 | 
			
		||||
		"tinyauth.providers.client1.clientSecret":             "my-client-secret",
 | 
			
		||||
		"tinyauth.providers.myAwesomeClient.clientId":         "my-awesome-client-id",
 | 
			
		||||
		"tinyauth.providers.myAwesomeClient.clientSecretFile": "/path/to/secret",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	normalized = decoders.NormalizeKeys(test, "tinyauth", "-")
 | 
			
		||||
	assert.DeepEqual(t, normalized, expected)
 | 
			
		||||
}
 | 
			
		||||
@@ -1,20 +1,19 @@
 | 
			
		||||
package decoders
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"tinyauth/internal/config"
 | 
			
		||||
 | 
			
		||||
	"github.com/traefik/paerser/parser"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func DecodeEnv(env map[string]string) (config.Providers, error) {
 | 
			
		||||
	normalized := NormalizeKeys(env, "tinyauth", "_")
 | 
			
		||||
	var providers config.Providers
 | 
			
		||||
func DecodeEnv[T any, C any](env map[string]string, subName string) (T, error) {
 | 
			
		||||
	var result T
 | 
			
		||||
 | 
			
		||||
	err := parser.Decode(normalized, &providers, "tinyauth", "tinyauth.providers")
 | 
			
		||||
	normalized := normalizeKeys[C](env, subName, "_")
 | 
			
		||||
 | 
			
		||||
	err := parser.Decode(normalized, &result, "tinyauth", "tinyauth."+subName)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return config.Providers{}, err
 | 
			
		||||
		return result, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return providers, nil
 | 
			
		||||
	return result, nil
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -9,52 +9,29 @@ import (
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestDecodeEnv(t *testing.T) {
 | 
			
		||||
	// Variables
 | 
			
		||||
	expected := config.Providers{
 | 
			
		||||
		Providers: map[string]config.OAuthServiceConfig{
 | 
			
		||||
			"client1": {
 | 
			
		||||
				ClientID:           "client1-id",
 | 
			
		||||
				ClientSecret:       "client1-secret",
 | 
			
		||||
				Scopes:             []string{"client1-scope1", "client1-scope2"},
 | 
			
		||||
				RedirectURL:        "client1-redirect-url",
 | 
			
		||||
				AuthURL:            "client1-auth-url",
 | 
			
		||||
				UserinfoURL:        "client1-user-info-url",
 | 
			
		||||
				Name:               "Client1",
 | 
			
		||||
				InsecureSkipVerify: false,
 | 
			
		||||
			},
 | 
			
		||||
			"client2": {
 | 
			
		||||
				ClientID:           "client2-id",
 | 
			
		||||
				ClientSecret:       "client2-secret",
 | 
			
		||||
				Scopes:             []string{"client2-scope1", "client2-scope2"},
 | 
			
		||||
				RedirectURL:        "client2-redirect-url",
 | 
			
		||||
				AuthURL:            "client2-auth-url",
 | 
			
		||||
				UserinfoURL:        "client2-user-info-url",
 | 
			
		||||
				Name:               "My Awesome Client2",
 | 
			
		||||
				InsecureSkipVerify: false,
 | 
			
		||||
			},
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
	test := map[string]string{
 | 
			
		||||
		"PROVIDERS_CLIENT1_CLIENT_ID":            "client1-id",
 | 
			
		||||
		"PROVIDERS_CLIENT1_CLIENT_SECRET":        "client1-secret",
 | 
			
		||||
		"PROVIDERS_CLIENT1_SCOPES":               "client1-scope1,client1-scope2",
 | 
			
		||||
		"PROVIDERS_CLIENT1_REDIRECT_URL":         "client1-redirect-url",
 | 
			
		||||
		"PROVIDERS_CLIENT1_AUTH_URL":             "client1-auth-url",
 | 
			
		||||
		"PROVIDERS_CLIENT1_USER_INFO_URL":        "client1-user-info-url",
 | 
			
		||||
		"PROVIDERS_CLIENT1_NAME":                 "Client1",
 | 
			
		||||
		"PROVIDERS_CLIENT1_INSECURE_SKIP_VERIFY": "false",
 | 
			
		||||
		"PROVIDERS_CLIENT2_CLIENT_ID":            "client2-id",
 | 
			
		||||
		"PROVIDERS_CLIENT2_CLIENT_SECRET":        "client2-secret",
 | 
			
		||||
		"PROVIDERS_CLIENT2_SCOPES":               "client2-scope1,client2-scope2",
 | 
			
		||||
		"PROVIDERS_CLIENT2_REDIRECT_URL":         "client2-redirect-url",
 | 
			
		||||
		"PROVIDERS_CLIENT2_AUTH_URL":             "client2-auth-url",
 | 
			
		||||
		"PROVIDERS_CLIENT2_USER_INFO_URL":        "client2-user-info-url",
 | 
			
		||||
		"PROVIDERS_CLIENT2_NAME":                 "My Awesome Client2",
 | 
			
		||||
		"PROVIDERS_CLIENT2_INSECURE_SKIP_VERIFY": "false",
 | 
			
		||||
	// Setup
 | 
			
		||||
	env := map[string]string{
 | 
			
		||||
		"PROVIDERS_GOOGLE_CLIENT_ID":        "google-client-id",
 | 
			
		||||
		"PROVIDERS_GOOGLE_CLIENT_SECRET":    "google-client-secret",
 | 
			
		||||
		"PROVIDERS_MY_GITHUB_CLIENT_ID":     "github-client-id",
 | 
			
		||||
		"PROVIDERS_MY_GITHUB_CLIENT_SECRET": "github-client-secret",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Test
 | 
			
		||||
	res, err := decoders.DecodeEnv(test)
 | 
			
		||||
	expected := config.Providers{
 | 
			
		||||
		Providers: map[string]config.OAuthServiceConfig{
 | 
			
		||||
			"google": {
 | 
			
		||||
				ClientID:     "google-client-id",
 | 
			
		||||
				ClientSecret: "google-client-secret",
 | 
			
		||||
			},
 | 
			
		||||
			"myGithub": {
 | 
			
		||||
				ClientID:     "github-client-id",
 | 
			
		||||
				ClientSecret: "github-client-secret",
 | 
			
		||||
			},
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Execute
 | 
			
		||||
	result, err := decoders.DecodeEnv[config.Providers, config.OAuthServiceConfig](env, "providers")
 | 
			
		||||
	assert.NilError(t, err)
 | 
			
		||||
	assert.DeepEqual(t, expected, res)
 | 
			
		||||
	assert.DeepEqual(t, result, expected)
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -2,23 +2,23 @@ package decoders
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"strings"
 | 
			
		||||
	"tinyauth/internal/config"
 | 
			
		||||
 | 
			
		||||
	"github.com/traefik/paerser/parser"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func DecodeFlags(flags map[string]string) (config.Providers, error) {
 | 
			
		||||
	filtered := filterFlags(flags)
 | 
			
		||||
	normalized := NormalizeKeys(filtered, "tinyauth", "-")
 | 
			
		||||
	var providers config.Providers
 | 
			
		||||
func DecodeFlags[T any, C any](flags map[string]string, subName string) (T, error) {
 | 
			
		||||
	var result T
 | 
			
		||||
 | 
			
		||||
	err := parser.Decode(normalized, &providers, "tinyauth", "tinyauth.providers")
 | 
			
		||||
	filtered := filterFlags(flags)
 | 
			
		||||
	normalized := normalizeKeys[C](filtered, subName, "_")
 | 
			
		||||
 | 
			
		||||
	err := parser.Decode(normalized, &result, "tinyauth", "tinyauth."+subName)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return config.Providers{}, err
 | 
			
		||||
		return result, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return providers, nil
 | 
			
		||||
	return result, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func filterFlags(flags map[string]string) map[string]string {
 | 
			
		||||
 
 | 
			
		||||
@@ -9,52 +9,29 @@ import (
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func TestDecodeFlags(t *testing.T) {
 | 
			
		||||
	// Variables
 | 
			
		||||
	expected := config.Providers{
 | 
			
		||||
		Providers: map[string]config.OAuthServiceConfig{
 | 
			
		||||
			"client1": {
 | 
			
		||||
				ClientID:           "client1-id",
 | 
			
		||||
				ClientSecret:       "client1-secret",
 | 
			
		||||
				Scopes:             []string{"client1-scope1", "client1-scope2"},
 | 
			
		||||
				RedirectURL:        "client1-redirect-url",
 | 
			
		||||
				AuthURL:            "client1-auth-url",
 | 
			
		||||
				UserinfoURL:        "client1-user-info-url",
 | 
			
		||||
				Name:               "Client1",
 | 
			
		||||
				InsecureSkipVerify: false,
 | 
			
		||||
			},
 | 
			
		||||
			"client2": {
 | 
			
		||||
				ClientID:           "client2-id",
 | 
			
		||||
				ClientSecret:       "client2-secret",
 | 
			
		||||
				Scopes:             []string{"client2-scope1", "client2-scope2"},
 | 
			
		||||
				RedirectURL:        "client2-redirect-url",
 | 
			
		||||
				AuthURL:            "client2-auth-url",
 | 
			
		||||
				UserinfoURL:        "client2-user-info-url",
 | 
			
		||||
				Name:               "My Awesome Client2",
 | 
			
		||||
				InsecureSkipVerify: false,
 | 
			
		||||
			},
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
	test := map[string]string{
 | 
			
		||||
		"--providers-client1-client-id":            "client1-id",
 | 
			
		||||
		"--providers-client1-client-secret":        "client1-secret",
 | 
			
		||||
		"--providers-client1-scopes":               "client1-scope1,client1-scope2",
 | 
			
		||||
		"--providers-client1-redirect-url":         "client1-redirect-url",
 | 
			
		||||
		"--providers-client1-auth-url":             "client1-auth-url",
 | 
			
		||||
		"--providers-client1-user-info-url":        "client1-user-info-url",
 | 
			
		||||
		"--providers-client1-name":                 "Client1",
 | 
			
		||||
		"--providers-client1-insecure-skip-verify": "false",
 | 
			
		||||
		"--providers-client2-client-id":            "client2-id",
 | 
			
		||||
		"--providers-client2-client-secret":        "client2-secret",
 | 
			
		||||
		"--providers-client2-scopes":               "client2-scope1,client2-scope2",
 | 
			
		||||
		"--providers-client2-redirect-url":         "client2-redirect-url",
 | 
			
		||||
		"--providers-client2-auth-url":             "client2-auth-url",
 | 
			
		||||
		"--providers-client2-user-info-url":        "client2-user-info-url",
 | 
			
		||||
		"--providers-client2-name":                 "My Awesome Client2",
 | 
			
		||||
		"--providers-client2-insecure-skip-verify": "false",
 | 
			
		||||
	// Setup
 | 
			
		||||
	flags := map[string]string{
 | 
			
		||||
		"--providers-google-client-id":        "google-client-id",
 | 
			
		||||
		"--providers-google-client-secret":    "google-client-secret",
 | 
			
		||||
		"--providers-my-github-client-id":     "github-client-id",
 | 
			
		||||
		"--providers-my-github-client-secret": "github-client-secret",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Test
 | 
			
		||||
	res, err := decoders.DecodeFlags(test)
 | 
			
		||||
	expected := config.Providers{
 | 
			
		||||
		Providers: map[string]config.OAuthServiceConfig{
 | 
			
		||||
			"google": {
 | 
			
		||||
				ClientID:     "google-client-id",
 | 
			
		||||
				ClientSecret: "google-client-secret",
 | 
			
		||||
			},
 | 
			
		||||
			"myGithub": {
 | 
			
		||||
				ClientID:     "github-client-id",
 | 
			
		||||
				ClientSecret: "github-client-secret",
 | 
			
		||||
			},
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Execute
 | 
			
		||||
	result, err := decoders.DecodeFlags[config.Providers, config.OAuthServiceConfig](flags, "providers")
 | 
			
		||||
	assert.NilError(t, err)
 | 
			
		||||
	assert.DeepEqual(t, expected, res)
 | 
			
		||||
	assert.DeepEqual(t, result, expected)
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user