mirror of
				https://github.com/steveiliop56/tinyauth.git
				synced 2025-10-31 06:05:43 +00:00 
			
		
		
		
	Compare commits
	
		
			49 Commits
		
	
	
		
			30fe695371
			...
			nightly
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 0227af6d2b | ||
|   | c5bb389258 | ||
|   | 6647c6cd78 | ||
|   | 7231efcbc3 | ||
|   | 5482430907 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 97639ae903 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 82350594c1 | ||
|   | 57b7b66813 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2ea921f3ca | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 473109b36a | ||
|   | f628d1f0b3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a9c1bf8865 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 81136eeb42 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8ee331a564 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0996711f08 | ||
|   | 64222b6d15 | ||
|   | 1b87ed9b99 | ||
|   | dc67be2ba0 | ||
|   | 9b76a84ee2 | ||
|   | ed20d2cf51 | ||
|   | fc7e395e66 | ||
|   | b940d681c3 | ||
|   | a1ec4a69cf | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 4047cea451 | ||
|   | 5a4855c12c | ||
|   | 05d4dbd68e | ||
|   | ae8347fd28 | ||
|   | 76f2014444 | ||
|   | 5b7bda3378 | ||
|   | e878516130 | ||
|   | e5f1df03c4 | ||
|   | c77da30d87 | ||
|   | 287c6f975f | ||
|   | 0255e954f7 | ||
|   | c5d70d7c93 | ||
|   | adffb4ac0a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cbe31d442d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4a530eebc9 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9ba1695274 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c337ba5b31 | ||
|   | bbf8112995 | ||
|   | 103285855e | ||
|   | 2cc6b6bdbb | ||
|   | adb1a9bee5 | ||
|   | 1ee0cee171 | ||
|   | 720f387908 | ||
|   | a629430a88 | ||
|   | f0a48cc91c | ||
|   | 2f8fa39a9b | 
							
								
								
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -23,7 +23,7 @@ jobs: | |||||||
|       - name: Install frontend dependencies |       - name: Install frontend dependencies | ||||||
|         run: | |         run: | | ||||||
|           cd frontend |           cd frontend | ||||||
|           bun install |           bun install --frozen-lockfile | ||||||
|  |  | ||||||
|       - name: Set version |       - name: Set version | ||||||
|         run: | |         run: | | ||||||
|   | |||||||
							
								
								
									
										173
									
								
								.github/workflows/nightly.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										173
									
								
								.github/workflows/nightly.yml
									
									
									
									
										vendored
									
									
								
							| @@ -66,7 +66,7 @@ jobs: | |||||||
|       - name: Install frontend dependencies |       - name: Install frontend dependencies | ||||||
|         run: | |         run: | | ||||||
|           cd frontend |           cd frontend | ||||||
|           bun install |           bun install --frozen-lockfile | ||||||
|  |  | ||||||
|       - name: Install backend dependencies |       - name: Install backend dependencies | ||||||
|         run: | |         run: | | ||||||
| @@ -112,7 +112,7 @@ jobs: | |||||||
|       - name: Install frontend dependencies |       - name: Install frontend dependencies | ||||||
|         run: | |         run: | | ||||||
|           cd frontend |           cd frontend | ||||||
|           bun install |           bun install --frozen-lockfile | ||||||
|  |  | ||||||
|       - name: Install backend dependencies |       - name: Install backend dependencies | ||||||
|         run: | |         run: | | ||||||
| @@ -171,6 +171,9 @@ jobs: | |||||||
|           labels: ${{ steps.meta.outputs.labels }} |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|           tags: ghcr.io/${{ github.repository_owner }}/tinyauth |           tags: ghcr.io/${{ github.repository_owner }}/tinyauth | ||||||
|           outputs: type=image,push-by-digest=true,name-canonical=true,push=true |           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: | |           build-args: | | ||||||
|             VERSION=${{ needs.generate-metadata.outputs.VERSION }} |             VERSION=${{ needs.generate-metadata.outputs.VERSION }} | ||||||
|             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} |             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} | ||||||
| @@ -190,6 +193,65 @@ jobs: | |||||||
|           if-no-files-found: error |           if-no-files-found: error | ||||||
|           retention-days: 1 |           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: |   image-build-arm: | ||||||
|     runs-on: ubuntu-24.04-arm |     runs-on: ubuntu-24.04-arm | ||||||
|     needs: |     needs: | ||||||
| @@ -217,10 +279,6 @@ jobs: | |||||||
|       - name: Set up Docker Buildx |       - name: Set up Docker Buildx | ||||||
|         uses: docker/setup-buildx-action@v3 |         uses: docker/setup-buildx-action@v3 | ||||||
|  |  | ||||||
|       - name: Set version |  | ||||||
|         run: | |  | ||||||
|           echo nightly > internal/assets/version |  | ||||||
|  |  | ||||||
|       - name: Build and push |       - name: Build and push | ||||||
|         uses: docker/build-push-action@v6 |         uses: docker/build-push-action@v6 | ||||||
|         id: build |         id: build | ||||||
| @@ -229,6 +287,9 @@ jobs: | |||||||
|           labels: ${{ steps.meta.outputs.labels }} |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|           tags: ghcr.io/${{ github.repository_owner }}/tinyauth |           tags: ghcr.io/${{ github.repository_owner }}/tinyauth | ||||||
|           outputs: type=image,push-by-digest=true,name-canonical=true,push=true |           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: | |           build-args: | | ||||||
|             VERSION=${{ needs.generate-metadata.outputs.VERSION }} |             VERSION=${{ needs.generate-metadata.outputs.VERSION }} | ||||||
|             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} |             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} | ||||||
| @@ -248,6 +309,65 @@ jobs: | |||||||
|           if-no-files-found: error |           if-no-files-found: error | ||||||
|           retention-days: 1 |           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: |   image-merge: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     needs: |     needs: | ||||||
| @@ -276,6 +396,8 @@ jobs: | |||||||
|         uses: docker/metadata-action@v5 |         uses: docker/metadata-action@v5 | ||||||
|         with: |         with: | ||||||
|           images: ghcr.io/${{ github.repository_owner }}/tinyauth |           images: ghcr.io/${{ github.repository_owner }}/tinyauth | ||||||
|  |           flavor: | | ||||||
|  |             latest=false | ||||||
|           tags: | |           tags: | | ||||||
|             type=raw,nightly |             type=raw,nightly | ||||||
|  |  | ||||||
| @@ -285,6 +407,45 @@ jobs: | |||||||
|           docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \ |           docker buildx imagetools create $(jq -cr '.tags | map("-t " + .) | join(" ")' <<< "$DOCKER_METADATA_OUTPUT_JSON") \ | ||||||
|             $(printf 'ghcr.io/${{ github.repository_owner }}/tinyauth@sha256:%s ' *) |             $(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: |   update-release: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     needs: |     needs: | ||||||
|   | |||||||
							
								
								
									
										173
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										173
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -44,7 +44,7 @@ jobs: | |||||||
|       - name: Install frontend dependencies |       - name: Install frontend dependencies | ||||||
|         run: | |         run: | | ||||||
|           cd frontend |           cd frontend | ||||||
|           bun install |           bun install --frozen-lockfile | ||||||
|  |  | ||||||
|       - name: Install backend dependencies |       - name: Install backend dependencies | ||||||
|         run: | |         run: | | ||||||
| @@ -87,7 +87,7 @@ jobs: | |||||||
|       - name: Install frontend dependencies |       - name: Install frontend dependencies | ||||||
|         run: | |         run: | | ||||||
|           cd frontend |           cd frontend | ||||||
|           bun install |           bun install --frozen-lockfile | ||||||
|  |  | ||||||
|       - name: Install backend dependencies |       - name: Install backend dependencies | ||||||
|         run: | |         run: | | ||||||
| @@ -143,6 +143,9 @@ jobs: | |||||||
|           labels: ${{ steps.meta.outputs.labels }} |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|           tags: ghcr.io/${{ github.repository_owner }}/tinyauth |           tags: ghcr.io/${{ github.repository_owner }}/tinyauth | ||||||
|           outputs: type=image,push-by-digest=true,name-canonical=true,push=true |           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: | |           build-args: | | ||||||
|             VERSION=${{ needs.generate-metadata.outputs.VERSION }} |             VERSION=${{ needs.generate-metadata.outputs.VERSION }} | ||||||
|             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} |             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} | ||||||
| @@ -162,6 +165,62 @@ jobs: | |||||||
|           if-no-files-found: error |           if-no-files-found: error | ||||||
|           retention-days: 1 |           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: |   image-build-arm: | ||||||
|     runs-on: ubuntu-24.04-arm |     runs-on: ubuntu-24.04-arm | ||||||
|     needs: |     needs: | ||||||
| @@ -194,6 +253,9 @@ jobs: | |||||||
|           labels: ${{ steps.meta.outputs.labels }} |           labels: ${{ steps.meta.outputs.labels }} | ||||||
|           tags: ghcr.io/${{ github.repository_owner }}/tinyauth |           tags: ghcr.io/${{ github.repository_owner }}/tinyauth | ||||||
|           outputs: type=image,push-by-digest=true,name-canonical=true,push=true |           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: | |           build-args: | | ||||||
|             VERSION=${{ needs.generate-metadata.outputs.VERSION }} |             VERSION=${{ needs.generate-metadata.outputs.VERSION }} | ||||||
|             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} |             COMMIT_HASH=${{ needs.generate-metadata.outputs.COMMIT_HASH }} | ||||||
| @@ -213,6 +275,62 @@ jobs: | |||||||
|           if-no-files-found: error |           if-no-files-found: error | ||||||
|           retention-days: 1 |           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: |   image-merge: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     needs: |     needs: | ||||||
| @@ -241,10 +359,55 @@ jobs: | |||||||
|         uses: docker/metadata-action@v5 |         uses: docker/metadata-action@v5 | ||||||
|         with: |         with: | ||||||
|           images: ghcr.io/${{ github.repository_owner }}/tinyauth |           images: ghcr.io/${{ github.repository_owner }}/tinyauth | ||||||
|  |           flavor: | | ||||||
|  |             prefix=v,onlatest=false | ||||||
|           tags: | |           tags: | | ||||||
|             type=semver,pattern={{version}},prefix=v |             type=semver,pattern={{version}} | ||||||
|             type=semver,pattern={{major}},prefix=v |             type=semver,pattern={{major}} | ||||||
|             type=semver,pattern={{major}}.{{minor}},prefix=v |             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 |       - name: Create manifest list and push | ||||||
|         working-directory: ${{ runner.temp }}/digests |         working-directory: ${{ runner.temp }}/digests | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								Dockerfile
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								Dockerfile
									
									
									
									
									
								
							| @@ -1,12 +1,12 @@ | |||||||
| # Site builder | # Site builder | ||||||
| FROM oven/bun:1.2.23-alpine AS frontend-builder | FROM oven/bun:1.3.0-alpine AS frontend-builder | ||||||
|  |  | ||||||
| WORKDIR /frontend | WORKDIR /frontend | ||||||
|  |  | ||||||
| COPY ./frontend/package.json ./ | COPY ./frontend/package.json ./ | ||||||
| COPY ./frontend/bun.lock ./ | COPY ./frontend/bun.lock ./ | ||||||
|  |  | ||||||
| RUN bun install | RUN bun install --frozen-lockfile | ||||||
|  |  | ||||||
| COPY ./frontend/public ./public | COPY ./frontend/public ./public | ||||||
| COPY ./frontend/src ./src | COPY ./frontend/src ./src | ||||||
| @@ -45,12 +45,18 @@ FROM alpine:3.22 AS runner | |||||||
|  |  | ||||||
| WORKDIR /tinyauth | WORKDIR /tinyauth | ||||||
|  |  | ||||||
| RUN apk add --no-cache curl |  | ||||||
|  |  | ||||||
| COPY --from=builder /tinyauth/tinyauth ./ | COPY --from=builder /tinyauth/tinyauth ./ | ||||||
|  |  | ||||||
|  | RUN mkdir -p /data | ||||||
|  |  | ||||||
| EXPOSE 3000 | EXPOSE 3000 | ||||||
|  |  | ||||||
| VOLUME ["/data"] | 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"> | <div align="center"> | ||||||
|     <img alt="Tinyauth" title="Tinyauth" width="96" src="assets/logo-rounded.png"> |     <img alt="Tinyauth" title="Tinyauth" width="96" src="assets/logo-rounded.png"> | ||||||
|     <h1>Tinyauth</h1> |     <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> | ||||||
|  |  | ||||||
| <div align="center"> | <div align="center"> | ||||||
| @@ -14,7 +14,7 @@ | |||||||
|  |  | ||||||
| <br /> | <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 | ## 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 | ## 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: | 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 | ## Acknowledgements | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								air.toml
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								air.toml
									
									
									
									
									
								
							| @@ -2,7 +2,7 @@ root = "/tinyauth" | |||||||
| tmp_dir = "tmp" | tmp_dir = "tmp" | ||||||
|  |  | ||||||
| [build] | [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 ." | 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" | 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"] | 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 ( | import ( | ||||||
| 	"strings" | 	"strings" | ||||||
| 	totpCmd "tinyauth/cmd/totp" |  | ||||||
| 	userCmd "tinyauth/cmd/user" |  | ||||||
| 	"tinyauth/internal/bootstrap" | 	"tinyauth/internal/bootstrap" | ||||||
| 	"tinyauth/internal/config" | 	"tinyauth/internal/config" | ||||||
| 	"tinyauth/internal/utils" | 	"tinyauth/internal/utils" | ||||||
| @@ -15,55 +13,28 @@ import ( | |||||||
| 	"github.com/spf13/viper" | 	"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", | 		Use:   "tinyauth", | ||||||
| 	Short: "The simplest way to protect your apps with a login screen.", | 		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.`, | 		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: func(cmd *cobra.Command, args []string) { | 		Run:   c.run, | ||||||
| 		var conf config.Config |  | ||||||
|  |  | ||||||
| 		err := viper.Unmarshal(&conf) |  | ||||||
| 		if err != nil { |  | ||||||
| 			log.Fatal().Err(err).Msg("Failed to parse config") |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 		// Validate config | 	c.viper.AutomaticEnv() | ||||||
| 		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() |  | ||||||
|  |  | ||||||
| 	configOptions := []struct { | 	configOptions := []struct { | ||||||
| 		name        string | 		name        string | ||||||
| @@ -101,17 +72,86 @@ func init() { | |||||||
| 	for _, opt := range configOptions { | 	for _, opt := range configOptions { | ||||||
| 		switch v := opt.defaultVal.(type) { | 		switch v := opt.defaultVal.(type) { | ||||||
| 		case bool: | 		case bool: | ||||||
| 			rootCmd.Flags().Bool(opt.name, v, opt.description) | 			c.cmd.Flags().Bool(opt.name, v, opt.description) | ||||||
| 		case int: | 		case int: | ||||||
| 			rootCmd.Flags().Int(opt.name, v, opt.description) | 			c.cmd.Flags().Int(opt.name, v, opt.description) | ||||||
| 		case string: | 		case string: | ||||||
| 			rootCmd.Flags().String(opt.name, v, opt.description) | 			c.cmd.Flags().String(opt.name, v, opt.description) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		// Create uppercase env var name | 		// Create uppercase env var name | ||||||
| 		envVar := strings.ReplaceAll(strings.ToUpper(opt.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" | 	"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", | 		Use:   "version", | ||||||
| 		Short: "Print the version number of Tinyauth", | 		Short: "Print the version number of Tinyauth", | ||||||
| 	Long:  `All software has versions. This is Tinyauth's`, | 		Long:  `All software has versions. This is Tinyauth's.`, | ||||||
| 	Run: func(cmd *cobra.Command, args []string) { | 		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("Version: %s\n", config.Version) | ||||||
| 	fmt.Printf("Commit Hash: %s\n", config.CommitHash) | 	fmt.Printf("Commit Hash: %s\n", config.CommitHash) | ||||||
| 	fmt.Printf("Build Timestamp: %s\n", config.BuildTimestamp) | 	fmt.Printf("Build Timestamp: %s\n", config.BuildTimestamp) | ||||||
| 	}, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func init() { |  | ||||||
| 	rootCmd.AddCommand(versionCmd) |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -10,43 +10,43 @@ | |||||||
|         "@radix-ui/react-separator": "^1.1.7", |         "@radix-ui/react-separator": "^1.1.7", | ||||||
|         "@radix-ui/react-slot": "^1.2.3", |         "@radix-ui/react-slot": "^1.2.3", | ||||||
|         "@tailwindcss/vite": "^4.1.14", |         "@tailwindcss/vite": "^4.1.14", | ||||||
|         "@tanstack/react-query": "^5.90.2", |         "@tanstack/react-query": "^5.90.3", | ||||||
|         "axios": "^1.12.2", |         "axios": "^1.12.2", | ||||||
|         "class-variance-authority": "^0.7.1", |         "class-variance-authority": "^0.7.1", | ||||||
|         "clsx": "^2.1.1", |         "clsx": "^2.1.1", | ||||||
|         "i18next": "^25.5.3", |         "i18next": "^25.6.0", | ||||||
|         "i18next-browser-languagedetector": "^8.2.0", |         "i18next-browser-languagedetector": "^8.2.0", | ||||||
|         "i18next-resources-to-backend": "^1.2.1", |         "i18next-resources-to-backend": "^1.2.1", | ||||||
|         "input-otp": "^1.4.2", |         "input-otp": "^1.4.2", | ||||||
|         "lucide-react": "^0.544.0", |         "lucide-react": "^0.545.0", | ||||||
|         "next-themes": "^0.4.6", |         "next-themes": "^0.4.6", | ||||||
|         "react": "^19.2.0", |         "react": "^19.2.0", | ||||||
|         "react-dom": "^19.2.0", |         "react-dom": "^19.2.0", | ||||||
|         "react-hook-form": "^7.63.0", |         "react-hook-form": "^7.65.0", | ||||||
|         "react-i18next": "^15.7.3", |         "react-i18next": "^16.0.1", | ||||||
|         "react-markdown": "^10.1.0", |         "react-markdown": "^10.1.0", | ||||||
|         "react-router": "^7.9.3", |         "react-router": "^7.9.4", | ||||||
|         "sonner": "^2.0.7", |         "sonner": "^2.0.7", | ||||||
|         "tailwind-merge": "^3.3.1", |         "tailwind-merge": "^3.3.1", | ||||||
|         "tailwindcss": "^4.1.14", |         "tailwindcss": "^4.1.14", | ||||||
|         "zod": "^4.1.11", |         "zod": "^4.1.12", | ||||||
|       }, |       }, | ||||||
|       "devDependencies": { |       "devDependencies": { | ||||||
|         "@eslint/js": "^9.36.0", |         "@eslint/js": "^9.37.0", | ||||||
|         "@tanstack/eslint-plugin-query": "^5.91.0", |         "@tanstack/eslint-plugin-query": "^5.91.0", | ||||||
|         "@types/node": "^24.6.2", |         "@types/node": "^24.7.2", | ||||||
|         "@types/react": "^19.2.0", |         "@types/react": "^19.2.2", | ||||||
|         "@types/react-dom": "^19.2.0", |         "@types/react-dom": "^19.2.2", | ||||||
|         "@vitejs/plugin-react": "^5.0.4", |         "@vitejs/plugin-react": "^5.0.4", | ||||||
|         "eslint": "^9.36.0", |         "eslint": "^9.37.0", | ||||||
|         "eslint-plugin-react-hooks": "^5.2.0", |         "eslint-plugin-react-hooks": "^7.0.0", | ||||||
|         "eslint-plugin-react-refresh": "^0.4.23", |         "eslint-plugin-react-refresh": "^0.4.23", | ||||||
|         "globals": "^16.4.0", |         "globals": "^16.4.0", | ||||||
|         "prettier": "3.6.2", |         "prettier": "3.6.2", | ||||||
|         "tw-animate-css": "^1.4.0", |         "tw-animate-css": "^1.4.0", | ||||||
|         "typescript": "~5.9.3", |         "typescript": "~5.9.3", | ||||||
|         "typescript-eslint": "^8.45.0", |         "typescript-eslint": "^8.46.1", | ||||||
|         "vite": "^7.1.8", |         "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-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/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.36.0", "", {}, "sha512-uhCbYtYynH30iZErszX78U+nR3pJU3RHGQ57NXy5QupD4SBVwDeU8TNBy+MjMngc1UyIW9noKqsRqfjQTBU2dw=="], |     "@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/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=="], |     "@floating-ui/core": ["@floating-ui/core@1.7.0", "", { "dependencies": { "@floating-ui/utils": "^0.2.9" } }, "sha512-FRdBLykrPPA6P76GGGqlex/e7fbe0F1ykgxHYNXQsH/iTEtjMj/f9bpY5oQqbjt5VgZvgz/uKXbGuROijh3VLA=="], | ||||||
|  |  | ||||||
| @@ -329,9 +329,9 @@ | |||||||
|  |  | ||||||
|     "@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/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.90.2", "", {}, "sha512-k/TcR3YalnzibscALLwxeiLUub6jN5EDLwKDiO7q5f4ICEoptJ+n9+7vcEFy5/x/i6Q+Lb/tXrsKCggf5uQJXQ=="], |     "@tanstack/query-core": ["@tanstack/query-core@5.90.3", "", {}, "sha512-HtPOnCwmx4dd35PfXU8jjkhwYrsHfuqgC8RCJIwWglmhIUIlzPP0ZcEkDAc+UtAWCiLm7T8rxeEfHZlz3hYMCA=="], | ||||||
|  |  | ||||||
|     "@tanstack/react-query": ["@tanstack/react-query@5.90.2", "", { "dependencies": { "@tanstack/query-core": "5.90.2" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-CLABiR+h5PYfOWr/z+vWFt5VsOA2ekQeRQBFSKlcoW6Ndx/f8rfyVmq4LbgOM4GG2qtxAxjLYLOpCNTYm4uKzw=="], |     "@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=="], |     "@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,33 +355,33 @@ | |||||||
|  |  | ||||||
|     "@types/ms": ["@types/ms@2.1.0", "", {}, "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA=="], |     "@types/ms": ["@types/ms@2.1.0", "", {}, "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA=="], | ||||||
|  |  | ||||||
|     "@types/node": ["@types/node@24.6.2", "", { "dependencies": { "undici-types": "~7.13.0" } }, "sha512-d2L25Y4j+W3ZlNAeMKcy7yDsK425ibcAOO2t7aPTz6gNMH0z2GThtwENCDc0d/Pw9wgyRqE5Px1wkV7naz8ang=="], |     "@types/node": ["@types/node@24.7.2", "", { "dependencies": { "undici-types": "~7.14.0" } }, "sha512-/NbVmcGTP+lj5oa4yiYxxeBjRivKQ5Ns1eSZeB99ExsEQ6rX5XYU1Zy/gGxY/ilqtD4Etx9mKyrPxZRetiahhA=="], | ||||||
|  |  | ||||||
|     "@types/react": ["@types/react@19.2.0", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-1LOH8xovvsKsCBq1wnT4ntDUdCJKmnEakhsuoUSy6ExlHCkGP2hqnatagYTgFk6oeL0VU31u7SNjunPN+GchtA=="], |     "@types/react": ["@types/react@19.2.2", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-6mDvHUFSjyT2B2yeNx2nUgMxh9LtOWvkhIU3uePn2I2oyNymUAX1NIsdgviM4CH+JSrp2D2hsMvJOkxY+0wNRA=="], | ||||||
|  |  | ||||||
|     "@types/react-dom": ["@types/react-dom@19.2.0", "", { "peerDependencies": { "@types/react": "^19.2.0" } }, "sha512-brtBs0MnE9SMx7px208g39lRmC5uHZs96caOJfTjFcYSLHNamvaSMfJNagChVNkup2SdtOxKX1FDBkRSJe1ZAg=="], |     "@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=="], |     "@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="], | ||||||
|  |  | ||||||
|     "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.45.0", "", { "dependencies": { "@eslint-community/regexpp": "^4.10.0", "@typescript-eslint/scope-manager": "8.45.0", "@typescript-eslint/type-utils": "8.45.0", "@typescript-eslint/utils": "8.45.0", "@typescript-eslint/visitor-keys": "8.45.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.45.0", "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-HC3y9CVuevvWCl/oyZuI47dOeDF9ztdMEfMH8/DW/Mhwa9cCLnK1oD7JoTVGW/u7kFzNZUKUoyJEqkaJh5y3Wg=="], |     "@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.45.0", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.45.0", "@typescript-eslint/types": "8.45.0", "@typescript-eslint/typescript-estree": "8.45.0", "@typescript-eslint/visitor-keys": "8.45.0", "debug": "^4.3.4" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-TGf22kon8KW+DeKaUmOibKWktRY8b2NSAZNdtWh798COm1NWx8+xJ6iFBtk3IvLdv6+LGLJLRlyhrhEDZWargQ=="], |     "@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.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/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.45.0", "", { "dependencies": { "@typescript-eslint/types": "8.45.0", "@typescript-eslint/visitor-keys": "8.45.0" } }, "sha512-clmm8XSNj/1dGvJeO6VGH7EUSeA0FMs+5au/u3lrA3KfG8iJ4u8ym9/j2tTEoacAffdW1TVUzXO30W1JTJS7dA=="], |     "@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.45.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <6.0.0" } }, "sha512-aFdr+c37sc+jqNMGhH+ajxPXwjv9UtFZk79k8pLoJ6p4y0snmYpPA52GuWHgt2ZF4gRRW6odsEj41uZLojDt5w=="], |     "@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.45.0", "", { "dependencies": { "@typescript-eslint/types": "8.45.0", "@typescript-eslint/typescript-estree": "8.45.0", "@typescript-eslint/utils": "8.45.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-bpjepLlHceKgyMEPglAeULX1vixJDgaKocp0RVJ5u4wLJIMNuKtUXIczpJCPcn2waII0yuvks/5m5/h3ZQKs0A=="], |     "@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.45.0", "", {}, "sha512-WugXLuOIq67BMgQInIxxnsSyRLFxdkJEJu8r4ngLR56q/4Q5LrbfkFRH27vMTjxEK8Pyz7QfzuZe/G15qQnVRA=="], |     "@typescript-eslint/types": ["@typescript-eslint/types@8.45.0", "", {}, "sha512-WugXLuOIq67BMgQInIxxnsSyRLFxdkJEJu8r4ngLR56q/4Q5LrbfkFRH27vMTjxEK8Pyz7QfzuZe/G15qQnVRA=="], | ||||||
|  |  | ||||||
|     "@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/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.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/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.45.0", "", { "dependencies": { "@typescript-eslint/types": "8.45.0", "eslint-visitor-keys": "^4.2.1" } }, "sha512-qsaFBA3e09MIDAGFUrTk+dzqtfv1XPVz8t8d1f0ybTzrCY7BKiMC5cjrl1O/P7UmHsNyW90EYSkU/ZWpmXelag=="], |     "@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=="], |     "@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="], | ||||||
|  |  | ||||||
| @@ -491,9 +491,9 @@ | |||||||
|  |  | ||||||
|     "escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="], |     "escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="], | ||||||
|  |  | ||||||
|     "eslint": ["eslint@9.36.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.36.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-hB4FIzXovouYzwzECDcUkJ4OcfOEkXTv2zRY6B9bkwjx/cprAq0uvm1nl7zvQ0/TsUk0zQiN4uPfJpB9m+rPMQ=="], |     "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.23", "", { "peerDependencies": { "eslint": ">=8.40" } }, "sha512-G4j+rv0NmbIR45kni5xJOrYvCtyD3/7LjpVH8MPPcudXDcNu8gv+4ATTDXTtbRR8rTCM5HxECvCSsRmxKnWDsA=="], |     "eslint-plugin-react-refresh": ["eslint-plugin-react-refresh@0.4.23", "", { "peerDependencies": { "eslint": ">=8.40" } }, "sha512-G4j+rv0NmbIR45kni5xJOrYvCtyD3/7LjpVH8MPPcudXDcNu8gv+4ATTDXTtbRR8rTCM5HxECvCSsRmxKnWDsA=="], | ||||||
|  |  | ||||||
| @@ -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=="], |     "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-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=="], |     "html-url-attributes": ["html-url-attributes@3.0.1", "", {}, "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ=="], | ||||||
|  |  | ||||||
|     "i18next": ["i18next@25.5.3", "", { "dependencies": { "@babel/runtime": "^7.27.6" }, "peerDependencies": { "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-joFqorDeQ6YpIXni944upwnuHBf5IoPMuqAchGVeQLdWC2JOjxgM9V8UGLhNIIH/Q8QleRxIi0BSRQehSrDLcg=="], |     "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=="], |     "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=="], |     "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=="], |     "magic-string": ["magic-string@0.30.19", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-2N21sPY9Ws53PZvsEpVtNuSW+ScYbQdp4b9qUaL+9QkHUrGFKo56Lg9Emg5s9V/qrtNBmiR01sYhUOwu3H+VOw=="], | ||||||
|  |  | ||||||
| @@ -787,9 +791,9 @@ | |||||||
|  |  | ||||||
|     "react-dom": ["react-dom@19.2.0", "", { "dependencies": { "scheduler": "^0.27.0" }, "peerDependencies": { "react": "^19.2.0" } }, "sha512-UlbRu4cAiGaIewkPyiRGJk0imDN2T3JjieT6spoL2UeSf5od4n5LB/mQ4ejmxhCFT1tYe8IvaFulzynWovsEFQ=="], |     "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.63.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17 || ^18 || ^19" } }, "sha512-ZwueDMvUeucovM2VjkCf7zIHcs1aAlDimZu2Hvel5C5907gUzMpm4xCrQXtRzCvsBqFjonB4m3x4LzCFI1ZKWA=="], |     "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=="], |     "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=="], | ||||||
|  |  | ||||||
| @@ -799,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-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.3", "", { "dependencies": { "cookie": "^1.0.1", "set-cookie-parser": "^2.6.0" }, "peerDependencies": { "react": ">=18", "react-dom": ">=18" }, "optionalPeers": ["react-dom"] }, "sha512-4o2iWCFIwhI/eYAIL43+cjORXYn/aRQPgtFRRZb3VzoyQ5Uej0Bmqj7437L97N9NJW4wnicSwLOLS+yCXfAPgg=="], |     "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=="], |     "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=="], | ||||||
|  |  | ||||||
| @@ -867,9 +871,9 @@ | |||||||
|  |  | ||||||
|     "typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="], |     "typescript": ["typescript@5.9.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw=="], | ||||||
|  |  | ||||||
|     "typescript-eslint": ["typescript-eslint@8.45.0", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.45.0", "@typescript-eslint/parser": "8.45.0", "@typescript-eslint/typescript-estree": "8.45.0", "@typescript-eslint/utils": "8.45.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <6.0.0" } }, "sha512-qzDmZw/Z5beNLUrXfd0HIW6MzIaAV5WNDxmMs9/3ojGOpYavofgNAAD/nC6tGV2PczIi0iw8vot2eAe/sBn7zg=="], |     "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.13.0", "", {}, "sha512-Ov2Rr9Sx+fRgagJ5AX0qvItZG/JKKoBRAVITs1zk7IqZGTJUwgUr7qoYBpWwakpWilTZFM98rG/AFRocu10iIQ=="], |     "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=="], |     "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=="], | ||||||
|  |  | ||||||
| @@ -895,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=="], |     "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.8", "", { "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-oBXvfSHEOL8jF+R9Am7h59Up07kVVGH1NrFGFoEG6bPDZP3tGpQhvkBpy5x7U6+E6wZCu9OihsWgJqDbQIm8LQ=="], |     "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=="], |     "void-elements": ["void-elements@3.1.0", "", {}, "sha512-Dhxzh5HZuiHQhbvTW9AMetFfBHDMYpo23Uo9btPXgdYP+3T5S+p+jgNy7spra+veYhBP2dCSgxR/i2Y02h5/6w=="], | ||||||
|  |  | ||||||
| @@ -907,7 +911,9 @@ | |||||||
|  |  | ||||||
|     "yocto-queue": ["yocto-queue@0.1.0", "", {}, "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q=="], |     "yocto-queue": ["yocto-queue@0.1.0", "", {}, "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q=="], | ||||||
|  |  | ||||||
|     "zod": ["zod@4.1.11", "", {}, "sha512-WPsqwxITS2tzx1bzhIKsEs19ABD5vmCVa4xBo2tq/SrV4RNZtfws1EnCWQXM6yh8bD08a1idvkB5MZSBiZsjwg=="], |     "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=="], |     "zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="], | ||||||
|  |  | ||||||
| @@ -965,12 +971,34 @@ | |||||||
|  |  | ||||||
|     "@types/estree-jsx/@types/estree": ["@types/estree@1.0.7", "", {}, "sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ=="], |     "@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.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.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/eslint-plugin/ignore": ["ignore@7.0.4", "", {}, "sha512-gJzzk+PQNznz8ysRrC0aOkBNVRBDtE1n53IqyqEf3PXrYwomFs5q4pGMizBMJF+ykh03insJ27hB8gSrD2Hn8A=="], | ||||||
|  |  | ||||||
|  |     "@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.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="], | ||||||
|  |  | ||||||
|  |     "@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.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.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="], | ||||||
|  |  | ||||||
|  |     "@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.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/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/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.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.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=="], |     "fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="], | ||||||
|  |  | ||||||
|     "hast-util-to-jsx-runtime/@types/estree": ["@types/estree@1.0.7", "", {}, "sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ=="], |     "hast-util-to-jsx-runtime/@types/estree": ["@types/estree@1.0.7", "", {}, "sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ=="], | ||||||
| @@ -985,6 +1013,8 @@ | |||||||
|  |  | ||||||
|     "parse-entities/@types/unist": ["@types/unist@2.0.11", "", {}, "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA=="], |     "parse-entities/@types/unist": ["@types/unist@2.0.11", "", {}, "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA=="], | ||||||
|  |  | ||||||
|  |     "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=="], |     "@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=="], | ||||||
|  |  | ||||||
|     "@babel/helper-module-imports/@babel/traverse/@babel/parser": ["@babel/parser@7.27.2", "", { "dependencies": { "@babel/types": "^7.27.1" }, "bin": "./bin/babel-parser.js" }, "sha512-QYLs8299NA7WM/bZAdp+CviYYkVoYXlDW2rzliy3chxd1PQjej7JORuMJDJXJUb9g0TT+B99EwaVLKmX+sPXWw=="], |     "@babel/helper-module-imports/@babel/traverse/@babel/parser": ["@babel/parser@7.27.2", "", { "dependencies": { "@babel/types": "^7.27.1" }, "bin": "./bin/babel-parser.js" }, "sha512-QYLs8299NA7WM/bZAdp+CviYYkVoYXlDW2rzliy3chxd1PQjej7JORuMJDJXJUb9g0TT+B99EwaVLKmX+sPXWw=="], | ||||||
| @@ -999,12 +1029,34 @@ | |||||||
|  |  | ||||||
|     "@eslint/eslintrc/espree/eslint-visitor-keys": ["eslint-visitor-keys@4.2.0", "", {}, "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw=="], |     "@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.46.1", "", {}, "sha512-C+soprGBHwWBdkDpbaRC4paGBrkIXxVlNohadL5o0kfhsXqOC6GYH2S/Obmig+I0HTDl8wMaRySwrfrXVP8/pQ=="], | ||||||
|  |  | ||||||
|  |     "@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.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/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.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.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.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.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.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=="], |     "@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=="], | ||||||
|  |  | ||||||
|     "@babel/helper-module-imports/@babel/traverse/@babel/generator/@jridgewell/trace-mapping": ["@jridgewell/trace-mapping@0.3.25", "", { "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ=="], |     "@babel/helper-module-imports/@babel/traverse/@babel/generator/@jridgewell/trace-mapping": ["@jridgewell/trace-mapping@0.3.25", "", { "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ=="], | ||||||
|  |  | ||||||
|  |     "@typescript-eslint/utils/@typescript-eslint/typescript-estree/minimatch/brace-expansion": ["brace-expansion@2.0.1", "", { "dependencies": { "balanced-match": "^1.0.0" } }, "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA=="], | ||||||
|  |  | ||||||
|     "@babel/helper-module-imports/@babel/traverse/@babel/generator/@jridgewell/gen-mapping/@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.0", "", {}, "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ=="], |     "@babel/helper-module-imports/@babel/traverse/@babel/generator/@jridgewell/gen-mapping/@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.0", "", {}, "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ=="], | ||||||
|  |  | ||||||
|     "@babel/helper-module-imports/@babel/traverse/@babel/generator/@jridgewell/trace-mapping/@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.0", "", {}, "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ=="], |     "@babel/helper-module-imports/@babel/traverse/@babel/generator/@jridgewell/trace-mapping/@jridgewell/sourcemap-codec": ["@jridgewell/sourcemap-codec@1.5.0", "", {}, "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ=="], | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ | |||||||
|     <link rel="shortcut icon" href="/favicon.ico" /> |     <link rel="shortcut icon" href="/favicon.ico" /> | ||||||
|     <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" /> |     <link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" /> | ||||||
|     <meta name="apple-mobile-web-app-title" content="Tinyauth" /> |     <meta name="apple-mobile-web-app-title" content="Tinyauth" /> | ||||||
|     <meta name="robots" content="none" /> |     <meta name="robots" content="nofollow, noindex" /> | ||||||
|     <link rel="manifest" href="/site.webmanifest" /> |     <link rel="manifest" href="/site.webmanifest" /> | ||||||
|     <title>Tinyauth</title> |     <title>Tinyauth</title> | ||||||
|   </head> |   </head> | ||||||
|   | |||||||
| @@ -16,42 +16,42 @@ | |||||||
|     "@radix-ui/react-separator": "^1.1.7", |     "@radix-ui/react-separator": "^1.1.7", | ||||||
|     "@radix-ui/react-slot": "^1.2.3", |     "@radix-ui/react-slot": "^1.2.3", | ||||||
|     "@tailwindcss/vite": "^4.1.14", |     "@tailwindcss/vite": "^4.1.14", | ||||||
|     "@tanstack/react-query": "^5.90.2", |     "@tanstack/react-query": "^5.90.3", | ||||||
|     "axios": "^1.12.2", |     "axios": "^1.12.2", | ||||||
|     "class-variance-authority": "^0.7.1", |     "class-variance-authority": "^0.7.1", | ||||||
|     "clsx": "^2.1.1", |     "clsx": "^2.1.1", | ||||||
|     "i18next": "^25.5.3", |     "i18next": "^25.6.0", | ||||||
|     "i18next-browser-languagedetector": "^8.2.0", |     "i18next-browser-languagedetector": "^8.2.0", | ||||||
|     "i18next-resources-to-backend": "^1.2.1", |     "i18next-resources-to-backend": "^1.2.1", | ||||||
|     "input-otp": "^1.4.2", |     "input-otp": "^1.4.2", | ||||||
|     "lucide-react": "^0.544.0", |     "lucide-react": "^0.545.0", | ||||||
|     "next-themes": "^0.4.6", |     "next-themes": "^0.4.6", | ||||||
|     "react": "^19.2.0", |     "react": "^19.2.0", | ||||||
|     "react-dom": "^19.2.0", |     "react-dom": "^19.2.0", | ||||||
|     "react-hook-form": "^7.63.0", |     "react-hook-form": "^7.65.0", | ||||||
|     "react-i18next": "^15.7.3", |     "react-i18next": "^16.0.1", | ||||||
|     "react-markdown": "^10.1.0", |     "react-markdown": "^10.1.0", | ||||||
|     "react-router": "^7.9.3", |     "react-router": "^7.9.4", | ||||||
|     "sonner": "^2.0.7", |     "sonner": "^2.0.7", | ||||||
|     "tailwind-merge": "^3.3.1", |     "tailwind-merge": "^3.3.1", | ||||||
|     "tailwindcss": "^4.1.14", |     "tailwindcss": "^4.1.14", | ||||||
|     "zod": "^4.1.11" |     "zod": "^4.1.12" | ||||||
|   }, |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@eslint/js": "^9.36.0", |     "@eslint/js": "^9.37.0", | ||||||
|     "@tanstack/eslint-plugin-query": "^5.91.0", |     "@tanstack/eslint-plugin-query": "^5.91.0", | ||||||
|     "@types/node": "^24.6.2", |     "@types/node": "^24.7.2", | ||||||
|     "@types/react": "^19.2.0", |     "@types/react": "^19.2.2", | ||||||
|     "@types/react-dom": "^19.2.0", |     "@types/react-dom": "^19.2.2", | ||||||
|     "@vitejs/plugin-react": "^5.0.4", |     "@vitejs/plugin-react": "^5.0.4", | ||||||
|     "eslint": "^9.36.0", |     "eslint": "^9.37.0", | ||||||
|     "eslint-plugin-react-hooks": "^5.2.0", |     "eslint-plugin-react-hooks": "^7.0.0", | ||||||
|     "eslint-plugin-react-refresh": "^0.4.23", |     "eslint-plugin-react-refresh": "^0.4.23", | ||||||
|     "globals": "^16.4.0", |     "globals": "^16.4.0", | ||||||
|     "prettier": "3.6.2", |     "prettier": "3.6.2", | ||||||
|     "tw-animate-css": "^1.4.0", |     "tw-animate-css": "^1.4.0", | ||||||
|     "typescript": "~5.9.3", |     "typescript": "~5.9.3", | ||||||
|     "typescript-eslint": "^8.45.0", |     "typescript-eslint": "^8.46.1", | ||||||
|     "vite": "^7.1.8" |     "vite": "^7.1.10" | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @@ -1,11 +1,15 @@ | |||||||
| import { useAppContext } from "@/context/app-context"; | import { useAppContext } from "@/context/app-context"; | ||||||
| import { LanguageSelector } from "../language/language"; | import { LanguageSelector } from "../language/language"; | ||||||
| import { Outlet } from "react-router"; | import { Outlet } from "react-router"; | ||||||
| import { useCallback, useState } from "react"; | import { useCallback, useEffect, useState } from "react"; | ||||||
| import { DomainWarning } from "../domain-warning/domain-warning"; | import { DomainWarning } from "../domain-warning/domain-warning"; | ||||||
|  |  | ||||||
| const BaseLayout = ({ children }: { children: React.ReactNode }) => { | const BaseLayout = ({ children }: { children: React.ReactNode }) => { | ||||||
|   const { backgroundImage } = useAppContext(); |   const { backgroundImage, title } = useAppContext(); | ||||||
|  |  | ||||||
|  |   useEffect(() => { | ||||||
|  |     document.title = title; | ||||||
|  |   }, [title]); | ||||||
|  |  | ||||||
|   return ( |   return ( | ||||||
|     <div |     <div | ||||||
|   | |||||||
							
								
								
									
										33
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								go.mod
									
									
									
									
									
								
							| @@ -8,7 +8,7 @@ require ( | |||||||
| 	github.com/cenkalti/backoff/v5 v5.0.3 | 	github.com/cenkalti/backoff/v5 v5.0.3 | ||||||
| 	github.com/gin-gonic/gin v1.11.0 | 	github.com/gin-gonic/gin v1.11.0 | ||||||
| 	github.com/glebarez/sqlite 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/golang-migrate/migrate/v4 v4.19.0 | ||||||
| 	github.com/google/go-querystring v1.1.0 | 	github.com/google/go-querystring v1.1.0 | ||||||
| 	github.com/google/uuid v1.6.0 | 	github.com/google/uuid v1.6.0 | ||||||
| @@ -18,7 +18,7 @@ require ( | |||||||
| 	github.com/spf13/viper v1.21.0 | 	github.com/spf13/viper v1.21.0 | ||||||
| 	github.com/traefik/paerser v0.2.2 | 	github.com/traefik/paerser v0.2.2 | ||||||
| 	github.com/weppos/publicsuffix-go v0.50.0 | 	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 | 	golang.org/x/exp v0.0.0-20250620022241-b7579e27df2b | ||||||
| 	gorm.io/gorm v1.31.0 | 	gorm.io/gorm v1.31.0 | ||||||
| 	gotest.tools/v3 v3.5.2 | 	gotest.tools/v3 v3.5.2 | ||||||
| @@ -45,17 +45,18 @@ require ( | |||||||
| 	github.com/moby/term v0.5.2 // indirect | 	github.com/moby/term v0.5.2 // indirect | ||||||
| 	github.com/ncruces/go-strftime v0.1.9 // indirect | 	github.com/ncruces/go-strftime v0.1.9 // indirect | ||||||
| 	github.com/quic-go/qpack v0.5.1 // 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/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 | 	github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect | ||||||
| 	go.opentelemetry.io/auto/sdk v1.1.0 // 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/exporters/otlp/otlptrace/otlptracehttp v1.34.0 // indirect | ||||||
| 	go.opentelemetry.io/otel/sdk v1.34.0 // indirect | 	go.opentelemetry.io/otel/sdk v1.34.0 // indirect | ||||||
| 	go.uber.org/mock v0.5.0 // indirect | 	go.uber.org/mock v0.5.0 // indirect | ||||||
| 	go.yaml.in/yaml/v3 v3.0.4 // indirect | 	go.yaml.in/yaml/v3 v3.0.4 // indirect | ||||||
| 	golang.org/x/mod v0.27.0 // indirect | 	golang.org/x/mod v0.28.0 // indirect | ||||||
| 	golang.org/x/term v0.35.0 // indirect | 	golang.org/x/term v0.36.0 // indirect | ||||||
| 	golang.org/x/tools v0.36.0 // indirect | 	golang.org/x/tools v0.37.0 // indirect | ||||||
| 	modernc.org/libc v1.66.3 // indirect | 	modernc.org/libc v1.66.3 // indirect | ||||||
| 	modernc.org/mathutil v1.7.1 // indirect | 	modernc.org/mathutil v1.7.1 // indirect | ||||||
| 	modernc.org/memory v1.11.0 // indirect | 	modernc.org/memory v1.11.0 // indirect | ||||||
| @@ -71,23 +72,23 @@ require ( | |||||||
| 	github.com/bytedance/sonic v1.14.0 // indirect | 	github.com/bytedance/sonic v1.14.0 // indirect | ||||||
| 	github.com/bytedance/sonic/loader v0.3.0 // indirect | 	github.com/bytedance/sonic/loader v0.3.0 // indirect | ||||||
| 	github.com/catppuccin/go v0.3.0 // indirect | 	github.com/catppuccin/go v0.3.0 // indirect | ||||||
| 	github.com/charmbracelet/bubbles v0.21.0 // indirect | 	github.com/charmbracelet/bubbles v0.21.1-0.20250623103423-23b8fd6302d7 // indirect | ||||||
| 	github.com/charmbracelet/bubbletea v1.3.4 // indirect | 	github.com/charmbracelet/bubbletea v1.3.6 // indirect | ||||||
| 	github.com/charmbracelet/huh v0.7.0 | 	github.com/charmbracelet/huh v0.8.0 | ||||||
| 	github.com/charmbracelet/lipgloss v1.1.0 // indirect | 	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/exp/strings v0.0.0-20240722160745-212f7b056ed0 // indirect | ||||||
| 	github.com/charmbracelet/x/term v0.2.1 // indirect | 	github.com/charmbracelet/x/term v0.2.1 // indirect | ||||||
| 	github.com/cloudwego/base64x v0.1.6 // indirect | 	github.com/cloudwego/base64x v0.1.6 // indirect | ||||||
| 	github.com/distribution/reference v0.6.0 // indirect | 	github.com/distribution/reference v0.6.0 // indirect | ||||||
| 	github.com/docker/docker v28.5.0+incompatible | 	github.com/docker/docker v28.5.1+incompatible | ||||||
| 	github.com/docker/go-connections v0.5.0 // indirect | 	github.com/docker/go-connections v0.5.0 // indirect | ||||||
| 	github.com/docker/go-units v0.5.0 // indirect | 	github.com/docker/go-units v0.5.0 // indirect | ||||||
| 	github.com/dustin/go-humanize v1.0.1 // indirect | 	github.com/dustin/go-humanize v1.0.1 // indirect | ||||||
| 	github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect | 	github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect | ||||||
| 	github.com/felixge/httpsnoop v1.0.4 // indirect | 	github.com/felixge/httpsnoop v1.0.4 // indirect | ||||||
| 	github.com/fsnotify/fsnotify v1.9.0 // 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/gin-contrib/sse v1.1.0 // indirect | ||||||
| 	github.com/go-ldap/ldap/v3 v3.4.12 | 	github.com/go-ldap/ldap/v3 v3.4.12 | ||||||
| 	github.com/go-logr/logr v1.4.3 // indirect | 	github.com/go-logr/logr v1.4.3 // indirect | ||||||
| @@ -130,10 +131,10 @@ require ( | |||||||
| 	go.opentelemetry.io/otel/metric v1.37.0 // indirect | 	go.opentelemetry.io/otel/metric v1.37.0 // indirect | ||||||
| 	go.opentelemetry.io/otel/trace v1.37.0 // indirect | 	go.opentelemetry.io/otel/trace v1.37.0 // indirect | ||||||
| 	golang.org/x/arch v0.20.0 // indirect | 	golang.org/x/arch v0.20.0 // indirect | ||||||
| 	golang.org/x/net v0.44.0 // indirect | 	golang.org/x/net v0.45.0 // indirect | ||||||
| 	golang.org/x/oauth2 v0.31.0 | 	golang.org/x/oauth2 v0.32.0 | ||||||
| 	golang.org/x/sync v0.17.0 // indirect | 	golang.org/x/sync v0.17.0 // indirect | ||||||
| 	golang.org/x/sys v0.36.0 // indirect | 	golang.org/x/sys v0.37.0 // indirect | ||||||
| 	golang.org/x/text v0.29.0 // indirect | 	golang.org/x/text v0.30.0 // indirect | ||||||
| 	google.golang.org/protobuf v1.36.9 // indirect | 	google.golang.org/protobuf v1.36.9 // indirect | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										70
									
								
								go.sum
									
									
									
									
									
								
							| @@ -12,8 +12,8 @@ github.com/atotto/clipboard v0.1.4 h1:EH0zSVneZPSuFR11BlR9YppQTVDbh5+16AmcJi4g1z | |||||||
| github.com/atotto/clipboard v0.1.4/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI= | 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 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k= | ||||||
| github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8= | 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.3.1 h1:LV+qyBQ2pqe0u42ZsUEtPiCaUoqgA9gYRDs3vj1nolY= | ||||||
| github.com/aymanbagabas/go-udiff v0.2.0/go.mod h1:RE4Ex0qsGkTAJoQdQQCA0uG+nAzJO/pI/QwceO5fgrA= | 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.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 h1:79yrbttoZrLGkL/oOI8hBrUKucwOL0oOjUgEguGMcJ4= | ||||||
| github.com/boombuler/barcode v1.0.2/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8= | 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/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 h1:ZN+IMa753KfX5hd8vVaMixjnqRZ3y8CuJKRKj1xcsSM= | ||||||
| github.com/cenkalti/backoff/v5 v5.0.3/go.mod h1:rkhZdG3JZukswDf7f0cwqPNk4K0sa+F97BxZthm/crw= | 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.1-0.20250623103423-23b8fd6302d7 h1:JFgG/xnwFfbezlUnFMJy0nusZvytYysV4SCS2cYbvws= | ||||||
| github.com/charmbracelet/bubbles v0.21.0/go.mod h1:HF+v6QUR4HkEpz62dx7ym2xc71/KBHg+zKwJtMw+qtg= | github.com/charmbracelet/bubbles v0.21.1-0.20250623103423-23b8fd6302d7/go.mod h1:ISC1gtLcVilLOf23wvTfoQuYbW2q0JevFxPfUzZ9Ybw= | ||||||
| github.com/charmbracelet/bubbletea v1.3.4 h1:kCg7B+jSCFPLYRA52SDZjr51kG/fMUEoPoZrkaDHyoI= | github.com/charmbracelet/bubbletea v1.3.6 h1:VkHIxPJQeDt0aFJIsVxw8BQdh/F/L2KKZGsK6et5taU= | ||||||
| github.com/charmbracelet/bubbletea v1.3.4/go.mod h1:dtcUCyCGEX3g9tosuYiut3MXgY/Jsv9nKVdibKKRRXo= | 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 h1:4pZI35227imm7yK2bGPcfpFEmuY1gc2YSTShr4iJBfs= | ||||||
| github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc/go.mod h1:X4/0JoqgTIPSFcRA/P6INZzIuyqdFY5rm8tb41s9okk= | 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.8.0 h1:Xz/Pm2h64cXQZn/Jvele4J3r7DDiqFCNIVteYukxDvY= | ||||||
| github.com/charmbracelet/huh v0.7.0/go.mod h1:UGC3DZHlgOKHvHC07a5vHag41zzhpPFj34U92sOmyuk= | 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 h1:vYXsiLHVkK7fp74RkV7b2kq9+zDLoEU4MZoFqR/noCY= | ||||||
| github.com/charmbracelet/lipgloss v1.1.0/go.mod h1:/6Q8FR2o+kj8rz4Dq0zQc3vYf7X+B0binUUBwA0aL30= | 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.9.3 h1:BXt5DHS/MKF+LjuK4huWrC6NCvHtexww7dMayh6GXd0= | ||||||
| github.com/charmbracelet/x/ansi v0.8.0/go.mod h1:wdYl/ONOLHLIVmQaxbIYEC/cRKOQyjTkowiI4blgS9Q= | 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 h1:/KBBKHuVRbq1lYx5BzEHBAFBP8VcQzJejZ/IA3iR28k= | ||||||
| github.com/charmbracelet/x/cellbuf v0.0.13/go.mod h1:xe0nKWGd3eJgtqZRaN9RjMtK7xUYchjzPr7q6kcvCCs= | github.com/charmbracelet/x/cellbuf v0.0.13/go.mod h1:xe0nKWGd3eJgtqZRaN9RjMtK7xUYchjzPr7q6kcvCCs= | ||||||
| github.com/charmbracelet/x/conpty v0.1.0 h1:4zc8KaIcbiL4mghEON8D72agYtSeIgq8FSThSPQIb+U= | 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/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 h1:0IXCQ5g4/QMHHkarYzh5l+u8T3t73zM5QvfrDyIgxBk= | ||||||
| github.com/distribution/reference v0.6.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E= | github.com/distribution/reference v0.6.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E= | ||||||
| github.com/docker/docker v28.5.0+incompatible h1:ZdSQoRUE9XxhFI/B8YLvhnEFMmYN9Pp8Egd2qcaFk1E= | github.com/docker/docker v28.5.1+incompatible h1:Bm8DchhSD2J6PsFzxC35TZo4TLGR2PdW/E69rU45NhM= | ||||||
| github.com/docker/docker v28.5.0+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk= | 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 h1:USnMq7hx7gwdVZq1L49hLXaFtUdTADjXGp+uj1Br63c= | ||||||
| github.com/docker/go-connections v0.5.0/go.mod h1:ov60Kzw0kKElRwhNs9UlUHAE/F9Fe6GLaXnqyDdmEXc= | 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= | 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/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 h1:2Ml+OJNzbYCTzsxtv8vKSFD9PbJjmhYF14k/jKC7S9k= | ||||||
| github.com/fsnotify/fsnotify v1.9.0/go.mod h1:8jBTzvmWwFyi3Pb8djgCCO5IBqzKJ/Jwo8TRcHyHii0= | 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.10 h1:zyueNbySn/z8mJZHLt6IPw0KoZsiQNszIpU+bX4+ZK0= | ||||||
| github.com/gabriel-vasile/mimetype v1.4.8/go.mod h1:ByKUIKGjh1ODkGM1asKUbQZOLGrPjydw3hYPU2YU9t8= | 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 h1:n0w2GMuUpWDVp7qSpvze6fAu9iRxJY4Hmj6AmBOU05w= | ||||||
| github.com/gin-contrib/sse v1.1.0/go.mod h1:hxRZ5gVpWMT7Z0B0gSNYqqsSCNIJMjzvm6fqCz9vjwM= | github.com/gin-contrib/sse v1.1.0/go.mod h1:hxRZ5gVpWMT7Z0B0gSNYqqsSCNIJMjzvm6fqCz9vjwM= | ||||||
| github.com/gin-gonic/gin v1.11.0 h1:OW/6PLjyusp2PPXtyxKHU0RbX6I/l28FTdDlae5ueWk= | github.com/gin-gonic/gin v1.11.0 h1:OW/6PLjyusp2PPXtyxKHU0RbX6I/l28FTdDlae5ueWk= | ||||||
| @@ -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/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 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY= | ||||||
| github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY= | 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.28.0 h1:Q7ibns33JjyW48gHkuFT91qX48KG0ktULL6FgHdG688= | ||||||
| github.com/go-playground/validator/v10 v10.27.0/go.mod h1:I5QpIEbmr8On7W0TktmJAumgzX4CA1XNl4ZmDuVHKKo= | 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 h1:EBsztssimR/CONLSZZ04E8qAkxNYq4Qp9LvH92wZUgs= | ||||||
| github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= | github.com/go-viper/mapstructure/v2 v2.4.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= | ||||||
| github.com/goccy/go-json v0.10.4 h1:JSwxQzIqKfmFX1swYPpUThQZp/Ka4wzJdK0LWVytLPM= | 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/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 h1:giqksBPnT/HDtZ6VhtFKgoLOWmlyo9Ei6u9PqzIMbhI= | ||||||
| github.com/quic-go/qpack v0.5.1/go.mod h1:+PC4XFrEskIVkcLzpEkbLqq1uCoxPhQuvK5rH1ZgaEg= | 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.1 h1:4ZAWm0AhCb6+hE+l5Q1NAL0iRn/ZrMwqHRGQiFwj2eg= | ||||||
| github.com/quic-go/quic-go v0.54.0/go.mod h1:e68ZEaCdyviluZmy44P6Iey98v/Wfz6HCjQEm+l8zTY= | 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 h1:W09IVJc94icq4NjY3clb7Lk8O1qJ8BdBEF8z0ibU0rE= | ||||||
| github.com/remyoudompheng/bigfft v0.0.0-20230129092748-24d4a6f8daec/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= | 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= | 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/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 h1:x5S+0EU27Lbphp4UKm1C+1oQO+rKx36vfCoaVebLFSU= | ||||||
| github.com/spf13/viper v1.21.0/go.mod h1:P0lhsswPGWD/1lZJ9ny3fYnVqxiegrlNrEmgLjbTCAY= | 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.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.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= | ||||||
| github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= | 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= | 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 h1:dx1zTU0MAE98U+TQ8BLl7XsJbgze2WnNKF/8tGp/Q6c= | ||||||
| golang.org/x/arch v0.20.0/go.mod h1:bdwinDaKcfZUGpH09BB7ZmOfhalA8lQdzl62l8gGWsk= | 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.43.0 h1:dduJYIi3A3KOfdGOHX8AVZ/jGiyPa3IbBozJ5kNuE04= | ||||||
| golang.org/x/crypto v0.42.0/go.mod h1:4+rDnOTJhQCx2q7/j6rAN5XDw8kPjeaXEUR2eL94ix8= | 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 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/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.28.0 h1:gQBtGhjxykdjY9YhZpSlZIsbnaE2+PgjfLWUQTnoZ1U= | ||||||
| golang.org/x/mod v0.27.0/go.mod h1:rWI627Fq0DEoudcK+MBkNkCe0EetEaDSwJJkCcjpazc= | golang.org/x/mod v0.28.0/go.mod h1:yfB/L0NOf/kmEbXjzCPOx1iK1fRutOydrCMsqRhEBxI= | ||||||
| golang.org/x/net v0.44.0 h1:evd8IRDyfNBMBTTY5XRF1vaZlD+EmWx6x8PkhR04H/I= | golang.org/x/net v0.45.0 h1:RLBg5JKixCy82FtLJpeNlVM0nrSqpCRYzVU1n8kj0tM= | ||||||
| golang.org/x/net v0.44.0/go.mod h1:ECOoLqd5U3Lhyeyo/QDCEVQ4sNgYsqvCZ722XogGieY= | golang.org/x/net v0.45.0/go.mod h1:ECOoLqd5U3Lhyeyo/QDCEVQ4sNgYsqvCZ722XogGieY= | ||||||
| golang.org/x/oauth2 v0.31.0 h1:8Fq0yVZLh4j4YA47vHKFTa9Ew5XIrCP8LC6UeNZnLxo= | golang.org/x/oauth2 v0.32.0 h1:jsCblLleRMDrxMN29H3z/k1KliIvpLgCkE6R8FXXNgY= | ||||||
| golang.org/x/oauth2 v0.31.0/go.mod h1:lzm5WQJQwKZ3nwavOZ3IS5Aulzxi68dUSgRHujetwEA= | 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 h1:l60nONMj9l5drqw6jlhIELNv9I0A4OFgRsG9k2oT9Ug= | ||||||
| golang.org/x/sync v0.17.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI= | 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-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.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.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= | ||||||
| golang.org/x/sys v0.12.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.37.0 h1:fdNQudmxPjkdUTPnLn5mdQv7Zwvbvpaxqs831goi9kQ= | ||||||
| golang.org/x/sys v0.36.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= | golang.org/x/sys v0.37.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks= | ||||||
| golang.org/x/term v0.35.0 h1:bZBVKBudEyhRcajGcNc3jIfWPqV4y/Kt2XcoigOWtDQ= | golang.org/x/term v0.36.0 h1:zMPR+aF8gfksFprF/Nc/rd1wRS1EI6nDBGyWAvDzx2Q= | ||||||
| golang.org/x/term v0.35.0/go.mod h1:TPGtkTLesOwf2DE8CgVYiZinHAOuy5AYUYT1lENIZnA= | golang.org/x/term v0.36.0/go.mod h1:Qu394IJq6V6dCBRgwqshf3mPF85AqzYEzofzRdZkWss= | ||||||
| golang.org/x/text v0.29.0 h1:1neNs90w9YzJ9BocxfsQNHKuAT4pkghyXc4nhZ6sJvk= | golang.org/x/text v0.30.0 h1:yznKA/E9zq54KzlzBEAWn1NXSQ8DIp/NYMy88xJjl4k= | ||||||
| golang.org/x/text v0.29.0/go.mod h1:7MhJOA9CD2qZyOKYazxdYMF85OwPdEr9jTtBpO7ydH4= | 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 h1:o7cqy6amK/52YcAKIPlM3a+Fpj35zvRj2TP+e1xFSfk= | ||||||
| golang.org/x/time v0.5.0/go.mod h1:3BpzKBy/shNhVucY/MWOyx10tF3SFh9QdLuxbVysPQM= | 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.37.0 h1:DVSRzp7FwePZW356yEAChSdNcQo6Nsp+fex1SUW09lE= | ||||||
| golang.org/x/tools v0.36.0/go.mod h1:WBDiHKJK8YgLHlcQPYQzNCkUxUypCaa5ZegCVutKm+s= | 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= | 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 v0.0.0-20240213162025-012b6fc9bca9 h1:9+tzLLstTlPTRyJTh+ah5wIMsBW5c4tQwGTN3thOW9Y= | ||||||
| google.golang.org/genproto/googleapis/api v0.0.0-20250115164207-1a7da9e5054f h1:gap6+3Gk41EItBuyi4XX/bp4oqJ3UwuIMl25yGinuAA= | google.golang.org/genproto/googleapis/api v0.0.0-20250115164207-1a7da9e5054f h1:gap6+3Gk41EItBuyi4XX/bp4oqJ3UwuIMl25yGinuAA= | ||||||
|   | |||||||
| @@ -2,21 +2,25 @@ package bootstrap | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
|  | 	"context" | ||||||
| 	"encoding/json" | 	"encoding/json" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"net/url" | 	"net/url" | ||||||
| 	"os" | 	"os" | ||||||
|  | 	"sort" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| 	"tinyauth/internal/config" | 	"tinyauth/internal/config" | ||||||
| 	"tinyauth/internal/controller" | 	"tinyauth/internal/controller" | ||||||
| 	"tinyauth/internal/middleware" | 	"tinyauth/internal/middleware" | ||||||
|  | 	"tinyauth/internal/model" | ||||||
| 	"tinyauth/internal/service" | 	"tinyauth/internal/service" | ||||||
| 	"tinyauth/internal/utils" | 	"tinyauth/internal/utils" | ||||||
|  |  | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
|  | 	"gorm.io/gorm" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| type Controller interface { | type Controller interface { | ||||||
| @@ -74,6 +78,15 @@ func (app *BootstrapApp) Setup() error { | |||||||
| 	csrfCookieName := fmt.Sprintf("%s-%s", config.CSRFCookieName, cookieId) | 	csrfCookieName := fmt.Sprintf("%s-%s", config.CSRFCookieName, cookieId) | ||||||
| 	redirectCookieName := fmt.Sprintf("%s-%s", config.RedirectCookieName, 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 | 	// Create configs | ||||||
| 	authConfig := service.AuthServiceConfig{ | 	authConfig := service.AuthServiceConfig{ | ||||||
| 		Users:             users, | 		Users:             users, | ||||||
| @@ -126,12 +139,14 @@ func (app *BootstrapApp) Setup() error { | |||||||
|  |  | ||||||
| 	// Create services | 	// Create services | ||||||
| 	dockerService := service.NewDockerService() | 	dockerService := service.NewDockerService() | ||||||
|  | 	aclsService := service.NewAccessControlsService(dockerService) | ||||||
| 	authService := service.NewAuthService(authConfig, dockerService, ldapService, database) | 	authService := service.NewAuthService(authConfig, dockerService, ldapService, database) | ||||||
| 	oauthBrokerService := service.NewOAuthBrokerService(oauthProviders) | 	oauthBrokerService := service.NewOAuthBrokerService(oauthProviders) | ||||||
|  |  | ||||||
| 	// Initialize services | 	// Initialize services (order matters) | ||||||
| 	services := []Service{ | 	services := []Service{ | ||||||
| 		dockerService, | 		dockerService, | ||||||
|  | 		aclsService, | ||||||
| 		authService, | 		authService, | ||||||
| 		oauthBrokerService, | 		oauthBrokerService, | ||||||
| 	} | 	} | ||||||
| @@ -150,18 +165,6 @@ func (app *BootstrapApp) Setup() error { | |||||||
| 	configuredProviders := make([]controller.Provider, 0) | 	configuredProviders := make([]controller.Provider, 0) | ||||||
|  |  | ||||||
| 	for id, provider := range oauthProviders { | 	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{ | 		configuredProviders = append(configuredProviders, controller.Provider{ | ||||||
| 			Name:  provider.Name, | 			Name:  provider.Name, | ||||||
| 			ID:    id, | 			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 { | 	if authService.UserAuthConfigured() || ldapService != nil { | ||||||
| 		configuredProviders = append(configuredProviders, controller.Provider{ | 		configuredProviders = append(configuredProviders, controller.Provider{ | ||||||
| 			Name:  "Username", | 			Name:  "Username", | ||||||
| @@ -184,11 +191,8 @@ func (app *BootstrapApp) Setup() error { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Create engine | 	// Create engine | ||||||
| 	if config.Version != "development" { |  | ||||||
| 		gin.SetMode(gin.ReleaseMode) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	engine := gin.New() | 	engine := gin.New() | ||||||
|  | 	engine.Use(gin.Recovery()) | ||||||
|  |  | ||||||
| 	if len(app.config.TrustedProxies) > 0 { | 	if len(app.config.TrustedProxies) > 0 { | ||||||
| 		err := engine.SetTrustedProxies(strings.Split(app.config.TrustedProxies, ",")) | 		err := engine.SetTrustedProxies(strings.Split(app.config.TrustedProxies, ",")) | ||||||
| @@ -244,7 +248,7 @@ func (app *BootstrapApp) Setup() error { | |||||||
|  |  | ||||||
| 	proxyController := controller.NewProxyController(controller.ProxyControllerConfig{ | 	proxyController := controller.NewProxyController(controller.ProxyControllerConfig{ | ||||||
| 		AppURL: app.config.AppURL, | 		AppURL: app.config.AppURL, | ||||||
| 	}, apiRouter, dockerService, authService) | 	}, apiRouter, aclsService, authService) | ||||||
|  |  | ||||||
| 	userController := controller.NewUserController(controller.UserControllerConfig{ | 	userController := controller.NewUserController(controller.UserControllerConfig{ | ||||||
| 		CookieDomain: cookieDomain, | 		CookieDomain: cookieDomain, | ||||||
| @@ -278,6 +282,10 @@ func (app *BootstrapApp) Setup() error { | |||||||
| 		go app.heartbeat() | 		go app.heartbeat() | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	// Start DB cleanup routine | ||||||
|  | 	log.Debug().Msg("Starting database cleanup routine") | ||||||
|  | 	go app.dbCleanup(database) | ||||||
|  |  | ||||||
| 	// Start server | 	// Start server | ||||||
| 	address := fmt.Sprintf("%s:%d", app.config.Address, app.config.Port) | 	address := fmt.Sprintf("%s:%d", app.config.Address, app.config.Port) | ||||||
| 	log.Info().Msgf("Starting server on %s", address) | 	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 { | type OAuthServiceConfig struct { | ||||||
| 	ClientID           string   `key:"client-id"` | 	ClientID           string `field:"client-id"` | ||||||
| 	ClientSecret       string   `key:"client-secret"` | 	ClientSecret       string | ||||||
| 	ClientSecretFile   string   `key:"client-secret-file"` | 	ClientSecretFile   string | ||||||
| 	Scopes             []string `key:"scopes"` | 	Scopes             []string | ||||||
| 	RedirectURL        string   `key:"redirect-url"` | 	RedirectURL        string `field:"redirect-url"` | ||||||
| 	AuthURL            string   `key:"auth-url"` | 	AuthURL            string `field:"auth-url"` | ||||||
| 	TokenURL           string   `key:"token-url"` | 	TokenURL           string `field:"token-url"` | ||||||
| 	UserinfoURL        string   `key:"user-info-url"` | 	UserinfoURL        string `field:"user-info-url"` | ||||||
| 	InsecureSkipVerify bool     `key:"insecure-skip-verify"` | 	InsecureSkipVerify bool | ||||||
| 	Name               string   `key:"name"` | 	Name               string | ||||||
| } | } | ||||||
|  |  | ||||||
| var OverrideProviders = map[string]string{ | var OverrideProviders = map[string]string{ | ||||||
|   | |||||||
| @@ -13,8 +13,8 @@ func NewHealthController(router *gin.RouterGroup) *HealthController { | |||||||
| } | } | ||||||
|  |  | ||||||
| func (controller *HealthController) SetupRoutes() { | func (controller *HealthController) SetupRoutes() { | ||||||
| 	controller.router.GET("/health", controller.healthHandler) | 	controller.router.GET("/healthz", controller.healthHandler) | ||||||
| 	controller.router.HEAD("/health", controller.healthHandler) | 	controller.router.HEAD("/healthz", controller.healthHandler) | ||||||
| } | } | ||||||
|  |  | ||||||
| func (controller *HealthController) healthHandler(c *gin.Context) { | func (controller *HealthController) healthHandler(c *gin.Context) { | ||||||
|   | |||||||
| @@ -72,6 +72,7 @@ func (controller *OAuthController) oauthURLHandler(c *gin.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	service.GenerateVerifier() | ||||||
| 	state := service.GenerateState() | 	state := service.GenerateState() | ||||||
| 	authURL := service.GetAuthURL(state) | 	authURL := service.GetAuthURL(state) | ||||||
| 	c.SetCookie(controller.config.CSRFCookieName, state, int(time.Hour.Seconds()), "/", fmt.Sprintf(".%s", controller.config.CookieDomain), controller.config.SecureCookie, true) | 	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 | 	var name string | ||||||
|  |  | ||||||
| 	if user.Name != "" { | 	if strings.TrimSpace(user.Name) != "" { | ||||||
| 		log.Debug().Msg("Using name from OAuth provider") | 		log.Debug().Msg("Using name from OAuth provider") | ||||||
| 		name = user.Name | 		name = user.Name | ||||||
| 	} else { | 	} else { | ||||||
| @@ -172,7 +173,7 @@ func (controller *OAuthController) oauthCallbackHandler(c *gin.Context) { | |||||||
|  |  | ||||||
| 	var username string | 	var username string | ||||||
|  |  | ||||||
| 	if user.PreferredUsername != "" { | 	if strings.TrimSpace(user.PreferredUsername) != "" { | ||||||
| 		log.Debug().Msg("Using preferred username from OAuth provider") | 		log.Debug().Msg("Using preferred username from OAuth provider") | ||||||
| 		username = user.PreferredUsername | 		username = user.PreferredUsername | ||||||
| 	} else { | 	} else { | ||||||
|   | |||||||
| @@ -24,15 +24,15 @@ type ProxyControllerConfig struct { | |||||||
| type ProxyController struct { | type ProxyController struct { | ||||||
| 	config ProxyControllerConfig | 	config ProxyControllerConfig | ||||||
| 	router *gin.RouterGroup | 	router *gin.RouterGroup | ||||||
| 	docker *service.DockerService | 	acls   *service.AccessControlsService | ||||||
| 	auth   *service.AuthService | 	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{ | 	return &ProxyController{ | ||||||
| 		config: config, | 		config: config, | ||||||
| 		router: router, | 		router: router, | ||||||
| 		docker: docker, | 		acls:   acls, | ||||||
| 		auth:   auth, | 		auth:   auth, | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @@ -76,18 +76,21 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) { | |||||||
| 	proto := c.Request.Header.Get("X-Forwarded-Proto") | 	proto := c.Request.Header.Get("X-Forwarded-Proto") | ||||||
| 	host := c.Request.Header.Get("X-Forwarded-Host") | 	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 { | 	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) | 		controller.handleError(c, req, isBrowser) | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	log.Trace().Interface("acls", acls).Msg("ACLs for resource") | ||||||
|  |  | ||||||
| 	clientIP := c.ClientIP() | 	clientIP := c.ClientIP() | ||||||
|  |  | ||||||
| 	if controller.auth.IsBypassedIP(labels.IP, clientIP) { | 	if controller.auth.IsBypassedIP(acls.IP, clientIP) { | ||||||
| 		controller.setHeaders(c, labels) | 		controller.setHeaders(c, acls) | ||||||
| 		c.JSON(200, gin.H{ | 		c.JSON(200, gin.H{ | ||||||
| 			"status":  200, | 			"status":  200, | ||||||
| 			"message": "Authenticated", | 			"message": "Authenticated", | ||||||
| @@ -95,7 +98,7 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	authEnabled, err := controller.auth.IsAuthEnabled(uri, labels.Path) | 	authEnabled, err := controller.auth.IsAuthEnabled(uri, acls.Path) | ||||||
|  |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		log.Error().Err(err).Msg("Failed to check if auth is enabled for resource") | 		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 { | 	if !authEnabled { | ||||||
| 		log.Debug().Msg("Authentication disabled for resource, allowing access") | 		log.Debug().Msg("Authentication disabled for resource, allowing access") | ||||||
| 		controller.setHeaders(c, labels) | 		controller.setHeaders(c, acls) | ||||||
| 		c.JSON(200, gin.H{ | 		c.JSON(200, gin.H{ | ||||||
| 			"status":  200, | 			"status":  200, | ||||||
| 			"message": "Authenticated", | 			"message": "Authenticated", | ||||||
| @@ -113,7 +116,7 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if !controller.auth.CheckIP(labels.IP, clientIP) { | 	if !controller.auth.CheckIP(acls.IP, clientIP) { | ||||||
| 		if req.Proxy == "nginx" || !isBrowser { | 		if req.Proxy == "nginx" || !isBrowser { | ||||||
| 			c.JSON(401, gin.H{ | 			c.JSON(401, gin.H{ | ||||||
| 				"status":  401, | 				"status":  401, | ||||||
| @@ -150,13 +153,15 @@ func (controller *ProxyController) proxyHandler(c *gin.Context) { | |||||||
| 		userContext = context | 		userContext = context | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	log.Trace().Interface("context", userContext).Msg("User context from request") | ||||||
|  |  | ||||||
| 	if userContext.Provider == "basic" && userContext.TotpEnabled { | 	if userContext.Provider == "basic" && userContext.TotpEnabled { | ||||||
| 		log.Debug().Msg("User has TOTP enabled, denying basic auth access") | 		log.Debug().Msg("User has TOTP enabled, denying basic auth access") | ||||||
| 		userContext.IsLoggedIn = false | 		userContext.IsLoggedIn = false | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if userContext.IsLoggedIn { | 	if userContext.IsLoggedIn { | ||||||
| 		appAllowed := controller.auth.IsResourceAllowed(c, userContext, labels) | 		appAllowed := controller.auth.IsResourceAllowed(c, userContext, acls) | ||||||
|  |  | ||||||
| 		if !appAllowed { | 		if !appAllowed { | ||||||
| 			log.Warn().Str("user", userContext.Username).Str("resource", strings.Split(host, ".")[0]).Msg("User not allowed to access resource") | 			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 { | 		if userContext.OAuth { | ||||||
| 			groupOK := controller.auth.IsInOAuthGroup(c, userContext, labels.OAuth.Groups) | 			groupOK := controller.auth.IsInOAuthGroup(c, userContext, acls.OAuth.Groups) | ||||||
|  |  | ||||||
| 			if !groupOK { | 			if !groupOK { | ||||||
| 				log.Warn().Str("user", userContext.Username).Str("resource", strings.Split(host, ".")[0]).Msg("User OAuth groups do not match resource requirements") | 				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-Email", utils.SanitizeHeader(userContext.Email)) | ||||||
| 		c.Header("Remote-Groups", utils.SanitizeHeader(userContext.OAuthGroups)) | 		c.Header("Remote-Groups", utils.SanitizeHeader(userContext.OAuthGroups)) | ||||||
|  |  | ||||||
| 		controller.setHeaders(c, labels) | 		controller.setHeaders(c, acls) | ||||||
|  |  | ||||||
| 		c.JSON(200, gin.H{ | 		c.JSON(200, gin.H{ | ||||||
| 			"status":  200, | 			"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())) | 	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")) | 	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 { | 	for key, value := range headers { | ||||||
| 		log.Debug().Str("header", key).Msg("Setting header") | 		log.Debug().Str("header", key).Msg("Setting header") | ||||||
| 		c.Header(key, value) | 		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 != "" { | 	if acls.Response.BasicAuth.Username != "" && basicPassword != "" { | ||||||
| 		log.Debug().Str("username", labels.Response.BasicAuth.Username).Msg("Setting basic auth header") | 		log.Debug().Str("username", acls.Response.BasicAuth.Username).Msg("Setting basic auth header") | ||||||
| 		c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Response.BasicAuth.Username, basicPassword))) | 		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()) | 	assert.NilError(t, dockerService.Init()) | ||||||
|  |  | ||||||
|  | 	// Access controls | ||||||
|  | 	accessControlsService := service.NewAccessControlsService(dockerService) | ||||||
|  |  | ||||||
|  | 	assert.NilError(t, accessControlsService.Init()) | ||||||
|  |  | ||||||
| 	// Auth service | 	// Auth service | ||||||
| 	authService := service.NewAuthService(service.AuthServiceConfig{ | 	authService := service.NewAuthService(service.AuthServiceConfig{ | ||||||
| 		Users: []config.User{ | 		Users: []config.User{ | ||||||
| @@ -59,7 +64,7 @@ func setupProxyController(t *testing.T, middlewares *[]gin.HandlerFunc) (*gin.En | |||||||
| 	// Controller | 	// Controller | ||||||
| 	ctrl := controller.NewProxyController(controller.ProxyControllerConfig{ | 	ctrl := controller.NewProxyController(controller.ProxyControllerConfig{ | ||||||
| 		AppURL: "http://localhost:8080", | 		AppURL: "http://localhost:8080", | ||||||
| 	}, group, dockerService, authService) | 	}, group, accessControlsService, authService) | ||||||
| 	ctrl.SetupRoutes() | 	ctrl.SetupRoutes() | ||||||
|  |  | ||||||
| 	return router, recorder, authService | 	return router, recorder, authService | ||||||
|   | |||||||
| @@ -1,10 +1,12 @@ | |||||||
| package middleware | package middleware | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"io/fs" | 	"io/fs" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
| 	"strings" | 	"strings" | ||||||
|  | 	"time" | ||||||
| 	"tinyauth/internal/assets" | 	"tinyauth/internal/assets" | ||||||
|  |  | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| @@ -27,14 +29,16 @@ func (m *UIMiddleware) Init() error { | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	m.uiFs = ui | 	m.uiFs = ui | ||||||
| 	m.uiFileServer = http.FileServer(http.FS(ui)) | 	m.uiFileServer = http.FileServerFS(ui) | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| func (m *UIMiddleware) Middleware() gin.HandlerFunc { | func (m *UIMiddleware) Middleware() gin.HandlerFunc { | ||||||
| 	return func(c *gin.Context) { | 	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": | 		case "api": | ||||||
| 			c.Next() | 			c.Next() | ||||||
| 			return | 			return | ||||||
| @@ -42,12 +46,19 @@ func (m *UIMiddleware) Middleware() gin.HandlerFunc { | |||||||
| 			c.Next() | 			c.Next() | ||||||
| 			return | 			return | ||||||
| 		default: | 		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) { | 			if os.IsNotExist(err) { | ||||||
| 				c.Request.URL.Path = "/" | 				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) | 			m.uiFileServer.ServeHTTP(c.Writer, c.Request) | ||||||
| 			c.Abort() | 			c.Abort() | ||||||
| 			return | 			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 | package service | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"regexp" | 	"regexp" | ||||||
| 	"strings" | 	"strings" | ||||||
| @@ -41,6 +43,7 @@ type AuthService struct { | |||||||
| 	loginMutex    sync.RWMutex | 	loginMutex    sync.RWMutex | ||||||
| 	ldap          *LdapService | 	ldap          *LdapService | ||||||
| 	database      *gorm.DB | 	database      *gorm.DB | ||||||
|  | 	ctx           context.Context | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewAuthService(config AuthServiceConfig, docker *DockerService, ldap *LdapService, database *gorm.DB) *AuthService { | 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 { | func (auth *AuthService) Init() error { | ||||||
|  | 	auth.ctx = context.Background() | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -213,7 +217,7 @@ func (auth *AuthService) CreateSessionCookie(c *gin.Context, data *config.Sessio | |||||||
| 		OAuthName:   data.OAuthName, | 		OAuthName:   data.OAuthName, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err = auth.database.Create(&session).Error | 	err = gorm.G[model.Session](auth.database).Create(auth.ctx, &session) | ||||||
|  |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| @@ -231,10 +235,10 @@ func (auth *AuthService) DeleteSessionCookie(c *gin.Context) error { | |||||||
| 		return err | 		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 { | 	if err != nil { | ||||||
| 		return res.Error | 		return err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	c.SetCookie(auth.config.SessionCookieName, "", -1, "/", fmt.Sprintf(".%s", auth.config.CookieDomain), auth.config.SecureCookie, true) | 	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 | 		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 err != nil { | ||||||
|  | 		return config.SessionCookie{}, err | ||||||
| 	if res.Error != nil { |  | ||||||
| 		return config.SessionCookie{}, res.Error |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if res.RowsAffected == 0 { | 	if errors.Is(err, gorm.ErrRecordNotFound) { | ||||||
| 		return config.SessionCookie{}, fmt.Errorf("session not found") | 		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 | 	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 { | 	if context.OAuth { | ||||||
| 		log.Debug().Msg("Checking OAuth whitelist") | 		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") | 		log.Debug().Msg("Checking blocked users") | ||||||
| 		if utils.CheckFilter(labels.Users.Block, context.Username) { | 		if utils.CheckFilter(acls.Users.Block, context.Username) { | ||||||
| 			return false | 			return false | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	log.Debug().Msg("Checking users") | 	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 { | 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, ",") { | 	for userGroup := range strings.SplitSeq(context.OAuthGroups, ",") { | ||||||
| 		if utils.CheckFilter(requiredGroups, strings.TrimSpace(userGroup)) { | 		if utils.CheckFilter(requiredGroups, strings.TrimSpace(userGroup)) { | ||||||
|  | 			log.Trace().Str("group", userGroup).Str("required", requiredGroups).Msg("User group matched") | ||||||
| 			return true | 			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 { | func (auth *AuthService) CheckIP(acls config.AppIP, ip string) bool { | ||||||
| 	for _, blocked := range labels.Block { | 	for _, blocked := range acls.Block { | ||||||
| 		res, err := utils.FilterIP(blocked, ip) | 		res, err := utils.FilterIP(blocked, ip) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Warn().Err(err).Str("item", blocked).Msg("Invalid IP/CIDR in block list") | 			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) | 		res, err := utils.FilterIP(allowed, ip) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Warn().Err(err).Str("item", allowed).Msg("Invalid IP/CIDR in allow list") | 			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") | 		log.Debug().Str("ip", ip).Msg("IP not in allow list, denying access") | ||||||
| 		return false | 		return false | ||||||
| 	} | 	} | ||||||
| @@ -402,8 +405,8 @@ func (auth *AuthService) CheckIP(labels config.AppIP, ip string) bool { | |||||||
| 	return true | 	return true | ||||||
| } | } | ||||||
|  |  | ||||||
| func (auth *AuthService) IsBypassedIP(labels config.AppIP, ip string) bool { | func (auth *AuthService) IsBypassedIP(acls config.AppIP, ip string) bool { | ||||||
| 	for _, bypassed := range labels.Bypass { | 	for _, bypassed := range acls.Bypass { | ||||||
| 		res, err := utils.FilterIP(bypassed, ip) | 		res, err := utils.FilterIP(bypassed, ip) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Warn().Err(err).Str("item", bypassed).Msg("Invalid IP/CIDR in bypass list") | 			log.Warn().Err(err).Str("item", bypassed).Msg("Invalid IP/CIDR in bypass list") | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ import ( | |||||||
| type DockerService struct { | type DockerService struct { | ||||||
| 	client      *client.Client | 	client      *client.Client | ||||||
| 	context     context.Context | 	context     context.Context | ||||||
|  | 	isConnected bool | ||||||
| } | } | ||||||
|  |  | ||||||
| func NewDockerService() *DockerService { | func NewDockerService() *DockerService { | ||||||
| @@ -31,10 +32,24 @@ func (docker *DockerService) Init() error { | |||||||
|  |  | ||||||
| 	docker.client = client | 	docker.client = client | ||||||
| 	docker.context = ctx | 	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 | 		return nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| func (docker *DockerService) GetContainers() ([]container.Summary, error) { | 	docker.isConnected = true | ||||||
|  | 	log.Debug().Msg("Docker connected") | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (docker *DockerService) getContainers() ([]container.Summary, error) { | ||||||
| 	containers, err := docker.client.ContainerList(docker.context, container.ListOptions{}) | 	containers, err := docker.client.ContainerList(docker.context, container.ListOptions{}) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return nil, err | 		return nil, err | ||||||
| @@ -42,7 +57,7 @@ func (docker *DockerService) GetContainers() ([]container.Summary, error) { | |||||||
| 	return containers, nil | 	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) | 	inspect, err := docker.client.ContainerInspect(docker.context, containerId) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return container.InspectResponse{}, err | 		return container.InspectResponse{}, err | ||||||
| @@ -50,45 +65,36 @@ func (docker *DockerService) InspectContainer(containerId string) (container.Ins | |||||||
| 	return inspect, nil | 	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) { | func (docker *DockerService) GetLabels(appDomain string) (config.App, error) { | ||||||
| 	isConnected := docker.DockerConnected() | 	if !docker.isConnected { | ||||||
|  |  | ||||||
| 	if !isConnected { |  | ||||||
| 		log.Debug().Msg("Docker not connected, returning empty labels") | 		log.Debug().Msg("Docker not connected, returning empty labels") | ||||||
| 		return config.App{}, nil | 		return config.App{}, nil | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	containers, err := docker.GetContainers() | 	containers, err := docker.getContainers() | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return config.App{}, err | 		return config.App{}, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	for _, ctr := range containers { | 	for _, ctr := range containers { | ||||||
| 		inspect, err := docker.InspectContainer(ctr.ID) | 		inspect, err := docker.inspectContainer(ctr.ID) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Warn().Str("id", ctr.ID).Err(err).Msg("Error inspecting container, skipping") | 			return config.App{}, err | ||||||
| 			continue |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		labels, err := decoders.DecodeLabels(inspect.Config.Labels) | 		labels, err := decoders.DecodeLabels(inspect.Config.Labels) | ||||||
| 		if err != nil { | 		if err != nil { | ||||||
| 			log.Warn().Str("id", ctr.ID).Err(err).Msg("Error getting container labels, skipping") | 			return config.App{}, err | ||||||
| 			continue |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		for appName, appLabels := range labels.Apps { | 		for appName, appLabels := range labels.Apps { | ||||||
| 			if appLabels.Config.Domain == appDomain { | 			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 | 				return appLabels, nil | ||||||
| 			} | 			} | ||||||
|  |  | ||||||
| 			if strings.TrimPrefix(inspect.Name, "/") == appName { | 			if strings.SplitN(appDomain, ".", 2)[0] == appName { | ||||||
| 				log.Debug().Str("id", inspect.ID).Msg("Found matching container by app name") | 				log.Debug().Str("id", inspect.ID).Str("name", inspect.Name).Msg("Found matching container by app name") | ||||||
| 				return appLabels, nil | 				return appLabels, nil | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ import ( | |||||||
| 	"time" | 	"time" | ||||||
| 	"tinyauth/internal/config" | 	"tinyauth/internal/config" | ||||||
|  |  | ||||||
|  | 	"github.com/rs/zerolog/log" | ||||||
| 	"golang.org/x/oauth2" | 	"golang.org/x/oauth2" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| @@ -58,10 +59,8 @@ func (generic *GenericOAuthService) Init() error { | |||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
|  |  | ||||||
| 	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient) | 	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient) | ||||||
| 	verifier := oauth2.GenerateVerifier() |  | ||||||
|  |  | ||||||
| 	generic.context = ctx | 	generic.context = ctx | ||||||
| 	generic.verifier = verifier |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -75,6 +74,12 @@ func (generic *GenericOAuthService) GenerateState() string { | |||||||
| 	return state | 	return state | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (generic *GenericOAuthService) GenerateVerifier() string { | ||||||
|  | 	verifier := oauth2.GenerateVerifier() | ||||||
|  | 	generic.verifier = verifier | ||||||
|  | 	return verifier | ||||||
|  | } | ||||||
|  |  | ||||||
| func (generic *GenericOAuthService) GetAuthURL(state string) string { | func (generic *GenericOAuthService) GetAuthURL(state string) string { | ||||||
| 	return generic.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(generic.verifier)) | 	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 | 		return user, err | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	log.Trace().Str("body", string(body)).Msg("Userinfo response body") | ||||||
|  |  | ||||||
| 	err = json.Unmarshal(body, &user) | 	err = json.Unmarshal(body, &user) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return user, err | 		return user, err | ||||||
|   | |||||||
| @@ -53,10 +53,7 @@ func (github *GithubOAuthService) Init() error { | |||||||
| 	httpClient := &http.Client{} | 	httpClient := &http.Client{} | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
| 	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient) | 	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient) | ||||||
| 	verifier := oauth2.GenerateVerifier() |  | ||||||
|  |  | ||||||
| 	github.context = ctx | 	github.context = ctx | ||||||
| 	github.verifier = verifier |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -70,6 +67,12 @@ func (github *GithubOAuthService) GenerateState() string { | |||||||
| 	return state | 	return state | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (github *GithubOAuthService) GenerateVerifier() string { | ||||||
|  | 	verifier := oauth2.GenerateVerifier() | ||||||
|  | 	github.verifier = verifier | ||||||
|  | 	return verifier | ||||||
|  | } | ||||||
|  |  | ||||||
| func (github *GithubOAuthService) GetAuthURL(state string) string { | func (github *GithubOAuthService) GetAuthURL(state string) string { | ||||||
| 	return github.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(github.verifier)) | 	return github.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(github.verifier)) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -48,10 +48,7 @@ func (google *GoogleOAuthService) Init() error { | |||||||
| 	httpClient := &http.Client{} | 	httpClient := &http.Client{} | ||||||
| 	ctx := context.Background() | 	ctx := context.Background() | ||||||
| 	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient) | 	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient) | ||||||
| 	verifier := oauth2.GenerateVerifier() |  | ||||||
|  |  | ||||||
| 	google.context = ctx | 	google.context = ctx | ||||||
| 	google.verifier = verifier |  | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -65,6 +62,12 @@ func (oauth *GoogleOAuthService) GenerateState() string { | |||||||
| 	return state | 	return state | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (google *GoogleOAuthService) GenerateVerifier() string { | ||||||
|  | 	verifier := oauth2.GenerateVerifier() | ||||||
|  | 	google.verifier = verifier | ||||||
|  | 	return verifier | ||||||
|  | } | ||||||
|  |  | ||||||
| func (google *GoogleOAuthService) GetAuthURL(state string) string { | func (google *GoogleOAuthService) GetAuthURL(state string) string { | ||||||
| 	return google.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(google.verifier)) | 	return google.config.AuthCodeURL(state, oauth2.AccessTypeOffline, oauth2.S256ChallengeOption(google.verifier)) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,6 +11,7 @@ import ( | |||||||
| type OAuthService interface { | type OAuthService interface { | ||||||
| 	Init() error | 	Init() error | ||||||
| 	GenerateState() string | 	GenerateState() string | ||||||
|  | 	GenerateVerifier() string | ||||||
| 	GetAuthURL(state string) string | 	GetAuthURL(state string) string | ||||||
| 	VerifyCode(code string) error | 	VerifyCode(code string) error | ||||||
| 	Userinfo() (config.Claims, 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) | 			log.Error().Err(err).Msgf("Failed to initialize OAuth service: %T", name) | ||||||
| 			return err | 			return err | ||||||
| 		} | 		} | ||||||
| 		log.Info().Str("service", service.GetName()).Msg("Initialized OAuth service") | 		log.Info().Str("service", name).Msg("Initialized OAuth service") | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return nil | 	return nil | ||||||
|   | |||||||
| @@ -100,17 +100,17 @@ func IsRedirectSafe(redirectURL string, domain string) bool { | |||||||
| 		return false | 		return false | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	cookieDomain, err := GetCookieDomain(redirectURL) | 	host := parsedURL.Hostname() | ||||||
|  | 	if host == domain { | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	cookieDomain, err := GetCookieDomain(redirectURL) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return false | 		return false | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	if cookieDomain != domain { | 	return cookieDomain == domain | ||||||
| 		return false |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	return true |  | ||||||
| } | } | ||||||
|  |  | ||||||
| func GetLogLevel(level string) zerolog.Level { | 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 { | 	if err != nil { | ||||||
| 		return nil, err | 		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 { | 	if err != nil { | ||||||
| 		return nil, err | 		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 | 	// If we have google/github providers and no redirect URL then set a default | ||||||
|  |  | ||||||
| 	for id := range config.OverrideProviders { | 	for id := range config.OverrideProviders { | ||||||
| 		if provider, exists := providers[id]; exists { | 		if provider, exists := providers[id]; exists { | ||||||
| 			if provider.RedirectURL == "" { | 			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 combined providers | ||||||
| 	return providers, nil | 	return providers, nil | ||||||
| } | } | ||||||
|   | |||||||
| @@ -164,7 +164,7 @@ func TestIsRedirectSafe(t *testing.T) { | |||||||
| 	// Case with no subdomain | 	// Case with no subdomain | ||||||
| 	redirectURL := "http://example.com/welcome" | 	redirectURL := "http://example.com/welcome" | ||||||
| 	result := utils.IsRedirectSafe(redirectURL, domain) | 	result := utils.IsRedirectSafe(redirectURL, domain) | ||||||
| 	assert.Equal(t, false, result) | 	assert.Equal(t, true, result) | ||||||
|  |  | ||||||
| 	// Case with different domain | 	// Case with different domain | ||||||
| 	redirectURL = "http://malicious.com/phishing" | 	redirectURL = "http://malicious.com/phishing" | ||||||
| @@ -202,6 +202,41 @@ func TestIsRedirectSafe(t *testing.T) { | |||||||
| 	assert.Equal(t, false, result) | 	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) { | func TestGetOAuthProvidersConfig(t *testing.T) { | ||||||
| 	env := []string{"PROVIDERS_CLIENT1_CLIENT_ID=client1-id", "PROVIDERS_CLIENT1_CLIENT_SECRET=client1-secret"} | 	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"} | 	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": { | 		"client1": { | ||||||
| 			ClientID:     "client1-id", | 			ClientID:     "client1-id", | ||||||
| 			ClientSecret: "client1-secret", | 			ClientSecret: "client1-secret", | ||||||
|  | 			Name:         "Client1", | ||||||
| 		}, | 		}, | ||||||
| 		"client2": { | 		"client2": { | ||||||
| 			ClientID:     "client2-id", | 			ClientID:     "client2-id", | ||||||
| 			ClientSecret: "client2-secret", | 			ClientSecret: "client2-secret", | ||||||
|  | 			Name:         "Client2", | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -247,6 +284,7 @@ func TestGetOAuthProvidersConfig(t *testing.T) { | |||||||
| 		"client1": { | 		"client1": { | ||||||
| 			ClientID:     "client1-id", | 			ClientID:     "client1-id", | ||||||
| 			ClientSecret: "file content", | 			ClientSecret: "file content", | ||||||
|  | 			Name:         "Client1", | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| @@ -262,6 +300,7 @@ func TestGetOAuthProvidersConfig(t *testing.T) { | |||||||
| 			ClientID:     "google-id", | 			ClientID:     "google-id", | ||||||
| 			ClientSecret: "google-secret", | 			ClientSecret: "google-secret", | ||||||
| 			RedirectURL:  "http://app.url/api/oauth/callback/google", | 			RedirectURL:  "http://app.url/api/oauth/callback/google", | ||||||
|  | 			Name:         "Google", | ||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|   | |||||||
| @@ -3,29 +3,24 @@ package decoders | |||||||
| import ( | import ( | ||||||
| 	"reflect" | 	"reflect" | ||||||
| 	"strings" | 	"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) | 	normalized := make(map[string]string) | ||||||
| 	knownKeys := getKnownKeys() |  | ||||||
|  |  | ||||||
| 	for k, v := range keys { | 	for k, v := range input { | ||||||
| 		var finalKey []string | 		parts := []string{"tinyauth"} | ||||||
| 		var suffix string |  | ||||||
| 		var camelClientName string |  | ||||||
| 		var camelField string |  | ||||||
|  |  | ||||||
| 		finalKey = append(finalKey, rootName) | 		key := strings.ToLower(k) | ||||||
| 		finalKey = append(finalKey, "providers") | 		key = strings.ReplaceAll(key, sep, "-") | ||||||
| 		lowerKey := strings.ToLower(k) |  | ||||||
|  |  | ||||||
| 		if !strings.HasPrefix(lowerKey, "providers"+sep) { | 		suffix := "" | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
|  |  | ||||||
| 		for _, known := range knownKeys { | 		for _, known := range knownKeys { | ||||||
| 			if strings.HasSuffix(lowerKey, strings.ReplaceAll(known, "-", sep)) { | 			if strings.HasSuffix(key, known) { | ||||||
| 				suffix = known | 				suffix = known | ||||||
| 				break | 				break | ||||||
| 			} | 			} | ||||||
| @@ -35,55 +30,47 @@ func NormalizeKeys(keys map[string]string, rootName string, sep string) map[stri | |||||||
| 			continue | 			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 | 			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 { | 		final := "" | ||||||
| 			if i == 0 { |  | ||||||
| 				camelClientName += p | 		for i, part := range parts { | ||||||
| 				continue | 			if i > 0 { | ||||||
|  | 				final += "." | ||||||
| 			} | 			} | ||||||
| 			if p == "" { | 			final += strcase.LowerCamelCase(part) | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			camelClientName += strings.ToUpper(string([]rune(p)[0])) + string([]rune(p)[1:]) |  | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		finalKey = append(finalKey, camelClientName) | 		normalized[final] = v | ||||||
|  |  | ||||||
| 		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 |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	return normalized | 	return normalized | ||||||
| } | } | ||||||
|  |  | ||||||
| func getKnownKeys() []string { | func getKnownKeys[T any]() []string { | ||||||
| 	var known []string | 	var keys []string | ||||||
|  | 	var t T | ||||||
|  |  | ||||||
| 	p := config.OAuthServiceConfig{} | 	v := reflect.ValueOf(t) | ||||||
| 	v := reflect.ValueOf(p) | 	typeOfT := v.Type() | ||||||
| 	typeOfP := v.Type() |  | ||||||
|  |  | ||||||
| 	for field := range typeOfP.NumField() { | 	for field := range typeOfT.NumField() { | ||||||
| 		known = append(known, typeOfP.Field(field).Tag.Get("key")) | 		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 | package decoders | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"tinyauth/internal/config" |  | ||||||
|  |  | ||||||
| 	"github.com/traefik/paerser/parser" | 	"github.com/traefik/paerser/parser" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func DecodeEnv(env map[string]string) (config.Providers, error) { | func DecodeEnv[T any, C any](env map[string]string, subName string) (T, error) { | ||||||
| 	normalized := NormalizeKeys(env, "tinyauth", "_") | 	var result T | ||||||
| 	var providers config.Providers |  | ||||||
|  |  | ||||||
| 	err := parser.Decode(normalized, &providers, "tinyauth", "tinyauth.providers") | 	normalized := normalizeKeys[C](env, subName, "_") | ||||||
|  |  | ||||||
|  | 	err := parser.Decode(normalized, &result, "tinyauth", "tinyauth."+subName) | ||||||
|  |  | ||||||
| 	if err != nil { | 	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) { | func TestDecodeEnv(t *testing.T) { | ||||||
| 	// Variables | 	// Setup | ||||||
| 	expected := config.Providers{ | 	env := map[string]string{ | ||||||
| 		Providers: map[string]config.OAuthServiceConfig{ | 		"PROVIDERS_GOOGLE_CLIENT_ID":        "google-client-id", | ||||||
| 			"client1": { | 		"PROVIDERS_GOOGLE_CLIENT_SECRET":    "google-client-secret", | ||||||
| 				ClientID:           "client1-id", | 		"PROVIDERS_MY_GITHUB_CLIENT_ID":     "github-client-id", | ||||||
| 				ClientSecret:       "client1-secret", | 		"PROVIDERS_MY_GITHUB_CLIENT_SECRET": "github-client-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", |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Test | 	expected := config.Providers{ | ||||||
| 	res, err := decoders.DecodeEnv(test) | 		Providers: map[string]config.OAuthServiceConfig{ | ||||||
| 	assert.NilError(t, err) | 			"google": { | ||||||
| 	assert.DeepEqual(t, expected, res) | 				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, result, expected) | ||||||
| } | } | ||||||
|   | |||||||
| @@ -2,23 +2,23 @@ package decoders | |||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"tinyauth/internal/config" |  | ||||||
|  |  | ||||||
| 	"github.com/traefik/paerser/parser" | 	"github.com/traefik/paerser/parser" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func DecodeFlags(flags map[string]string) (config.Providers, error) { | func DecodeFlags[T any, C any](flags map[string]string, subName string) (T, error) { | ||||||
| 	filtered := filterFlags(flags) | 	var result T | ||||||
| 	normalized := NormalizeKeys(filtered, "tinyauth", "-") |  | ||||||
| 	var providers config.Providers |  | ||||||
|  |  | ||||||
| 	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 { | 	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 { | func filterFlags(flags map[string]string) map[string]string { | ||||||
|   | |||||||
| @@ -9,52 +9,29 @@ import ( | |||||||
| ) | ) | ||||||
|  |  | ||||||
| func TestDecodeFlags(t *testing.T) { | func TestDecodeFlags(t *testing.T) { | ||||||
| 	// Variables | 	// Setup | ||||||
| 	expected := config.Providers{ | 	flags := map[string]string{ | ||||||
| 		Providers: map[string]config.OAuthServiceConfig{ | 		"--providers-google-client-id":        "google-client-id", | ||||||
| 			"client1": { | 		"--providers-google-client-secret":    "google-client-secret", | ||||||
| 				ClientID:           "client1-id", | 		"--providers-my-github-client-id":     "github-client-id", | ||||||
| 				ClientSecret:       "client1-secret", | 		"--providers-my-github-client-secret": "github-client-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", |  | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// Test | 	expected := config.Providers{ | ||||||
| 	res, err := decoders.DecodeFlags(test) | 		Providers: map[string]config.OAuthServiceConfig{ | ||||||
| 	assert.NilError(t, err) | 			"google": { | ||||||
| 	assert.DeepEqual(t, expected, res) | 				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, result, expected) | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								main.go
									
									
									
									
									
								
							| @@ -11,5 +11,5 @@ import ( | |||||||
|  |  | ||||||
| func main() { | func main() { | ||||||
| 	log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stderr, TimeFormat: time.RFC3339}).With().Timestamp().Caller().Logger() | 	log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stderr, TimeFormat: time.RFC3339}).With().Timestamp().Caller().Logger() | ||||||
| 	cmd.Execute() | 	cmd.Run() | ||||||
| } | } | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user