mirror of
				https://github.com/steveiliop56/tinyauth.git
				synced 2025-11-03 23:55:44 +00:00 
			
		
		
		
	Compare commits
	
		
			3 Commits
		
	
	
		
			9ed254cbe3
			...
			feat/ip-al
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					5dda1d22e7 | ||
| 
						 | 
					1770eb3e8e | ||
| 
						 | 
					fae5e7919a | 
							
								
								
									
										7
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							@@ -39,9 +39,4 @@ jobs:
 | 
			
		||||
          cp -r frontend/dist internal/assets/dist
 | 
			
		||||
 | 
			
		||||
      - name: Run tests
 | 
			
		||||
        run: go test -coverprofile=coverage.txt -v ./...
 | 
			
		||||
 | 
			
		||||
      - name: Upload coverage reports to Codecov
 | 
			
		||||
        uses: codecov/codecov-action@v5
 | 
			
		||||
        with:
 | 
			
		||||
          token: ${{ secrets.CODECOV_TOKEN }}
 | 
			
		||||
        run: go test -v ./...
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							@@ -11,7 +11,8 @@ docker-compose.test*
 | 
			
		||||
users.txt
 | 
			
		||||
 | 
			
		||||
# secret test file
 | 
			
		||||
secret*
 | 
			
		||||
secret.txt
 | 
			
		||||
secret_oauth.txt
 | 
			
		||||
 | 
			
		||||
# vscode
 | 
			
		||||
.vscode
 | 
			
		||||
 
 | 
			
		||||
@@ -1,5 +1,5 @@
 | 
			
		||||
# Site builder
 | 
			
		||||
FROM oven/bun:1.2.18-alpine AS frontend-builder
 | 
			
		||||
FROM oven/bun:1.2.16-alpine AS frontend-builder
 | 
			
		||||
 | 
			
		||||
WORKDIR /frontend
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										16
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
									
									
									
									
								
							@@ -23,27 +23,27 @@ Tinyauth is a simple authentication middleware that adds a simple login screen o
 | 
			
		||||
 | 
			
		||||
## Getting Started
 | 
			
		||||
 | 
			
		||||
You can easily get started with Tinyauth by following the guide in the [documentation](https://tinyauth.app/docs/getting-started.html). There is also an available [docker compose](./docker-compose.example.yml) file that has Traefik, Whoami and Tinyauth to demonstrate its capabilities.
 | 
			
		||||
You can easily get started with tinyauth by following the guide in the [documentation](https://tinyauth.app/docs/getting-started.html). There is also an available [docker compose](./docker-compose.example.yml) file that has traefik, whoami and tinyauth to demonstrate its capabilities.
 | 
			
		||||
 | 
			
		||||
## Demo
 | 
			
		||||
 | 
			
		||||
If you are still not sure if Tinyauth suits your needs you can try out the [demo](https://demo.tinyauth.app). The default username is `user` and the default password is `password`.
 | 
			
		||||
If you are still not sure if tinyauth suits your needs you can try out the [demo](https://demo.tinyauth.app). The default username is `user` and the default password is `password`.
 | 
			
		||||
 | 
			
		||||
## Documentation
 | 
			
		||||
 | 
			
		||||
You can find documentation and guides on all of the available configuration of Tinyauth in the [website](https://tinyauth.app).
 | 
			
		||||
You can find documentation and guides on all of the available configuration of tinyauth in the [website](https://tinyauth.app).
 | 
			
		||||
 | 
			
		||||
## Discord
 | 
			
		||||
 | 
			
		||||
Tinyauth has a [discord](https://discord.gg/eHzVaCzRRd) server. Feel free to hop in to chat about self-hosting, homelabs and of course Tinyauth. See you there!
 | 
			
		||||
Tinyauth has a [discord](https://discord.gg/eHzVaCzRRd) server. Feel free to hop in to chat about self-hosting, homelabs and of course tinyauth. See you there!
 | 
			
		||||
 | 
			
		||||
## Contributing
 | 
			
		||||
 | 
			
		||||
All contributions to the codebase are welcome! If you have any free time feel free to pick up an [issue](https://github.com/steveiliop56/tinyauth/issues) or add your own missing features. Make sure to check out the [contributing guide](./CONTRIBUTING.md) for instructions on how to get the development server up and running.
 | 
			
		||||
All contributions to the codebase are welcome! If you have any free time feel free to pick up an [Issue](https://github.com/steveiliop56/tinyauth/issues) or add your own missing features. Make sure to check out the [contributing guide](./CONTRIBUTING.md) for instructions on how to get the development server up and running.
 | 
			
		||||
 | 
			
		||||
## Localization
 | 
			
		||||
 | 
			
		||||
If you would like to help translate Tinyauth into more languages, visit the [Crowdin](https://crowdin.com/project/tinyauth) page.
 | 
			
		||||
If you would like to help translate tinyauth into more languages, visit the [Crowdin](https://crowdin.com/project/tinyauth) page.
 | 
			
		||||
 | 
			
		||||
## License
 | 
			
		||||
 | 
			
		||||
@@ -51,9 +51,9 @@ Tinyauth is licensed under the GNU General Public License v3.0. TL;DR — You ma
 | 
			
		||||
 | 
			
		||||
## Sponsors
 | 
			
		||||
 | 
			
		||||
A big thank you to the following people for providing me with more coffee:
 | 
			
		||||
Thanks a lot 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>  <!-- sponsors -->
 | 
			
		||||
 | 
			
		||||
## Acknowledgements
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										86
									
								
								cmd/root.go
									
									
									
									
									
								
							
							
						
						
									
										86
									
								
								cmd/root.go
									
									
									
									
									
								
							@@ -8,14 +8,13 @@ import (
 | 
			
		||||
	"time"
 | 
			
		||||
	totpCmd "tinyauth/cmd/totp"
 | 
			
		||||
	userCmd "tinyauth/cmd/user"
 | 
			
		||||
	"tinyauth/internal/api"
 | 
			
		||||
	"tinyauth/internal/auth"
 | 
			
		||||
	"tinyauth/internal/constants"
 | 
			
		||||
	"tinyauth/internal/docker"
 | 
			
		||||
	"tinyauth/internal/handlers"
 | 
			
		||||
	"tinyauth/internal/hooks"
 | 
			
		||||
	"tinyauth/internal/ldap"
 | 
			
		||||
	"tinyauth/internal/providers"
 | 
			
		||||
	"tinyauth/internal/server"
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
@@ -59,6 +58,10 @@ var rootCmd = &cobra.Command{
 | 
			
		||||
		users, err := utils.GetUsers(config.Users, config.UsersFile)
 | 
			
		||||
		HandleError(err, "Failed to parse users")
 | 
			
		||||
 | 
			
		||||
		if len(users) == 0 && !utils.OAuthConfigured(config) {
 | 
			
		||||
			HandleError(errors.New("no users or OAuth configured"), "No users or OAuth configured")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Get domain
 | 
			
		||||
		log.Debug().Msg("Getting domain")
 | 
			
		||||
		domain, err := utils.GetUpperDomain(config.AppURL)
 | 
			
		||||
@@ -71,15 +74,6 @@ var rootCmd = &cobra.Command{
 | 
			
		||||
		csrfCookieName := fmt.Sprintf("%s-%s", constants.CsrfCookieName, cookieId)
 | 
			
		||||
		redirectCookieName := fmt.Sprintf("%s-%s", constants.RedirectCookieName, cookieId)
 | 
			
		||||
 | 
			
		||||
		// Generate HMAC and encryption secrets
 | 
			
		||||
		log.Debug().Msg("Deriving HMAC and encryption secrets")
 | 
			
		||||
 | 
			
		||||
		hmacSecret, err := utils.DeriveKey(config.Secret, "hmac")
 | 
			
		||||
		HandleError(err, "Failed to derive HMAC secret")
 | 
			
		||||
 | 
			
		||||
		encryptionSecret, err := utils.DeriveKey(config.Secret, "encryption")
 | 
			
		||||
		HandleError(err, "Failed to derive encryption secret")
 | 
			
		||||
 | 
			
		||||
		// Create OAuth config
 | 
			
		||||
		oauthConfig := types.OAuthConfig{
 | 
			
		||||
			GithubClientId:      config.GithubClientId,
 | 
			
		||||
@@ -111,8 +105,8 @@ var rootCmd = &cobra.Command{
 | 
			
		||||
			RedirectCookieName:    redirectCookieName,
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Create server config
 | 
			
		||||
		serverConfig := types.ServerConfig{
 | 
			
		||||
		// Create api config
 | 
			
		||||
		apiConfig := types.APIConfig{
 | 
			
		||||
			Port:    config.Port,
 | 
			
		||||
			Address: config.Address,
 | 
			
		||||
		}
 | 
			
		||||
@@ -121,14 +115,13 @@ var rootCmd = &cobra.Command{
 | 
			
		||||
		authConfig := types.AuthConfig{
 | 
			
		||||
			Users:             users,
 | 
			
		||||
			OauthWhitelist:    config.OAuthWhitelist,
 | 
			
		||||
			Secret:            config.Secret,
 | 
			
		||||
			CookieSecure:      config.CookieSecure,
 | 
			
		||||
			SessionExpiry:     config.SessionExpiry,
 | 
			
		||||
			Domain:            domain,
 | 
			
		||||
			LoginTimeout:      config.LoginTimeout,
 | 
			
		||||
			LoginMaxRetries:   config.LoginMaxRetries,
 | 
			
		||||
			SessionCookieName: sessionCookieName,
 | 
			
		||||
			HMACSecret:        hmacSecret,
 | 
			
		||||
			EncryptionSecret:  encryptionSecret,
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Create hooks config
 | 
			
		||||
@@ -137,55 +130,36 @@ var rootCmd = &cobra.Command{
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Create docker service
 | 
			
		||||
		docker, err := docker.NewDocker()
 | 
			
		||||
		docker := docker.NewDocker()
 | 
			
		||||
 | 
			
		||||
		// Initialize docker
 | 
			
		||||
		err = docker.Init()
 | 
			
		||||
		HandleError(err, "Failed to initialize docker")
 | 
			
		||||
 | 
			
		||||
		// Create LDAP service if configured
 | 
			
		||||
		var ldapService *ldap.LDAP
 | 
			
		||||
 | 
			
		||||
		if config.LdapAddress != "" {
 | 
			
		||||
			log.Info().Msg("Using LDAP for authentication")
 | 
			
		||||
 | 
			
		||||
			ldapConfig := types.LdapConfig{
 | 
			
		||||
				Address:      config.LdapAddress,
 | 
			
		||||
				BindDN:       config.LdapBindDN,
 | 
			
		||||
				BindPassword: config.LdapBindPassword,
 | 
			
		||||
				BaseDN:       config.LdapBaseDN,
 | 
			
		||||
				Insecure:     config.LdapInsecure,
 | 
			
		||||
				SearchFilter: config.LdapSearchFilter,
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			// Create LDAP service
 | 
			
		||||
			ldapService, err = ldap.NewLDAP(ldapConfig)
 | 
			
		||||
			HandleError(err, "Failed to create LDAP service")
 | 
			
		||||
		} else {
 | 
			
		||||
			log.Info().Msg("LDAP not configured, using local users or OAuth")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Check if we have any users configured
 | 
			
		||||
		if len(users) == 0 && !utils.OAuthConfigured(config) && ldapService == nil {
 | 
			
		||||
			HandleError(errors.New("err no users"), "Unable to find a source of users")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Create auth service
 | 
			
		||||
		auth := auth.NewAuth(authConfig, docker, ldapService)
 | 
			
		||||
		auth := auth.NewAuth(authConfig, docker)
 | 
			
		||||
 | 
			
		||||
		// Create OAuth providers service
 | 
			
		||||
		providers := providers.NewProviders(oauthConfig)
 | 
			
		||||
 | 
			
		||||
		// Initialize providers
 | 
			
		||||
		providers.Init()
 | 
			
		||||
 | 
			
		||||
		// Create hooks service
 | 
			
		||||
		hooks := hooks.NewHooks(hooksConfig, auth, providers)
 | 
			
		||||
 | 
			
		||||
		// Create handlers
 | 
			
		||||
		handlers := handlers.NewHandlers(handlersConfig, auth, hooks, providers, docker)
 | 
			
		||||
 | 
			
		||||
		// Create server
 | 
			
		||||
		srv, err := server.NewServer(serverConfig, handlers)
 | 
			
		||||
		HandleError(err, "Failed to create server")
 | 
			
		||||
		// Create API
 | 
			
		||||
		api := api.NewAPI(apiConfig, handlers)
 | 
			
		||||
 | 
			
		||||
		// Start server
 | 
			
		||||
		err = srv.Start()
 | 
			
		||||
		HandleError(err, "Failed to start server")
 | 
			
		||||
		// Setup routes
 | 
			
		||||
		api.Init()
 | 
			
		||||
		api.SetupRoutes()
 | 
			
		||||
 | 
			
		||||
		// Start
 | 
			
		||||
		api.Run()
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -245,12 +219,6 @@ func init() {
 | 
			
		||||
	rootCmd.Flags().String("app-title", "Tinyauth", "Title of the app.")
 | 
			
		||||
	rootCmd.Flags().String("forgot-password-message", "You can reset your password by changing the `USERS` environment variable.", "Message to show on the forgot password page.")
 | 
			
		||||
	rootCmd.Flags().String("background-image", "/background.jpg", "Background image URL for the login page.")
 | 
			
		||||
	rootCmd.Flags().String("ldap-address", "", "LDAP server address (e.g. ldap://localhost:389).")
 | 
			
		||||
	rootCmd.Flags().String("ldap-bind-dn", "", "LDAP bind DN (e.g. uid=user,dc=example,dc=com).")
 | 
			
		||||
	rootCmd.Flags().String("ldap-bind-password", "", "LDAP bind password.")
 | 
			
		||||
	rootCmd.Flags().String("ldap-base-dn", "", "LDAP base DN (e.g. dc=example,dc=com).")
 | 
			
		||||
	rootCmd.Flags().Bool("ldap-insecure", false, "Skip certificate verification for the LDAP server.")
 | 
			
		||||
	rootCmd.Flags().String("ldap-search-filter", "(uid=%s)", "LDAP search filter for user lookup.")
 | 
			
		||||
 | 
			
		||||
	// Bind flags to environment
 | 
			
		||||
	viper.BindEnv("port", "PORT")
 | 
			
		||||
@@ -286,12 +254,6 @@ func init() {
 | 
			
		||||
	viper.BindEnv("login-max-retries", "LOGIN_MAX_RETRIES")
 | 
			
		||||
	viper.BindEnv("forgot-password-message", "FORGOT_PASSWORD_MESSAGE")
 | 
			
		||||
	viper.BindEnv("background-image", "BACKGROUND_IMAGE")
 | 
			
		||||
	viper.BindEnv("ldap-address", "LDAP_ADDRESS")
 | 
			
		||||
	viper.BindEnv("ldap-bind-dn", "LDAP_BIND_DN")
 | 
			
		||||
	viper.BindEnv("ldap-bind-password", "LDAP_BIND_PASSWORD")
 | 
			
		||||
	viper.BindEnv("ldap-base-dn", "LDAP_BASE_DN")
 | 
			
		||||
	viper.BindEnv("ldap-insecure", "LDAP_INSECURE")
 | 
			
		||||
	viper.BindEnv("ldap-search-filter", "LDAP_SEARCH_FILTER")
 | 
			
		||||
 | 
			
		||||
	// Bind flags to viper
 | 
			
		||||
	viper.BindPFlags(rootCmd.Flags())
 | 
			
		||||
 
 | 
			
		||||
@@ -9,45 +9,45 @@
 | 
			
		||||
        "@radix-ui/react-select": "^2.2.5",
 | 
			
		||||
        "@radix-ui/react-separator": "^1.1.7",
 | 
			
		||||
        "@radix-ui/react-slot": "^1.2.3",
 | 
			
		||||
        "@tailwindcss/vite": "^4.1.11",
 | 
			
		||||
        "@tanstack/react-query": "^5.81.5",
 | 
			
		||||
        "@tailwindcss/vite": "^4.1.10",
 | 
			
		||||
        "@tanstack/react-query": "^5.80.7",
 | 
			
		||||
        "axios": "^1.10.0",
 | 
			
		||||
        "class-variance-authority": "^0.7.1",
 | 
			
		||||
        "clsx": "^2.1.1",
 | 
			
		||||
        "dompurify": "^3.2.6",
 | 
			
		||||
        "i18next": "^25.3.2",
 | 
			
		||||
        "i18next": "^25.2.1",
 | 
			
		||||
        "i18next-browser-languagedetector": "^8.2.0",
 | 
			
		||||
        "i18next-resources-to-backend": "^1.2.1",
 | 
			
		||||
        "input-otp": "^1.4.2",
 | 
			
		||||
        "lucide-react": "^0.525.0",
 | 
			
		||||
        "lucide-react": "^0.516.0",
 | 
			
		||||
        "next-themes": "^0.4.6",
 | 
			
		||||
        "react": "^19.0.0",
 | 
			
		||||
        "react-dom": "^19.0.0",
 | 
			
		||||
        "react-hook-form": "^7.60.0",
 | 
			
		||||
        "react-i18next": "^15.6.0",
 | 
			
		||||
        "react-hook-form": "^7.58.0",
 | 
			
		||||
        "react-i18next": "^15.5.3",
 | 
			
		||||
        "react-markdown": "^10.1.0",
 | 
			
		||||
        "react-router": "^7.6.3",
 | 
			
		||||
        "sonner": "^2.0.6",
 | 
			
		||||
        "react-router": "^7.6.2",
 | 
			
		||||
        "sonner": "^2.0.5",
 | 
			
		||||
        "tailwind-merge": "^3.3.1",
 | 
			
		||||
        "tailwindcss": "^4.1.11",
 | 
			
		||||
        "zod": "^3.25.76",
 | 
			
		||||
        "tailwindcss": "^4.1.10",
 | 
			
		||||
        "zod": "^3.25.67",
 | 
			
		||||
      },
 | 
			
		||||
      "devDependencies": {
 | 
			
		||||
        "@eslint/js": "^9.30.1",
 | 
			
		||||
        "@tanstack/eslint-plugin-query": "^5.81.2",
 | 
			
		||||
        "@types/node": "^24.0.12",
 | 
			
		||||
        "@eslint/js": "^9.29.0",
 | 
			
		||||
        "@tanstack/eslint-plugin-query": "^5.78.0",
 | 
			
		||||
        "@types/node": "^24.0.3",
 | 
			
		||||
        "@types/react": "^19.1.8",
 | 
			
		||||
        "@types/react-dom": "^19.1.6",
 | 
			
		||||
        "@vitejs/plugin-react": "^4.6.0",
 | 
			
		||||
        "eslint": "^9.30.1",
 | 
			
		||||
        "@vitejs/plugin-react": "^4.5.2",
 | 
			
		||||
        "eslint": "^9.29.0",
 | 
			
		||||
        "eslint-plugin-react-hooks": "^5.2.0",
 | 
			
		||||
        "eslint-plugin-react-refresh": "^0.4.19",
 | 
			
		||||
        "globals": "^16.3.0",
 | 
			
		||||
        "prettier": "3.6.2",
 | 
			
		||||
        "tw-animate-css": "^1.3.5",
 | 
			
		||||
        "globals": "^16.2.0",
 | 
			
		||||
        "prettier": "3.5.3",
 | 
			
		||||
        "tw-animate-css": "^1.3.4",
 | 
			
		||||
        "typescript": "~5.8.3",
 | 
			
		||||
        "typescript-eslint": "^8.36.0",
 | 
			
		||||
        "vite": "^7.0.3",
 | 
			
		||||
        "typescript-eslint": "^8.34.1",
 | 
			
		||||
        "vite": "^6.3.1",
 | 
			
		||||
      },
 | 
			
		||||
    },
 | 
			
		||||
  },
 | 
			
		||||
@@ -84,7 +84,7 @@
 | 
			
		||||
 | 
			
		||||
    "@babel/plugin-transform-react-jsx-source": ["@babel/plugin-transform-react-jsx-source@7.27.1", "", { "dependencies": { "@babel/helper-plugin-utils": "^7.27.1" }, "peerDependencies": { "@babel/core": "^7.0.0-0" } }, "sha512-zbwoTsBruTeKB9hSq73ha66iFeJHuaFkUbwvqElnygoNbj/jHRsSeokowZFN3CZ64IvEqcmmkVe89OPXc7ldAw=="],
 | 
			
		||||
 | 
			
		||||
    "@babel/runtime": ["@babel/runtime@7.27.6", "", {}, "sha512-vbavdySgbTTrmFE+EsiqUTzlOr5bzlnJtUv9PynGCAKvfQqjIXbvFdumPM/GxMDfyuGMJaJAU6TO4zc1Jf1i8Q=="],
 | 
			
		||||
    "@babel/runtime": ["@babel/runtime@7.27.1", "", {}, "sha512-1x3D2xEk2fRo3PAhwQwu5UubzgiVWSXTBfWpVd2Mx2AzRqJuDJCsgaDVZ7HB5iGzDW1Hl1sWN2mFyKjmR9uAog=="],
 | 
			
		||||
 | 
			
		||||
    "@babel/template": ["@babel/template@7.27.2", "", { "dependencies": { "@babel/code-frame": "^7.27.1", "@babel/parser": "^7.27.2", "@babel/types": "^7.27.1" } }, "sha512-LPDZ85aEJyYSd18/DkjNh4/y1ntkE5KwUHWTiqgRxruuZL2F1yuHligVHLvcHY2vMHXttKFpJn6LwfI7cw7ODw=="],
 | 
			
		||||
 | 
			
		||||
@@ -146,15 +146,15 @@
 | 
			
		||||
 | 
			
		||||
    "@eslint-community/regexpp": ["@eslint-community/regexpp@4.12.1", "", {}, "sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ=="],
 | 
			
		||||
 | 
			
		||||
    "@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.20.1", "", { "dependencies": { "@eslint/object-schema": "^2.1.6", "debug": "^4.3.1", "minimatch": "^3.1.2" } }, "sha512-OL0RJzC/CBzli0DrrR31qzj6d6i6Mm3HByuhflhl4LOBiWxN+3i6/t/ZQQNii4tjksXi8r2CRW1wMpWA2ULUEw=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/config-helpers": ["@eslint/config-helpers@0.3.0", "", {}, "sha512-ViuymvFmcJi04qdZeDc2whTHryouGcDlaxPqarTD0ZE10ISpxGUVZGZDx4w01upyIynL3iu6IXH2bS1NhclQMw=="],
 | 
			
		||||
    "@eslint/config-helpers": ["@eslint/config-helpers@0.2.2", "", {}, "sha512-+GPzk8PlG0sPpzdU5ZvIRMPidzAnZDl/s9L+y13iodqvb8leL53bTannOrQ/Im7UkpsmFU5Ily5U60LWixnmLg=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/core": ["@eslint/core@0.14.0", "", { "dependencies": { "@types/json-schema": "^7.0.15" } }, "sha512-qIbV0/JZr7iSDjqAc60IqbLdsj9GDt16xQtWD+B78d/HAlvysGdZZ6rpJHGAc2T0FQx1X6thsSPdnoiGKdNtdg=="],
 | 
			
		||||
 | 
			
		||||
    "@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.30.1", "", {}, "sha512-zXhuECFlyep42KZUhWjfvsmXGX39W8K8LFb8AWXM9gSV9dQB+MrJGLKvW6Zw0Ggnbpw0VHTtrhFXYe3Gym18jg=="],
 | 
			
		||||
    "@eslint/js": ["@eslint/js@9.29.0", "", {}, "sha512-3PIF4cBw/y+1u2EazflInpV+lYsSG0aByVIQzAgb1m1MhHFSbqTyNqtBKHgWf/9Ykud+DhILS9EGkmekVhbKoQ=="],
 | 
			
		||||
 | 
			
		||||
    "@eslint/object-schema": ["@eslint/object-schema@2.1.6", "", {}, "sha512-RBMg5FRL0I0gs51M/guSAj5/e14VQ4tpZnQNWwuDT66P14I43ItmPfIZRhO9fUVIPOAQXU47atlywZ/czoqFPA=="],
 | 
			
		||||
 | 
			
		||||
@@ -252,7 +252,7 @@
 | 
			
		||||
 | 
			
		||||
    "@radix-ui/rect": ["@radix-ui/rect@1.1.1", "", {}, "sha512-HPwpGIzkl28mWyZqG52jiqDJ12waP11Pa1lGoiyUkIEuMLBP0oeK/C89esbXrxsky5we7dfd8U58nm0SgAWpVw=="],
 | 
			
		||||
 | 
			
		||||
    "@rolldown/pluginutils": ["@rolldown/pluginutils@1.0.0-beta.19", "", {}, "sha512-3FL3mnMbPu0muGOCaKAhhFEYmqv9eTfPSJRJmANrCwtgK8VuxpsZDGK+m0LYAGoyO8+0j5uRe4PeyPDK1yA/hA=="],
 | 
			
		||||
    "@rolldown/pluginutils": ["@rolldown/pluginutils@1.0.0-beta.11", "", {}, "sha512-L/gAA/hyCSuzTF1ftlzUSI/IKr2POHsv1Dd78GfqkR83KMNuswWD61JxGV2L7nRwBBBSDr6R1gCkdTmoN7W4ag=="],
 | 
			
		||||
 | 
			
		||||
    "@rollup/rollup-android-arm-eabi": ["@rollup/rollup-android-arm-eabi@4.40.2", "", { "os": "android", "cpu": "arm" }, "sha512-JkdNEq+DFxZfUwxvB58tHMHBHVgX23ew41g1OQinthJ+ryhdRk67O31S7sYw8u2lTjHUPFxwar07BBt1KHp/hg=="],
 | 
			
		||||
 | 
			
		||||
@@ -296,41 +296,41 @@
 | 
			
		||||
 | 
			
		||||
    "@standard-schema/utils": ["@standard-schema/utils@0.3.0", "", {}, "sha512-e7Mew686owMaPJVNNLs55PUvgz371nKgwsc4vxE49zsODpJEnxgxRo2y/OKrqueavXgZNMDVj3DdHFlaSAeU8g=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node": ["@tailwindcss/node@4.1.11", "", { "dependencies": { "@ampproject/remapping": "^2.3.0", "enhanced-resolve": "^5.18.1", "jiti": "^2.4.2", "lightningcss": "1.30.1", "magic-string": "^0.30.17", "source-map-js": "^1.2.1", "tailwindcss": "4.1.11" } }, "sha512-yzhzuGRmv5QyU9qLNg4GTlYI6STedBWRE7NjxP45CsFYYq9taI0zJXZBMqIC/c8fViNLhmrbpSFS57EoxUmD6Q=="],
 | 
			
		||||
    "@tailwindcss/node": ["@tailwindcss/node@4.1.10", "", { "dependencies": { "@ampproject/remapping": "^2.3.0", "enhanced-resolve": "^5.18.1", "jiti": "^2.4.2", "lightningcss": "1.30.1", "magic-string": "^0.30.17", "source-map-js": "^1.2.1", "tailwindcss": "4.1.10" } }, "sha512-2ACf1znY5fpRBwRhMgj9ZXvb2XZW8qs+oTfotJ2C5xR0/WNL7UHZ7zXl6s+rUqedL1mNi+0O+WQr5awGowS3PQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide": ["@tailwindcss/oxide@4.1.11", "", { "dependencies": { "detect-libc": "^2.0.4", "tar": "^7.4.3" }, "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.1.11", "@tailwindcss/oxide-darwin-arm64": "4.1.11", "@tailwindcss/oxide-darwin-x64": "4.1.11", "@tailwindcss/oxide-freebsd-x64": "4.1.11", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.1.11", "@tailwindcss/oxide-linux-arm64-gnu": "4.1.11", "@tailwindcss/oxide-linux-arm64-musl": "4.1.11", "@tailwindcss/oxide-linux-x64-gnu": "4.1.11", "@tailwindcss/oxide-linux-x64-musl": "4.1.11", "@tailwindcss/oxide-wasm32-wasi": "4.1.11", "@tailwindcss/oxide-win32-arm64-msvc": "4.1.11", "@tailwindcss/oxide-win32-x64-msvc": "4.1.11" } }, "sha512-Q69XzrtAhuyfHo+5/HMgr1lAiPP/G40OMFAnws7xcFEYqcypZmdW8eGXaOUIeOl1dzPJBPENXgbjsOyhg2nkrg=="],
 | 
			
		||||
    "@tailwindcss/oxide": ["@tailwindcss/oxide@4.1.10", "", { "dependencies": { "detect-libc": "^2.0.4", "tar": "^7.4.3" }, "optionalDependencies": { "@tailwindcss/oxide-android-arm64": "4.1.10", "@tailwindcss/oxide-darwin-arm64": "4.1.10", "@tailwindcss/oxide-darwin-x64": "4.1.10", "@tailwindcss/oxide-freebsd-x64": "4.1.10", "@tailwindcss/oxide-linux-arm-gnueabihf": "4.1.10", "@tailwindcss/oxide-linux-arm64-gnu": "4.1.10", "@tailwindcss/oxide-linux-arm64-musl": "4.1.10", "@tailwindcss/oxide-linux-x64-gnu": "4.1.10", "@tailwindcss/oxide-linux-x64-musl": "4.1.10", "@tailwindcss/oxide-wasm32-wasi": "4.1.10", "@tailwindcss/oxide-win32-arm64-msvc": "4.1.10", "@tailwindcss/oxide-win32-x64-msvc": "4.1.10" } }, "sha512-v0C43s7Pjw+B9w21htrQwuFObSkio2aV/qPx/mhrRldbqxbWJK6KizM+q7BF1/1CmuLqZqX3CeYF7s7P9fbA8Q=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.1.11", "", { "os": "android", "cpu": "arm64" }, "sha512-3IfFuATVRUMZZprEIx9OGDjG3Ou3jG4xQzNTvjDoKmU9JdmoCohQJ83MYd0GPnQIu89YoJqvMM0G3uqLRFtetg=="],
 | 
			
		||||
    "@tailwindcss/oxide-android-arm64": ["@tailwindcss/oxide-android-arm64@4.1.10", "", { "os": "android", "cpu": "arm64" }, "sha512-VGLazCoRQ7rtsCzThaI1UyDu/XRYVyH4/EWiaSX6tFglE+xZB5cvtC5Omt0OQ+FfiIVP98su16jDVHDEIuH4iQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.1.11", "", { "os": "darwin", "cpu": "arm64" }, "sha512-ESgStEOEsyg8J5YcMb1xl8WFOXfeBmrhAwGsFxxB2CxY9evy63+AtpbDLAyRkJnxLy2WsD1qF13E97uQyP1lfQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-darwin-arm64": ["@tailwindcss/oxide-darwin-arm64@4.1.10", "", { "os": "darwin", "cpu": "arm64" }, "sha512-ZIFqvR1irX2yNjWJzKCqTCcHZbgkSkSkZKbRM3BPzhDL/18idA8uWCoopYA2CSDdSGFlDAxYdU2yBHwAwx8euQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.1.11", "", { "os": "darwin", "cpu": "x64" }, "sha512-EgnK8kRchgmgzG6jE10UQNaH9Mwi2n+yw1jWmof9Vyg2lpKNX2ioe7CJdf9M5f8V9uaQxInenZkOxnTVL3fhAw=="],
 | 
			
		||||
    "@tailwindcss/oxide-darwin-x64": ["@tailwindcss/oxide-darwin-x64@4.1.10", "", { "os": "darwin", "cpu": "x64" }, "sha512-eCA4zbIhWUFDXoamNztmS0MjXHSEJYlvATzWnRiTqJkcUteSjO94PoRHJy1Xbwp9bptjeIxxBHh+zBWFhttbrQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.1.11", "", { "os": "freebsd", "cpu": "x64" }, "sha512-xdqKtbpHs7pQhIKmqVpxStnY1skuNh4CtbcyOHeX1YBE0hArj2romsFGb6yUmzkq/6M24nkxDqU8GYrKrz+UcA=="],
 | 
			
		||||
    "@tailwindcss/oxide-freebsd-x64": ["@tailwindcss/oxide-freebsd-x64@4.1.10", "", { "os": "freebsd", "cpu": "x64" }, "sha512-8/392Xu12R0cc93DpiJvNpJ4wYVSiciUlkiOHOSOQNH3adq9Gi/dtySK7dVQjXIOzlpSHjeCL89RUUI8/GTI6g=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.1.11", "", { "os": "linux", "cpu": "arm" }, "sha512-ryHQK2eyDYYMwB5wZL46uoxz2zzDZsFBwfjssgB7pzytAeCCa6glsiJGjhTEddq/4OsIjsLNMAiMlHNYnkEEeg=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm-gnueabihf": ["@tailwindcss/oxide-linux-arm-gnueabihf@4.1.10", "", { "os": "linux", "cpu": "arm" }, "sha512-t9rhmLT6EqeuPT+MXhWhlRYIMSfh5LZ6kBrC4FS6/+M1yXwfCtp24UumgCWOAJVyjQwG+lYva6wWZxrfvB+NhQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.1.11", "", { "os": "linux", "cpu": "arm64" }, "sha512-mYwqheq4BXF83j/w75ewkPJmPZIqqP1nhoghS9D57CLjsh3Nfq0m4ftTotRYtGnZd3eCztgbSPJ9QhfC91gDZQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-gnu": ["@tailwindcss/oxide-linux-arm64-gnu@4.1.10", "", { "os": "linux", "cpu": "arm64" }, "sha512-3oWrlNlxLRxXejQ8zImzrVLuZ/9Z2SeKoLhtCu0hpo38hTO2iL86eFOu4sVR8cZc6n3z7eRXXqtHJECa6mFOvA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.1.11", "", { "os": "linux", "cpu": "arm64" }, "sha512-m/NVRFNGlEHJrNVk3O6I9ggVuNjXHIPoD6bqay/pubtYC9QIdAMpS+cswZQPBLvVvEF6GtSNONbDkZrjWZXYNQ=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-arm64-musl": ["@tailwindcss/oxide-linux-arm64-musl@4.1.10", "", { "os": "linux", "cpu": "arm64" }, "sha512-saScU0cmWvg/Ez4gUmQWr9pvY9Kssxt+Xenfx1LG7LmqjcrvBnw4r9VjkFcqmbBb7GCBwYNcZi9X3/oMda9sqQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.1.11", "", { "os": "linux", "cpu": "x64" }, "sha512-YW6sblI7xukSD2TdbbaeQVDysIm/UPJtObHJHKxDEcW2exAtY47j52f8jZXkqE1krdnkhCMGqP3dbniu1Te2Fg=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-gnu": ["@tailwindcss/oxide-linux-x64-gnu@4.1.10", "", { "os": "linux", "cpu": "x64" }, "sha512-/G3ao/ybV9YEEgAXeEg28dyH6gs1QG8tvdN9c2MNZdUXYBaIY/Gx0N6RlJzfLy/7Nkdok4kaxKPHKJUlAaoTdA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.1.11", "", { "os": "linux", "cpu": "x64" }, "sha512-e3C/RRhGunWYNC3aSF7exsQkdXzQ/M+aYuZHKnw4U7KQwTJotnWsGOIVih0s2qQzmEzOFIJ3+xt7iq67K/p56Q=="],
 | 
			
		||||
    "@tailwindcss/oxide-linux-x64-musl": ["@tailwindcss/oxide-linux-x64-musl@4.1.10", "", { "os": "linux", "cpu": "x64" }, "sha512-LNr7X8fTiKGRtQGOerSayc2pWJp/9ptRYAa4G+U+cjw9kJZvkopav1AQc5HHD+U364f71tZv6XamaHKgrIoVzA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.1.11", "", { "dependencies": { "@emnapi/core": "^1.4.3", "@emnapi/runtime": "^1.4.3", "@emnapi/wasi-threads": "^1.0.2", "@napi-rs/wasm-runtime": "^0.2.11", "@tybys/wasm-util": "^0.9.0", "tslib": "^2.8.0" }, "cpu": "none" }, "sha512-Xo1+/GU0JEN/C/dvcammKHzeM6NqKovG+6921MR6oadee5XPBaKOumrJCXvopJ/Qb5TH7LX/UAywbqrP4lax0g=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi": ["@tailwindcss/oxide-wasm32-wasi@4.1.10", "", { "dependencies": { "@emnapi/core": "^1.4.3", "@emnapi/runtime": "^1.4.3", "@emnapi/wasi-threads": "^1.0.2", "@napi-rs/wasm-runtime": "^0.2.10", "@tybys/wasm-util": "^0.9.0", "tslib": "^2.8.0" }, "cpu": "none" }, "sha512-d6ekQpopFQJAcIK2i7ZzWOYGZ+A6NzzvQ3ozBvWFdeyqfOZdYHU66g5yr+/HC4ipP1ZgWsqa80+ISNILk+ae/Q=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.1.11", "", { "os": "win32", "cpu": "arm64" }, "sha512-UgKYx5PwEKrac3GPNPf6HVMNhUIGuUh4wlDFR2jYYdkX6pL/rn73zTq/4pzUm8fOjAn5L8zDeHp9iXmUGOXZ+w=="],
 | 
			
		||||
    "@tailwindcss/oxide-win32-arm64-msvc": ["@tailwindcss/oxide-win32-arm64-msvc@4.1.10", "", { "os": "win32", "cpu": "arm64" }, "sha512-i1Iwg9gRbwNVOCYmnigWCCgow8nDWSFmeTUU5nbNx3rqbe4p0kRbEqLwLJbYZKmSSp23g4N6rCDmm7OuPBXhDA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.1.11", "", { "os": "win32", "cpu": "x64" }, "sha512-YfHoggn1j0LK7wR82TOucWc5LDCguHnoS879idHekmmiR7g9HUtMw9MI0NHatS28u/Xlkfi9w5RJWgz2Dl+5Qg=="],
 | 
			
		||||
    "@tailwindcss/oxide-win32-x64-msvc": ["@tailwindcss/oxide-win32-x64-msvc@4.1.10", "", { "os": "win32", "cpu": "x64" }, "sha512-sGiJTjcBSfGq2DVRtaSljq5ZgZS2SDHSIfhOylkBvHVjwOsodBhnb3HdmiKkVuUGKD0I7G63abMOVaskj1KpOA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/vite": ["@tailwindcss/vite@4.1.11", "", { "dependencies": { "@tailwindcss/node": "4.1.11", "@tailwindcss/oxide": "4.1.11", "tailwindcss": "4.1.11" }, "peerDependencies": { "vite": "^5.2.0 || ^6 || ^7" } }, "sha512-RHYhrR3hku0MJFRV+fN2gNbDNEh3dwKvY8XJvTxCSXeMOsCRSr+uKvDWQcbizrHgjML6ZmTE5OwMrl5wKcujCw=="],
 | 
			
		||||
    "@tailwindcss/vite": ["@tailwindcss/vite@4.1.10", "", { "dependencies": { "@tailwindcss/node": "4.1.10", "@tailwindcss/oxide": "4.1.10", "tailwindcss": "4.1.10" }, "peerDependencies": { "vite": "^5.2.0 || ^6" } }, "sha512-QWnD5HDY2IADv+vYR82lOhqOlS1jSCUUAmfem52cXAhRTKxpDh3ARX8TTXJTCCO7Rv7cD2Nlekabv02bwP3a2A=="],
 | 
			
		||||
 | 
			
		||||
    "@tanstack/eslint-plugin-query": ["@tanstack/eslint-plugin-query@5.81.2", "", { "dependencies": { "@typescript-eslint/utils": "^8.18.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0" } }, "sha512-h4k6P6fm5VhKP5NkK+0TTVpGGyKQdx6tk7NYYG7J7PkSu7ClpLgBihw7yzK8N3n5zPaF3IMyErxfoNiXWH/3/A=="],
 | 
			
		||||
    "@tanstack/eslint-plugin-query": ["@tanstack/eslint-plugin-query@5.78.0", "", { "dependencies": { "@typescript-eslint/utils": "^8.18.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0" } }, "sha512-hYkhWr3UP0CkAsn/phBVR98UQawbw8CmTSgWtdgEBUjI60/GBaEIkpgi/Bp/2I8eIDK4+vdY7ac6jZx+GR+hEQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tanstack/query-core": ["@tanstack/query-core@5.81.5", "", {}, "sha512-ZJOgCy/z2qpZXWaj/oxvodDx07XcQa9BF92c0oINjHkoqUPsmm3uG08HpTaviviZ/N9eP1f9CM7mKSEkIo7O1Q=="],
 | 
			
		||||
    "@tanstack/query-core": ["@tanstack/query-core@5.80.7", "", {}, "sha512-s09l5zeUKC8q7DCCCIkVSns8zZrK4ZDT6ryEjxNBFi68G4z2EBobBS7rdOY3r6W1WbUDpc1fe5oY+YO/+2UVUg=="],
 | 
			
		||||
 | 
			
		||||
    "@tanstack/react-query": ["@tanstack/react-query@5.81.5", "", { "dependencies": { "@tanstack/query-core": "5.81.5" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-lOf2KqRRiYWpQT86eeeftAGnjuTR35myTP8MXyvHa81VlomoAWNEd8x5vkcAfQefu0qtYCvyqLropFZqgI2EQw=="],
 | 
			
		||||
    "@tanstack/react-query": ["@tanstack/react-query@5.80.7", "", { "dependencies": { "@tanstack/query-core": "5.80.7" }, "peerDependencies": { "react": "^18 || ^19" } }, "sha512-u2F0VK6+anItoEvB3+rfvTO9GEh2vb00Je05OwlUe/A0lkJBgW1HckiY3f9YZa+jx6IOe4dHPh10dyp9aY3iRQ=="],
 | 
			
		||||
 | 
			
		||||
    "@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=="],
 | 
			
		||||
 | 
			
		||||
@@ -354,7 +354,7 @@
 | 
			
		||||
 | 
			
		||||
    "@types/ms": ["@types/ms@2.1.0", "", {}, "sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA=="],
 | 
			
		||||
 | 
			
		||||
    "@types/node": ["@types/node@24.0.12", "", { "dependencies": { "undici-types": "~7.8.0" } }, "sha512-LtOrbvDf5ndC9Xi+4QZjVL0woFymF/xSTKZKPgrrl7H7XoeDvnD+E2IclKVDyaK9UM756W/3BXqSU+JEHopA9g=="],
 | 
			
		||||
    "@types/node": ["@types/node@24.0.3", "", { "dependencies": { "undici-types": "~7.8.0" } }, "sha512-R4I/kzCYAdRLzfiCabn9hxWfbuHS573x+r0dJMkkzThEa7pbrcDWK+9zu3e7aBOouf+rQAciqPFMnxwr0aWgKg=="],
 | 
			
		||||
 | 
			
		||||
    "@types/react": ["@types/react@19.1.8", "", { "dependencies": { "csstype": "^3.0.2" } }, "sha512-AwAfQ2Wa5bCx9WP8nZL2uMZWod7J7/JSplxbTmBQ5ms6QpqNYm672H0Vu9ZVKVngQ+ii4R/byguVEUZQyeg44g=="],
 | 
			
		||||
 | 
			
		||||
@@ -364,29 +364,29 @@
 | 
			
		||||
 | 
			
		||||
    "@types/unist": ["@types/unist@3.0.3", "", {}, "sha512-ko/gIFJRv177XgZsZcBwnqJN5x/Gien8qNOn0D5bQU/zAzVf9Zt3BlcUiLqhV9y4ARk0GbT3tnUiPNgnTXzc/Q=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.36.0", "", { "dependencies": { "@eslint-community/regexpp": "^4.10.0", "@typescript-eslint/scope-manager": "8.36.0", "@typescript-eslint/type-utils": "8.36.0", "@typescript-eslint/utils": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.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.36.0", "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-lZNihHUVB6ZZiPBNgOQGSxUASI7UJWhT8nHyUGCnaQ28XFCw98IfrMCG3rUl1uwUWoAvodJQby2KTs79UTcrAg=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin": ["@typescript-eslint/eslint-plugin@8.34.1", "", { "dependencies": { "@eslint-community/regexpp": "^4.10.0", "@typescript-eslint/scope-manager": "8.34.1", "@typescript-eslint/type-utils": "8.34.1", "@typescript-eslint/utils": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.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.34.1", "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-STXcN6ebF6li4PxwNeFnqF8/2BNDvBupf2OPx2yWNzr6mKNGF7q49VM00Pz5FaomJyqvbXpY6PhO+T9w139YEQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser": ["@typescript-eslint/parser@8.36.0", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/typescript-estree": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.0", "debug": "^4.3.4" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-FuYgkHwZLuPbZjQHzJXrtXreJdFMKl16BFYyRrLxDhWr6Qr7Kbcu2s1Yhu8tsiMXw1S0W1pjfFfYEt+R604s+Q=="],
 | 
			
		||||
    "@typescript-eslint/parser": ["@typescript-eslint/parser@8.34.1", "", { "dependencies": { "@typescript-eslint/scope-manager": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/typescript-estree": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.1", "debug": "^4.3.4" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-4O3idHxhyzjClSMJ0a29AcoK0+YwnEqzI6oz3vlRf3xw0zbzt15MzXwItOlnr5nIth6zlY2RENLsOPvhyrKAQA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.34.1", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.34.1", "@typescript-eslint/types": "^8.34.1", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-nuHlOmFZfuRwLJKDGQOVc0xnQrAmuq1Mj/ISou5044y1ajGNp2BNliIqp7F2LPQ5sForz8lempMFCovfeS1XoA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.1" } }, "sha512-beu6o6QY4hJAgL1E8RaXNC071G4Kso2MGmJskCFQhRhg8VOH/FDbC8soP8NHN7e/Hdphwp8G8cE6OBzC8o41ZA=="],
 | 
			
		||||
    "@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.32.0", "", { "dependencies": { "@typescript-eslint/types": "8.32.0", "@typescript-eslint/visitor-keys": "8.32.0" } }, "sha512-jc/4IxGNedXkmG4mx4nJTILb6TMjL66D41vyeaPWvDUmeYQzF3lKtN15WsAeTr65ce4mPxwopPSo1yUUAWw0hQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.34.1", "", { "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-K4Sjdo4/xF9NEeA2khOb7Y5nY6NSXBnod87uniVYW9kHP+hNlDV8trUSFeynA2uxWam4gIWgWoygPrv9VMWrYg=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils": ["@typescript-eslint/type-utils@8.36.0", "", { "dependencies": { "@typescript-eslint/typescript-estree": "8.36.0", "@typescript-eslint/utils": "8.36.0", "debug": "^4.3.4", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-5aaGYG8cVDd6cxfk/ynpYzxBRZJk7w/ymto6uiyUFtdCozQIsQWh7M28/6r57Fwkbweng8qAzoMCPwSJfWlmsg=="],
 | 
			
		||||
    "@typescript-eslint/type-utils": ["@typescript-eslint/type-utils@8.34.1", "", { "dependencies": { "@typescript-eslint/typescript-estree": "8.34.1", "@typescript-eslint/utils": "8.34.1", "debug": "^4.3.4", "ts-api-utils": "^2.1.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-Tv7tCCr6e5m8hP4+xFugcrwTOucB8lshffJ6zf1mF1TbU67R+ntCc6DzLNKM+s/uzDyv8gLq7tufaAhIBYeV8g=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
    "@typescript-eslint/types": ["@typescript-eslint/types@8.32.0", "", {}, "sha512-O5Id6tGadAZEMThM6L9HmVf5hQUXNSxLVKeGJYWNhhVseps/0LddMkp7//VDkzwJ69lPL0UmZdcZwggj9akJaA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.34.1", "", { "dependencies": { "@typescript-eslint/project-service": "8.34.1", "@typescript-eslint/tsconfig-utils": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.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 <5.9.0" } }, "sha512-rjCNqqYPuMUF5ODD+hWBNmOitjBWghkGKJg6hiCHzUvXRy6rK22Jd3rwbP2Xi+R7oYVvIKhokHVhH41BxPV5mA=="],
 | 
			
		||||
    "@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.32.0", "", { "dependencies": { "@typescript-eslint/types": "8.32.0", "@typescript-eslint/visitor-keys": "8.32.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 <5.9.0" } }, "sha512-pU9VD7anSCOIoBFnhTGfOzlVFQIA1XXiQpH/CezqOBaDppRwTglJzCC6fUQGpfwey4T183NKhF1/mfatYmjRqQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/utils": ["@typescript-eslint/utils@8.34.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/typescript-estree": "8.34.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-mqOwUdZ3KjtGk7xJJnLbHxTuWVn3GO2WZZuM+Slhkun4+qthLdXx32C8xIXbO1kfCECb3jIs3eoxK3eryk7aoQ=="],
 | 
			
		||||
    "@typescript-eslint/utils": ["@typescript-eslint/utils@8.32.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.32.0", "@typescript-eslint/types": "8.32.0", "@typescript-eslint/typescript-estree": "8.32.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-8S9hXau6nQ/sYVtC3D6ISIDoJzS1NsCK+gluVhLN2YkBPX+/1wkwyUiDKnxRh15579WoOIyVWnoyIf3yGI9REw=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.36.0", "", { "dependencies": { "@typescript-eslint/types": "8.36.0", "eslint-visitor-keys": "^4.2.1" } }, "sha512-vZrhV2lRPWDuGoxcmrzRZyxAggPL+qp3WzUrlZD+slFueDiYHxeBa34dUXPuC0RmGKzl4lS5kFJYvKCq9cnNDA=="],
 | 
			
		||||
    "@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "eslint-visitor-keys": "^4.2.1" } }, "sha512-xoh5rJ+tgsRKoXnkBPFRLZ7rjKM0AfVbC68UZ/ECXoDbfggb9RbEySN359acY1vS3qZ0jVTVWzbtfapwm5ztxw=="],
 | 
			
		||||
 | 
			
		||||
    "@ungap/structured-clone": ["@ungap/structured-clone@1.3.0", "", {}, "sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g=="],
 | 
			
		||||
 | 
			
		||||
    "@vitejs/plugin-react": ["@vitejs/plugin-react@4.6.0", "", { "dependencies": { "@babel/core": "^7.27.4", "@babel/plugin-transform-react-jsx-self": "^7.27.1", "@babel/plugin-transform-react-jsx-source": "^7.27.1", "@rolldown/pluginutils": "1.0.0-beta.19", "@types/babel__core": "^7.20.5", "react-refresh": "^0.17.0" }, "peerDependencies": { "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0-beta.0" } }, "sha512-5Kgff+m8e2PB+9j51eGHEpn5kUzRKH2Ry0qGoe8ItJg7pqnkPrYPkDQZGgGmTa0EGarHrkjLvOdU3b1fzI8otQ=="],
 | 
			
		||||
    "@vitejs/plugin-react": ["@vitejs/plugin-react@4.5.2", "", { "dependencies": { "@babel/core": "^7.27.4", "@babel/plugin-transform-react-jsx-self": "^7.27.1", "@babel/plugin-transform-react-jsx-source": "^7.27.1", "@rolldown/pluginutils": "1.0.0-beta.11", "@types/babel__core": "^7.20.5", "react-refresh": "^0.17.0" }, "peerDependencies": { "vite": "^4.2.0 || ^5.0.0 || ^6.0.0 || ^7.0.0-beta.0" } }, "sha512-QNVT3/Lxx99nMQWJWF7K4N6apUEuT0KlZA3mx/mVaoGj3smm/8rc8ezz15J1pcbcjDK0V15rpHetVfya08r76Q=="],
 | 
			
		||||
 | 
			
		||||
    "acorn": ["acorn@8.15.0", "", { "bin": { "acorn": "bin/acorn" } }, "sha512-NZyJarBfL7nWwIq+FDL6Zp/yHEhePMNnnJ0y3qfieCrmNvYct8uvtiV41UvlSe6apAfk0fY1FbWx+NwfmpvtTg=="],
 | 
			
		||||
 | 
			
		||||
@@ -494,7 +494,7 @@
 | 
			
		||||
 | 
			
		||||
    "escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
 | 
			
		||||
 | 
			
		||||
    "eslint": ["eslint@9.30.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", "@eslint-community/regexpp": "^4.12.1", "@eslint/config-array": "^0.21.0", "@eslint/config-helpers": "^0.3.0", "@eslint/core": "^0.14.0", "@eslint/eslintrc": "^3.3.1", "@eslint/js": "9.30.1", "@eslint/plugin-kit": "^0.3.1", "@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-zmxXPNMOXmwm9E0yQLi5uqXHs7uq2UIiqEKo3Gq+3fwo1XrJ+hijAZImyF7hclW3E6oHz43Yk3RP8at6OTKflQ=="],
 | 
			
		||||
    "eslint": ["eslint@9.29.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.2.0", "@eslint-community/regexpp": "^4.12.1", "@eslint/config-array": "^0.20.1", "@eslint/config-helpers": "^0.2.1", "@eslint/core": "^0.14.0", "@eslint/eslintrc": "^3.3.1", "@eslint/js": "9.29.0", "@eslint/plugin-kit": "^0.3.1", "@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-GsGizj2Y1rCWDu6XoEekL3RLilp0voSePurjZIkxL3wlm5o5EC9VpgaP7lrCvjnkuLvzFBQWB3vWB3K5KQTveQ=="],
 | 
			
		||||
 | 
			
		||||
    "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=="],
 | 
			
		||||
 | 
			
		||||
@@ -528,7 +528,7 @@
 | 
			
		||||
 | 
			
		||||
    "fastq": ["fastq@1.19.1", "", { "dependencies": { "reusify": "^1.0.4" } }, "sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ=="],
 | 
			
		||||
 | 
			
		||||
    "fdir": ["fdir@6.4.6", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-hiFoqpyZcfNm1yc4u8oWCf9A2c4D3QjCrks3zmoVKVxpQRzmPNar1hUJcBG2RQHvEVGDN+Jm81ZheVLAQMK6+w=="],
 | 
			
		||||
    "fdir": ["fdir@6.4.4", "", { "peerDependencies": { "picomatch": "^3 || ^4" }, "optionalPeers": ["picomatch"] }, "sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg=="],
 | 
			
		||||
 | 
			
		||||
    "file-entry-cache": ["file-entry-cache@8.0.0", "", { "dependencies": { "flat-cache": "^4.0.0" } }, "sha512-XXTUwCvisa5oacNGRP9SfNtYBNAMi+RPwBFmblZEF7N7swHYQS6/Zfk7SRwx4D5j3CH211YNRco1DEMNVfZCnQ=="],
 | 
			
		||||
 | 
			
		||||
@@ -558,7 +558,7 @@
 | 
			
		||||
 | 
			
		||||
    "glob-parent": ["glob-parent@6.0.2", "", { "dependencies": { "is-glob": "^4.0.3" } }, "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A=="],
 | 
			
		||||
 | 
			
		||||
    "globals": ["globals@16.3.0", "", {}, "sha512-bqWEnJ1Nt3neqx2q5SFfGS8r/ahumIakg3HcwtNlrVlwXIeNumWn/c7Pn/wKzGhf6SaW6H6uWXLqC30STCMchQ=="],
 | 
			
		||||
    "globals": ["globals@16.2.0", "", {}, "sha512-O+7l9tPdHCU320IigZZPj5zmRCFG9xHmx9cU8FqU2Rp+JN714seHV+2S9+JslCpY4gJwU2vOGox0wzgae/MCEg=="],
 | 
			
		||||
 | 
			
		||||
    "gopd": ["gopd@1.2.0", "", {}, "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg=="],
 | 
			
		||||
 | 
			
		||||
@@ -582,7 +582,7 @@
 | 
			
		||||
 | 
			
		||||
    "html-url-attributes": ["html-url-attributes@3.0.1", "", {}, "sha512-ol6UPyBWqsrO6EJySPz2O7ZSr856WDrEzM5zMqp+FJJLGMW35cLYmmZnl0vztAZxRUoNZJFTCohfjuIJ8I4QBQ=="],
 | 
			
		||||
 | 
			
		||||
    "i18next": ["i18next@25.3.2", "", { "dependencies": { "@babel/runtime": "^7.27.6" }, "peerDependencies": { "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-JSnbZDxRVbphc5jiptxr3o2zocy5dEqpVm9qCGdJwRNO+9saUJS0/u4LnM/13C23fUEWxAylPqKU/NpMV/IjqA=="],
 | 
			
		||||
    "i18next": ["i18next@25.2.1", "", { "dependencies": { "@babel/runtime": "^7.27.1" }, "peerDependencies": { "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-+UoXK5wh+VlE1Zy5p6MjcvctHXAhRwQKCxiJD8noKZzIXmnAX8gdHX5fLPA3MEVxEN4vbZkQFy8N0LyD9tUqPw=="],
 | 
			
		||||
 | 
			
		||||
    "i18next-browser-languagedetector": ["i18next-browser-languagedetector@8.2.0", "", { "dependencies": { "@babel/runtime": "^7.23.2" } }, "sha512-P+3zEKLnOF0qmiesW383vsLdtQVyKtCNA9cjSoKCppTKPQVfKd2W8hbVo5ZhNJKDqeM7BOcvNoKJOjpHh4Js9g=="],
 | 
			
		||||
 | 
			
		||||
@@ -636,27 +636,27 @@
 | 
			
		||||
 | 
			
		||||
    "levn": ["levn@0.4.1", "", { "dependencies": { "prelude-ls": "^1.2.1", "type-check": "~0.4.0" } }, "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss": ["lightningcss@1.30.1", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-darwin-arm64": "1.30.1", "lightningcss-darwin-x64": "1.30.1", "lightningcss-freebsd-x64": "1.30.1", "lightningcss-linux-arm-gnueabihf": "1.30.1", "lightningcss-linux-arm64-gnu": "1.30.1", "lightningcss-linux-arm64-musl": "1.30.1", "lightningcss-linux-x64-gnu": "1.30.1", "lightningcss-linux-x64-musl": "1.30.1", "lightningcss-win32-arm64-msvc": "1.30.1", "lightningcss-win32-x64-msvc": "1.30.1" } }, "sha512-xi6IyHML+c9+Q3W0S4fCQJOym42pyurFiJUHEcEyHS0CeKzia4yZDEsLlqOFykxOdHpNy0NmvVO31vcSqAxJCg=="],
 | 
			
		||||
    "lightningcss": ["lightningcss@1.29.2", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-darwin-arm64": "1.29.2", "lightningcss-darwin-x64": "1.29.2", "lightningcss-freebsd-x64": "1.29.2", "lightningcss-linux-arm-gnueabihf": "1.29.2", "lightningcss-linux-arm64-gnu": "1.29.2", "lightningcss-linux-arm64-musl": "1.29.2", "lightningcss-linux-x64-gnu": "1.29.2", "lightningcss-linux-x64-musl": "1.29.2", "lightningcss-win32-arm64-msvc": "1.29.2", "lightningcss-win32-x64-msvc": "1.29.2" } }, "sha512-6b6gd/RUXKaw5keVdSEtqFVdzWnU5jMxTUjA2bVcMNPLwSQ08Sv/UodBVtETLCn7k4S1Ibxwh7k68IwLZPgKaA=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.30.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-c8JK7hyE65X1MHMN+Viq9n11RRC7hgin3HhYKhrMyaXflk5GVplZ60IxyoVtzILeKr+xAJwg6zK6sjTBJ0FKYQ=="],
 | 
			
		||||
    "lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.29.2", "", { "os": "darwin", "cpu": "arm64" }, "sha512-cK/eMabSViKn/PG8U/a7aCorpeKLMlK0bQeNHmdb7qUnBkNPnL+oV5DjJUo0kqWsJUapZsM4jCfYItbqBDvlcA=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-darwin-x64": ["lightningcss-darwin-x64@1.30.1", "", { "os": "darwin", "cpu": "x64" }, "sha512-k1EvjakfumAQoTfcXUcHQZhSpLlkAuEkdMBsI/ivWw9hL+7FtilQc0Cy3hrx0AAQrVtQAbMI7YjCgYgvn37PzA=="],
 | 
			
		||||
    "lightningcss-darwin-x64": ["lightningcss-darwin-x64@1.29.2", "", { "os": "darwin", "cpu": "x64" }, "sha512-j5qYxamyQw4kDXX5hnnCKMf3mLlHvG44f24Qyi2965/Ycz829MYqjrVg2H8BidybHBp9kom4D7DR5VqCKDXS0w=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-freebsd-x64": ["lightningcss-freebsd-x64@1.30.1", "", { "os": "freebsd", "cpu": "x64" }, "sha512-kmW6UGCGg2PcyUE59K5r0kWfKPAVy4SltVeut+umLCFoJ53RdCUWxcRDzO1eTaxf/7Q2H7LTquFHPL5R+Gjyig=="],
 | 
			
		||||
    "lightningcss-freebsd-x64": ["lightningcss-freebsd-x64@1.29.2", "", { "os": "freebsd", "cpu": "x64" }, "sha512-wDk7M2tM78Ii8ek9YjnY8MjV5f5JN2qNVO+/0BAGZRvXKtQrBC4/cn4ssQIpKIPP44YXw6gFdpUF+Ps+RGsCwg=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-linux-arm-gnueabihf": ["lightningcss-linux-arm-gnueabihf@1.30.1", "", { "os": "linux", "cpu": "arm" }, "sha512-MjxUShl1v8pit+6D/zSPq9S9dQ2NPFSQwGvxBCYaBYLPlCWuPh9/t1MRS8iUaR8i+a6w7aps+B4N0S1TYP/R+Q=="],
 | 
			
		||||
    "lightningcss-linux-arm-gnueabihf": ["lightningcss-linux-arm-gnueabihf@1.29.2", "", { "os": "linux", "cpu": "arm" }, "sha512-IRUrOrAF2Z+KExdExe3Rz7NSTuuJ2HvCGlMKoquK5pjvo2JY4Rybr+NrKnq0U0hZnx5AnGsuFHjGnNT14w26sg=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-linux-arm64-gnu": ["lightningcss-linux-arm64-gnu@1.30.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-gB72maP8rmrKsnKYy8XUuXi/4OctJiuQjcuqWNlJQ6jZiWqtPvqFziskH3hnajfvKB27ynbVCucKSm2rkQp4Bw=="],
 | 
			
		||||
    "lightningcss-linux-arm64-gnu": ["lightningcss-linux-arm64-gnu@1.29.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-KKCpOlmhdjvUTX/mBuaKemp0oeDIBBLFiU5Fnqxh1/DZ4JPZi4evEH7TKoSBFOSOV3J7iEmmBaw/8dpiUvRKlQ=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-linux-arm64-musl": ["lightningcss-linux-arm64-musl@1.30.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-jmUQVx4331m6LIX+0wUhBbmMX7TCfjF5FoOH6SD1CttzuYlGNVpA7QnrmLxrsub43ClTINfGSYyHe2HWeLl5CQ=="],
 | 
			
		||||
    "lightningcss-linux-arm64-musl": ["lightningcss-linux-arm64-musl@1.29.2", "", { "os": "linux", "cpu": "arm64" }, "sha512-Q64eM1bPlOOUgxFmoPUefqzY1yV3ctFPE6d/Vt7WzLW4rKTv7MyYNky+FWxRpLkNASTnKQUaiMJ87zNODIrrKQ=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-linux-x64-gnu": ["lightningcss-linux-x64-gnu@1.30.1", "", { "os": "linux", "cpu": "x64" }, "sha512-piWx3z4wN8J8z3+O5kO74+yr6ze/dKmPnI7vLqfSqI8bccaTGY5xiSGVIJBDd5K5BHlvVLpUB3S2YCfelyJ1bw=="],
 | 
			
		||||
    "lightningcss-linux-x64-gnu": ["lightningcss-linux-x64-gnu@1.29.2", "", { "os": "linux", "cpu": "x64" }, "sha512-0v6idDCPG6epLXtBH/RPkHvYx74CVziHo6TMYga8O2EiQApnUPZsbR9nFNrg2cgBzk1AYqEd95TlrsL7nYABQg=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-linux-x64-musl": ["lightningcss-linux-x64-musl@1.30.1", "", { "os": "linux", "cpu": "x64" }, "sha512-rRomAK7eIkL+tHY0YPxbc5Dra2gXlI63HL+v1Pdi1a3sC+tJTcFrHX+E86sulgAXeI7rSzDYhPSeHHjqFhqfeQ=="],
 | 
			
		||||
    "lightningcss-linux-x64-musl": ["lightningcss-linux-x64-musl@1.29.2", "", { "os": "linux", "cpu": "x64" }, "sha512-rMpz2yawkgGT8RULc5S4WiZopVMOFWjiItBT7aSfDX4NQav6M44rhn5hjtkKzB+wMTRlLLqxkeYEtQ3dd9696w=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-win32-arm64-msvc": ["lightningcss-win32-arm64-msvc@1.30.1", "", { "os": "win32", "cpu": "arm64" }, "sha512-mSL4rqPi4iXq5YVqzSsJgMVFENoa4nGTT/GjO2c0Yl9OuQfPsIfncvLrEW6RbbB24WtZ3xP/2CCmI3tNkNV4oA=="],
 | 
			
		||||
    "lightningcss-win32-arm64-msvc": ["lightningcss-win32-arm64-msvc@1.29.2", "", { "os": "win32", "cpu": "arm64" }, "sha512-nL7zRW6evGQqYVu/bKGK+zShyz8OVzsCotFgc7judbt6wnB2KbiKKJwBE4SGoDBQ1O94RjW4asrCjQL4i8Fhbw=="],
 | 
			
		||||
 | 
			
		||||
    "lightningcss-win32-x64-msvc": ["lightningcss-win32-x64-msvc@1.30.1", "", { "os": "win32", "cpu": "x64" }, "sha512-PVqXh48wh4T53F/1CCu8PIPCxLzWyCnn/9T5W1Jpmdy5h9Cwd+0YQS6/LwhHXSafuc61/xg9Lv5OrCby6a++jg=="],
 | 
			
		||||
    "lightningcss-win32-x64-msvc": ["lightningcss-win32-x64-msvc@1.29.2", "", { "os": "win32", "cpu": "x64" }, "sha512-EdIUW3B2vLuHmv7urfzMI/h2fmlnOQBk1xlsDxkN1tCWKjNFjfLhGxYk8C8mzpSfr+A6jFFIi8fU6LbQGsRWjA=="],
 | 
			
		||||
 | 
			
		||||
    "locate-path": ["locate-path@6.0.0", "", { "dependencies": { "p-locate": "^5.0.0" } }, "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw=="],
 | 
			
		||||
 | 
			
		||||
@@ -666,7 +666,7 @@
 | 
			
		||||
 | 
			
		||||
    "lru-cache": ["lru-cache@5.1.1", "", { "dependencies": { "yallist": "^3.0.2" } }, "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w=="],
 | 
			
		||||
 | 
			
		||||
    "lucide-react": ["lucide-react@0.525.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-Tm1txJ2OkymCGkvwoHt33Y2JpN5xucVq1slHcgE6Lk0WjDfjgKWor5CdVER8U6DvcfMwh4M8XxmpTiyzfmfDYQ=="],
 | 
			
		||||
    "lucide-react": ["lucide-react@0.516.0", "", { "peerDependencies": { "react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, "sha512-aybBJzLHcw1CIn3rUcRkztB37dsJATtpffLNX+0/w+ws2p21nYIlOwX/B5fqxq8F/BjqVemnJX8chKwRidvROg=="],
 | 
			
		||||
 | 
			
		||||
    "magic-string": ["magic-string@0.30.17", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.0" } }, "sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA=="],
 | 
			
		||||
 | 
			
		||||
@@ -774,11 +774,11 @@
 | 
			
		||||
 | 
			
		||||
    "picomatch": ["picomatch@4.0.2", "", {}, "sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg=="],
 | 
			
		||||
 | 
			
		||||
    "postcss": ["postcss@8.5.6", "", { "dependencies": { "nanoid": "^3.3.11", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-3Ybi1tAuwAP9s0r1UQ2J4n5Y0G05bJkpUIO0/bI9MhwmD70S5aTWbXGBwxHrelT+XM1k6dM0pk+SwNkpTRN7Pg=="],
 | 
			
		||||
    "postcss": ["postcss@8.5.3", "", { "dependencies": { "nanoid": "^3.3.8", "picocolors": "^1.1.1", "source-map-js": "^1.2.1" } }, "sha512-dle9A3yYxlBSrt8Fu+IpjGT8SY8hN0mlaA6GY8t0P5PjIOZemULz/E2Bnm/2dcUOena75OTNkHI76uZBNUUq3A=="],
 | 
			
		||||
 | 
			
		||||
    "prelude-ls": ["prelude-ls@1.2.1", "", {}, "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g=="],
 | 
			
		||||
 | 
			
		||||
    "prettier": ["prettier@3.6.2", "", { "bin": { "prettier": "bin/prettier.cjs" } }, "sha512-I7AIg5boAr5R0FFtJ6rCfD+LFsWHp81dolrFD8S79U9tb8Az2nGrJncnMSnys+bpQJfRUzqs9hnA81OAA3hCuQ=="],
 | 
			
		||||
    "prettier": ["prettier@3.5.3", "", { "bin": { "prettier": "bin/prettier.cjs" } }, "sha512-QQtaxnoDJeAkDvDKWCLiwIXkTgRhwYDEQCghU9Z6q03iyek/rxRh/2lC3HB7P8sWT2xC/y5JDctPLBIGzHKbhw=="],
 | 
			
		||||
 | 
			
		||||
    "property-information": ["property-information@7.1.0", "", {}, "sha512-TwEZ+X+yCJmYfL7TPUOcvBZ4QfoT5YenQiJuX//0th53DE6w0xxLEtfK3iyryQFddXuvkIk51EEgrJQ0WJkOmQ=="],
 | 
			
		||||
 | 
			
		||||
@@ -792,9 +792,9 @@
 | 
			
		||||
 | 
			
		||||
    "react-dom": ["react-dom@19.1.0", "", { "dependencies": { "scheduler": "^0.26.0" }, "peerDependencies": { "react": "^19.1.0" } }, "sha512-Xs1hdnE+DyKgeHJeJznQmYMIBG3TKIHJJT95Q58nHLSrElKlGQqDTR2HQ9fx5CN/Gk6Vh/kupBTDLU11/nDk/g=="],
 | 
			
		||||
 | 
			
		||||
    "react-hook-form": ["react-hook-form@7.60.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17 || ^18 || ^19" } }, "sha512-SBrYOvMbDB7cV8ZfNpaiLcgjH/a1c7aK0lK+aNigpf4xWLO8q+o4tcvVurv3c4EOyzn/3dCsYt4GKD42VvJ/+A=="],
 | 
			
		||||
    "react-hook-form": ["react-hook-form@7.58.0", "", { "peerDependencies": { "react": "^16.8.0 || ^17 || ^18 || ^19" } }, "sha512-zGijmEed35oNfOfy7ub99jfjkiLhHwA3dl5AgyKdWC6QQzhnc7tkWewSa+T+A2EpLrc6wo5DUoZctS9kufWJjA=="],
 | 
			
		||||
 | 
			
		||||
    "react-i18next": ["react-i18next@15.6.0", "", { "dependencies": { "@babel/runtime": "^7.27.6", "html-parse-stringify": "^3.0.1" }, "peerDependencies": { "i18next": ">= 23.2.3", "react": ">= 16.8.0", "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-W135dB0rDfiFmbMipC17nOhGdttO5mzH8BivY+2ybsQBbXvxWIwl3cmeH3T9d+YPBSJu/ouyJKFJTtkK7rJofw=="],
 | 
			
		||||
    "react-i18next": ["react-i18next@15.5.3", "", { "dependencies": { "@babel/runtime": "^7.27.6", "html-parse-stringify": "^3.0.1" }, "peerDependencies": { "i18next": ">= 23.2.3", "react": ">= 16.8.0", "typescript": "^5" }, "optionalPeers": ["typescript"] }, "sha512-ypYmOKOnjqPEJZO4m1BI0kS8kWqkBNsKYyhVUfij0gvjy9xJNoG/VcGkxq5dRlVwzmrmY1BQMAmpbbUBLwC4Kw=="],
 | 
			
		||||
 | 
			
		||||
    "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=="],
 | 
			
		||||
 | 
			
		||||
@@ -804,7 +804,7 @@
 | 
			
		||||
 | 
			
		||||
    "react-remove-scroll-bar": ["react-remove-scroll-bar@2.3.8", "", { "dependencies": { "react-style-singleton": "^2.2.2", "tslib": "^2.0.0" }, "peerDependencies": { "@types/react": "*", "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" }, "optionalPeers": ["@types/react"] }, "sha512-9r+yi9+mgU33AKcj6IbT9oRCO78WriSj6t/cF8DWBZJ9aOGPOTEDvdUDz1FwKim7QXWwmHqtdHnRJfhAxEG46Q=="],
 | 
			
		||||
 | 
			
		||||
    "react-router": ["react-router@7.6.3", "", { "dependencies": { "cookie": "^1.0.1", "set-cookie-parser": "^2.6.0" }, "peerDependencies": { "react": ">=18", "react-dom": ">=18" }, "optionalPeers": ["react-dom"] }, "sha512-zf45LZp5skDC6I3jDLXQUu0u26jtuP4lEGbc7BbdyxenBN1vJSTA18czM2D+h5qyMBuMrD+9uB+mU37HIoKGRA=="],
 | 
			
		||||
    "react-router": ["react-router@7.6.2", "", { "dependencies": { "cookie": "^1.0.1", "set-cookie-parser": "^2.6.0" }, "peerDependencies": { "react": ">=18", "react-dom": ">=18" }, "optionalPeers": ["react-dom"] }, "sha512-U7Nv3y+bMimgWjhlT5CRdzHPu2/KVmqPwKUCChW8en5P3znxUqwlYFlbmyj8Rgp1SF6zs5X4+77kBVknkg6a0w=="],
 | 
			
		||||
 | 
			
		||||
    "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=="],
 | 
			
		||||
 | 
			
		||||
@@ -830,7 +830,7 @@
 | 
			
		||||
 | 
			
		||||
    "shebang-regex": ["shebang-regex@3.0.0", "", {}, "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A=="],
 | 
			
		||||
 | 
			
		||||
    "sonner": ["sonner@2.0.6", "", { "peerDependencies": { "react": "^18.0.0 || ^19.0.0 || ^19.0.0-rc", "react-dom": "^18.0.0 || ^19.0.0 || ^19.0.0-rc" } }, "sha512-yHFhk8T/DK3YxjFQXIrcHT1rGEeTLliVzWbO0xN8GberVun2RiBnxAjXAYpZrqwEVHBG9asI/Li8TAAhN9m59Q=="],
 | 
			
		||||
    "sonner": ["sonner@2.0.5", "", { "peerDependencies": { "react": "^18.0.0 || ^19.0.0 || ^19.0.0-rc", "react-dom": "^18.0.0 || ^19.0.0 || ^19.0.0-rc" } }, "sha512-YwbHQO6cSso3HBXlbCkgrgzDNIhws14r4MO87Ofy+cV2X7ES4pOoAK3+veSmVTvqNx1BWUxlhPmZzP00Crk2aQ=="],
 | 
			
		||||
 | 
			
		||||
    "source-map-js": ["source-map-js@1.2.1", "", {}, "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA=="],
 | 
			
		||||
 | 
			
		||||
@@ -848,13 +848,13 @@
 | 
			
		||||
 | 
			
		||||
    "tailwind-merge": ["tailwind-merge@3.3.1", "", {}, "sha512-gBXpgUm/3rp1lMZZrM/w7D8GKqshif0zAymAhbCyIt8KMe+0v9DQ7cdYLR4FHH/cKpdTXb+A/tKKU3eolfsI+g=="],
 | 
			
		||||
 | 
			
		||||
    "tailwindcss": ["tailwindcss@4.1.11", "", {}, "sha512-2E9TBm6MDD/xKYe+dvJZAmg3yxIEDNRc0jwlNyDg/4Fil2QcSLjFKGVff0lAf1jjeaArlG/M75Ey/EYr/OJtBA=="],
 | 
			
		||||
    "tailwindcss": ["tailwindcss@4.1.10", "", {}, "sha512-P3nr6WkvKV/ONsTzj6Gb57sWPMX29EPNPopo7+FcpkQaNsrNpZ1pv8QmrYI2RqEKD7mlGqLnGovlcYnBK0IqUA=="],
 | 
			
		||||
 | 
			
		||||
    "tapable": ["tapable@2.2.1", "", {}, "sha512-GNzQvQTOIP6RyTfE2Qxb8ZVlNmw0n88vp1szwWRimP02mnTsx3Wtn5qRdqY9w2XduFNUgvOwhNnQsjwCp+kqaQ=="],
 | 
			
		||||
 | 
			
		||||
    "tar": ["tar@7.4.3", "", { "dependencies": { "@isaacs/fs-minipass": "^4.0.0", "chownr": "^3.0.0", "minipass": "^7.1.2", "minizlib": "^3.0.1", "mkdirp": "^3.0.1", "yallist": "^5.0.0" } }, "sha512-5S7Va8hKfV7W5U6g3aYxXmlPoZVAwUMy9AOKyF2fVuZa2UD3qZjg578OrLRt8PcNN1PleVaL/5/yYATNL0ICUw=="],
 | 
			
		||||
 | 
			
		||||
    "tinyglobby": ["tinyglobby@0.2.14", "", { "dependencies": { "fdir": "^6.4.4", "picomatch": "^4.0.2" } }, "sha512-tX5e7OM1HnYr2+a2C/4V0htOcSQcoSTH9KgJnVvNm5zm/cyEWKJ7j7YutsH9CxMdtOkkLFy2AHrMci9IM8IPZQ=="],
 | 
			
		||||
    "tinyglobby": ["tinyglobby@0.2.13", "", { "dependencies": { "fdir": "^6.4.4", "picomatch": "^4.0.2" } }, "sha512-mEwzpUgrLySlveBwEVDMKk5B57bhLPYovRfPAXD5gA/98Opn0rCDj3GtLwFvCvH5RK9uPCExUROW5NjDwvqkxw=="],
 | 
			
		||||
 | 
			
		||||
    "to-regex-range": ["to-regex-range@5.0.1", "", { "dependencies": { "is-number": "^7.0.0" } }, "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ=="],
 | 
			
		||||
 | 
			
		||||
@@ -866,13 +866,13 @@
 | 
			
		||||
 | 
			
		||||
    "tslib": ["tslib@2.8.1", "", {}, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
 | 
			
		||||
 | 
			
		||||
    "tw-animate-css": ["tw-animate-css@1.3.5", "", {}, "sha512-t3u+0YNoloIhj1mMXs779P6MO9q3p3mvGn4k1n3nJPqJw/glZcuijG2qTSN4z4mgNRfW5ZC3aXJFLwDtiipZXA=="],
 | 
			
		||||
    "tw-animate-css": ["tw-animate-css@1.3.4", "", {}, "sha512-dd1Ht6/YQHcNbq0znIT6dG8uhO7Ce+VIIhZUhjsryXsMPJQz3bZg7Q2eNzLwipb25bRZslGb2myio5mScd1TFg=="],
 | 
			
		||||
 | 
			
		||||
    "type-check": ["type-check@0.4.0", "", { "dependencies": { "prelude-ls": "^1.2.1" } }, "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew=="],
 | 
			
		||||
 | 
			
		||||
    "typescript": ["typescript@5.8.3", "", { "bin": { "tsc": "bin/tsc", "tsserver": "bin/tsserver" } }, "sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint": ["typescript-eslint@8.36.0", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.36.0", "@typescript-eslint/parser": "8.36.0", "@typescript-eslint/utils": "8.36.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-fTCqxthY+h9QbEgSIBfL9iV6CvKDFuoxg6bHPNpJ9HIUzS+jy2lCEyCmGyZRWEBSaykqcDPf1SJ+BfCI8DRopA=="],
 | 
			
		||||
    "typescript-eslint": ["typescript-eslint@8.34.1", "", { "dependencies": { "@typescript-eslint/eslint-plugin": "8.34.1", "@typescript-eslint/parser": "8.34.1", "@typescript-eslint/utils": "8.34.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-XjS+b6Vg9oT1BaIUfkW3M3LvqZE++rbzAMEHuccCfO/YkP43ha6w3jTEMilQxMF92nVOYCcdjv1ZUhAa1D/0ow=="],
 | 
			
		||||
 | 
			
		||||
    "undici-types": ["undici-types@7.8.0", "", {}, "sha512-9UJ2xGDvQ43tYyVMpuHlsgApydB8ZKfVYTsLDhXkFL/6gfkp+U8xTGdh8pMJv1SpZna0zxG1DwsKZsreLbXBxw=="],
 | 
			
		||||
 | 
			
		||||
@@ -900,7 +900,7 @@
 | 
			
		||||
 | 
			
		||||
    "vfile-message": ["vfile-message@4.0.2", "", { "dependencies": { "@types/unist": "^3.0.0", "unist-util-stringify-position": "^4.0.0" } }, "sha512-jRDZ1IMLttGj41KcZvlrYAaI3CfqpLpfpf+Mfig13viT6NKvRzWZ+lXz0Y5D60w6uJIBAOGq9mSHf0gktF0duw=="],
 | 
			
		||||
 | 
			
		||||
    "vite": ["vite@7.0.3", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.6", "picomatch": "^4.0.2", "postcss": "^8.5.6", "rollup": "^4.40.0", "tinyglobby": "^0.2.14" }, "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-y2L5oJZF7bj4c0jgGYgBNSdIu+5HF+m68rn2cQXFbGoShdhV1phX9rbnxy9YXj82aS8MMsCLAAFkRxZeWdldrQ=="],
 | 
			
		||||
    "vite": ["vite@6.3.5", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.4", "picomatch": "^4.0.2", "postcss": "^8.5.3", "rollup": "^4.34.9", "tinyglobby": "^0.2.13" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "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-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ=="],
 | 
			
		||||
 | 
			
		||||
    "void-elements": ["void-elements@3.1.0", "", {}, "sha512-Dhxzh5HZuiHQhbvTW9AMetFfBHDMYpo23Uo9btPXgdYP+3T5S+p+jgNy7spra+veYhBP2dCSgxR/i2Y02h5/6w=="],
 | 
			
		||||
 | 
			
		||||
@@ -912,7 +912,7 @@
 | 
			
		||||
 | 
			
		||||
    "yocto-queue": ["yocto-queue@0.1.0", "", {}, "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q=="],
 | 
			
		||||
 | 
			
		||||
    "zod": ["zod@3.25.76", "", {}, "sha512-gzUt/qt81nXsFGKIFcC3YnfEAx5NkunCfnDlvuBSSFS02bcXu4Lmea0AFIUwbLWxWPx3d9p8S5QoaujKcNQxcQ=="],
 | 
			
		||||
    "zod": ["zod@3.25.67", "", {}, "sha512-idA2YXwpCdqUSKRCACDE6ItZD9TZzy3OZMtpfLoh6oPR47lipysRrJfjzMqFxQ3uJuUPyUeWe1r9vLH33xO/Qw=="],
 | 
			
		||||
 | 
			
		||||
    "zwitch": ["zwitch@2.0.4", "", {}, "sha512-bXE4cR/kVZhKZX/RjPEflHaKVhUVl85noU3v6b8apfQEc1x4A+zBxjZ4lN8LqGd6WZ3dl98pY4o717VFmoPp+A=="],
 | 
			
		||||
 | 
			
		||||
@@ -934,13 +934,15 @@
 | 
			
		||||
 | 
			
		||||
    "@humanfs/node/@humanwhocodes/retry": ["@humanwhocodes/retry@0.3.1", "", {}, "sha512-JBxkERygn7Bv/GbN5Rv8Ul6LVknS+5Bp6RgDC/O8gEBU/yeH5Ui5C/OlWrTb6qct7LjjfT6Re2NxB0ln0yYybA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss": ["lightningcss@1.30.1", "", { "dependencies": { "detect-libc": "^2.0.3" }, "optionalDependencies": { "lightningcss-darwin-arm64": "1.30.1", "lightningcss-darwin-x64": "1.30.1", "lightningcss-freebsd-x64": "1.30.1", "lightningcss-linux-arm-gnueabihf": "1.30.1", "lightningcss-linux-arm64-gnu": "1.30.1", "lightningcss-linux-arm64-musl": "1.30.1", "lightningcss-linux-x64-gnu": "1.30.1", "lightningcss-linux-x64-musl": "1.30.1", "lightningcss-win32-arm64-msvc": "1.30.1", "lightningcss-win32-x64-msvc": "1.30.1" } }, "sha512-xi6IyHML+c9+Q3W0S4fCQJOym42pyurFiJUHEcEyHS0CeKzia4yZDEsLlqOFykxOdHpNy0NmvVO31vcSqAxJCg=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/core": ["@emnapi/core@1.4.3", "", { "dependencies": { "@emnapi/wasi-threads": "1.0.2", "tslib": "^2.4.0" }, "bundled": true }, "sha512-4m62DuCE07lw01soJwPiBGC0nAww0Q+RY70VZ+n49yDIO13yyinhbWCeNnaob0lakDtWQzSdtNWzJeOJt2ma+g=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/runtime": ["@emnapi/runtime@1.4.3", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-pBPWdu6MLKROBX05wSNKcNb++m5Er+KQ9QkB+WVM+pW2Kx9hoSrVTnu3BdkI5eBLZoKu/J6mW/B6i6bJB2ytXQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.0.2", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-5n3nTJblwRi8LlXkJ9eBzu+kZR8Yxcc7ubakyQTFzPMtIhFpUBRbsnc2Dv88IZDIbCDlBiWrknhB4Lsz7mg6BA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@0.2.11", "", { "dependencies": { "@emnapi/core": "^1.4.3", "@emnapi/runtime": "^1.4.3", "@tybys/wasm-util": "^0.9.0" }, "bundled": true }, "sha512-9DPkXtvHydrcOsopiYpUgPHpmj0HWZKMUnL2dZqpvC42lsratuBG06V5ipyno0fUek5VlFsNQ+AcFATSrJXgMA=="],
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@0.2.10", "", { "dependencies": { "@emnapi/core": "^1.4.3", "@emnapi/runtime": "^1.4.3", "@tybys/wasm-util": "^0.9.0" }, "bundled": true }, "sha512-bCsCyeZEwVErsGmyPNSzwfwFn4OdxBj0mmv6hOFucB/k81Ojdu68RbZdxYsRQUPc9l6SU5F/cG+bXgWs3oUgsQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@tybys/wasm-util": ["@tybys/wasm-util@0.9.0", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-6+7nlbMVX/PVDCwaIQ8nTOPveOcFLSt8GcXdx8hD0bt39uWxYT88uXzqTd4fTvqta7oeUJqudepapKNt2DYJFw=="],
 | 
			
		||||
 | 
			
		||||
@@ -958,45 +960,45 @@
 | 
			
		||||
 | 
			
		||||
    "@types/babel__traverse/@babel/types": ["@babel/types@7.27.1", "", { "dependencies": { "@babel/helper-string-parser": "^7.27.1", "@babel/helper-validator-identifier": "^7.27.1" } }, "sha512-+EzkxvLNfiUeKMgy/3luqfsCWFRXLb7U6wNQTk60tovuckwB15B191tJWvpp4HjiQWdJkCxO3Wbvc6jlk3Xb2Q=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.36.0", "", { "dependencies": { "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.0" } }, "sha512-wCnapIKnDkN62fYtTGv2+RY8FlnBYA3tNm0fm91kc2BjPhV2vIjwwozJ7LToaLAyb1ca8BxrS7vT+Pvvf7RvqA=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.1" } }, "sha512-beu6o6QY4hJAgL1E8RaXNC071G4Kso2MGmJskCFQhRhg8VOH/FDbC8soP8NHN7e/Hdphwp8G8cE6OBzC8o41ZA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils": ["@typescript-eslint/utils@8.36.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/typescript-estree": "8.36.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-VOqmHu42aEMT+P2qYjylw6zP/3E/HvptRwdn/PZxyV27KhZg2IOszXod4NcXisWzPAGSS4trE/g4moNj6XmH2g=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils": ["@typescript-eslint/utils@8.34.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/typescript-estree": "8.34.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-mqOwUdZ3KjtGk7xJJnLbHxTuWVn3GO2WZZuM+Slhkun4+qthLdXx32C8xIXbO1kfCECb3jIs3eoxK3eryk7aoQ=="],
 | 
			
		||||
 | 
			
		||||
    "@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.36.0", "", { "dependencies": { "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.0" } }, "sha512-wCnapIKnDkN62fYtTGv2+RY8FlnBYA3tNm0fm91kc2BjPhV2vIjwwozJ7LToaLAyb1ca8BxrS7vT+Pvvf7RvqA=="],
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.1" } }, "sha512-beu6o6QY4hJAgL1E8RaXNC071G4Kso2MGmJskCFQhRhg8VOH/FDbC8soP8NHN7e/Hdphwp8G8cE6OBzC8o41ZA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/types": ["@typescript-eslint/types@8.36.0", "", {}, "sha512-xGms6l5cTJKQPZOKM75Dl9yBfNdGeLRsIyufewnxT4vZTrjC0ImQT4fj8QmtJK84F58uSh5HVBSANwcfiXxABQ=="],
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.36.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.36.0", "@typescript-eslint/tsconfig-utils": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.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 <5.9.0" } }, "sha512-JaS8bDVrfVJX4av0jLpe4ye0BpAaUW7+tnS4Y4ETa3q7NoZgzYbN9zDQTJ8kPb5fQ4n0hliAt9tA4Pfs2zA2Hg=="],
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.34.1", "", { "dependencies": { "@typescript-eslint/project-service": "8.34.1", "@typescript-eslint/tsconfig-utils": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.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 <5.9.0" } }, "sha512-rjCNqqYPuMUF5ODD+hWBNmOitjBWghkGKJg6hiCHzUvXRy6rK22Jd3rwbP2Xi+R7oYVvIKhokHVhH41BxPV5mA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/scope-manager/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "eslint-visitor-keys": "^4.2.1" } }, "sha512-xoh5rJ+tgsRKoXnkBPFRLZ7rjKM0AfVbC68UZ/ECXoDbfggb9RbEySN359acY1vS3qZ0jVTVWzbtfapwm5ztxw=="],
 | 
			
		||||
    "@typescript-eslint/project-service/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.36.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.36.0", "@typescript-eslint/tsconfig-utils": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.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 <5.9.0" } }, "sha512-JaS8bDVrfVJX4av0jLpe4ye0BpAaUW7+tnS4Y4ETa3q7NoZgzYbN9zDQTJ8kPb5fQ4n0hliAt9tA4Pfs2zA2Hg=="],
 | 
			
		||||
    "@typescript-eslint/scope-manager/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.32.0", "", { "dependencies": { "@typescript-eslint/types": "8.32.0", "eslint-visitor-keys": "^4.2.0" } }, "sha512-1rYQTCLFFzOI5Nl0c8LUpJT8HxpwVRn9E4CkMsYfuN6ctmQqExjSTzzSk0Tz2apmXy7WU6/6fyaZVVA/thPN+w=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils": ["@typescript-eslint/utils@8.36.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/typescript-estree": "8.36.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-VOqmHu42aEMT+P2qYjylw6zP/3E/HvptRwdn/PZxyV27KhZg2IOszXod4NcXisWzPAGSS4trE/g4moNj6XmH2g=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.34.1", "", { "dependencies": { "@typescript-eslint/project-service": "8.34.1", "@typescript-eslint/tsconfig-utils": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.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 <5.9.0" } }, "sha512-rjCNqqYPuMUF5ODD+hWBNmOitjBWghkGKJg6hiCHzUvXRy6rK22Jd3rwbP2Xi+R7oYVvIKhokHVhH41BxPV5mA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "eslint-visitor-keys": "^4.2.1" } }, "sha512-xoh5rJ+tgsRKoXnkBPFRLZ7rjKM0AfVbC68UZ/ECXoDbfggb9RbEySN359acY1vS3qZ0jVTVWzbtfapwm5ztxw=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils": ["@typescript-eslint/utils@8.34.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/typescript-estree": "8.34.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-mqOwUdZ3KjtGk7xJJnLbHxTuWVn3GO2WZZuM+Slhkun4+qthLdXx32C8xIXbO1kfCECb3jIs3eoxK3eryk7aoQ=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree/@typescript-eslint/visitor-keys": ["@typescript-eslint/visitor-keys@8.32.0", "", { "dependencies": { "@typescript-eslint/types": "8.32.0", "eslint-visitor-keys": "^4.2.0" } }, "sha512-1rYQTCLFFzOI5Nl0c8LUpJT8HxpwVRn9E4CkMsYfuN6ctmQqExjSTzzSk0Tz2apmXy7WU6/6fyaZVVA/thPN+w=="],
 | 
			
		||||
 | 
			
		||||
    "@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/visitor-keys/@typescript-eslint/types": ["@typescript-eslint/types@8.36.0", "", {}, "sha512-xGms6l5cTJKQPZOKM75Dl9yBfNdGeLRsIyufewnxT4vZTrjC0ImQT4fj8QmtJK84F58uSh5HVBSANwcfiXxABQ=="],
 | 
			
		||||
    "@typescript-eslint/visitor-keys/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "fast-glob/glob-parent": ["glob-parent@5.1.2", "", { "dependencies": { "is-glob": "^4.0.1" } }, "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow=="],
 | 
			
		||||
 | 
			
		||||
    "i18next-browser-languagedetector/@babel/runtime": ["@babel/runtime@7.27.1", "", {}, "sha512-1x3D2xEk2fRo3PAhwQwu5UubzgiVWSXTBfWpVd2Mx2AzRqJuDJCsgaDVZ7HB5iGzDW1Hl1sWN2mFyKjmR9uAog=="],
 | 
			
		||||
 | 
			
		||||
    "i18next-resources-to-backend/@babel/runtime": ["@babel/runtime@7.27.1", "", {}, "sha512-1x3D2xEk2fRo3PAhwQwu5UubzgiVWSXTBfWpVd2Mx2AzRqJuDJCsgaDVZ7HB5iGzDW1Hl1sWN2mFyKjmR9uAog=="],
 | 
			
		||||
 | 
			
		||||
    "lru-cache/yallist": ["yallist@3.1.1", "", {}, "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g=="],
 | 
			
		||||
 | 
			
		||||
    "micromatch/picomatch": ["picomatch@2.3.1", "", {}, "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA=="],
 | 
			
		||||
 | 
			
		||||
    "parse-entities/@types/unist": ["@types/unist@2.0.11", "", {}, "sha512-CmBKiL6NNo/OqgmMn95Fk9Whlp2mtvIv+KNpQKN2F4SjvrEesubTRWGYSg+BnWZOnlCaSTU1sMpsBOzgbYhnsA=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils": ["@typescript-eslint/utils@8.36.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/typescript-estree": "8.36.0" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-VOqmHu42aEMT+P2qYjylw6zP/3E/HvptRwdn/PZxyV27KhZg2IOszXod4NcXisWzPAGSS4trE/g4moNj6XmH2g=="],
 | 
			
		||||
    "react-i18next/@babel/runtime": ["@babel/runtime@7.27.6", "", {}, "sha512-vbavdySgbTTrmFE+EsiqUTzlOr5bzlnJtUv9PynGCAKvfQqjIXbvFdumPM/GxMDfyuGMJaJAU6TO4zc1Jf1i8Q=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils": ["@typescript-eslint/utils@8.34.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.7.0", "@typescript-eslint/scope-manager": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/typescript-estree": "8.34.1" }, "peerDependencies": { "eslint": "^8.57.0 || ^9.0.0", "typescript": ">=4.8.4 <5.9.0" } }, "sha512-mqOwUdZ3KjtGk7xJJnLbHxTuWVn3GO2WZZuM+Slhkun4+qthLdXx32C8xIXbO1kfCECb3jIs3eoxK3eryk7aoQ=="],
 | 
			
		||||
 | 
			
		||||
    "@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=="],
 | 
			
		||||
 | 
			
		||||
@@ -1008,6 +1010,26 @@
 | 
			
		||||
 | 
			
		||||
    "@eslint/eslintrc/espree/eslint-visitor-keys": ["eslint-visitor-keys@4.2.0", "", {}, "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-darwin-arm64": ["lightningcss-darwin-arm64@1.30.1", "", { "os": "darwin", "cpu": "arm64" }, "sha512-c8JK7hyE65X1MHMN+Viq9n11RRC7hgin3HhYKhrMyaXflk5GVplZ60IxyoVtzILeKr+xAJwg6zK6sjTBJ0FKYQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-darwin-x64": ["lightningcss-darwin-x64@1.30.1", "", { "os": "darwin", "cpu": "x64" }, "sha512-k1EvjakfumAQoTfcXUcHQZhSpLlkAuEkdMBsI/ivWw9hL+7FtilQc0Cy3hrx0AAQrVtQAbMI7YjCgYgvn37PzA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-freebsd-x64": ["lightningcss-freebsd-x64@1.30.1", "", { "os": "freebsd", "cpu": "x64" }, "sha512-kmW6UGCGg2PcyUE59K5r0kWfKPAVy4SltVeut+umLCFoJ53RdCUWxcRDzO1eTaxf/7Q2H7LTquFHPL5R+Gjyig=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-linux-arm-gnueabihf": ["lightningcss-linux-arm-gnueabihf@1.30.1", "", { "os": "linux", "cpu": "arm" }, "sha512-MjxUShl1v8pit+6D/zSPq9S9dQ2NPFSQwGvxBCYaBYLPlCWuPh9/t1MRS8iUaR8i+a6w7aps+B4N0S1TYP/R+Q=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-linux-arm64-gnu": ["lightningcss-linux-arm64-gnu@1.30.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-gB72maP8rmrKsnKYy8XUuXi/4OctJiuQjcuqWNlJQ6jZiWqtPvqFziskH3hnajfvKB27ynbVCucKSm2rkQp4Bw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-linux-arm64-musl": ["lightningcss-linux-arm64-musl@1.30.1", "", { "os": "linux", "cpu": "arm64" }, "sha512-jmUQVx4331m6LIX+0wUhBbmMX7TCfjF5FoOH6SD1CttzuYlGNVpA7QnrmLxrsub43ClTINfGSYyHe2HWeLl5CQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-linux-x64-gnu": ["lightningcss-linux-x64-gnu@1.30.1", "", { "os": "linux", "cpu": "x64" }, "sha512-piWx3z4wN8J8z3+O5kO74+yr6ze/dKmPnI7vLqfSqI8bccaTGY5xiSGVIJBDd5K5BHlvVLpUB3S2YCfelyJ1bw=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-linux-x64-musl": ["lightningcss-linux-x64-musl@1.30.1", "", { "os": "linux", "cpu": "x64" }, "sha512-rRomAK7eIkL+tHY0YPxbc5Dra2gXlI63HL+v1Pdi1a3sC+tJTcFrHX+E86sulgAXeI7rSzDYhPSeHHjqFhqfeQ=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-win32-arm64-msvc": ["lightningcss-win32-arm64-msvc@1.30.1", "", { "os": "win32", "cpu": "arm64" }, "sha512-mSL4rqPi4iXq5YVqzSsJgMVFENoa4nGTT/GjO2c0Yl9OuQfPsIfncvLrEW6RbbB24WtZ3xP/2CCmI3tNkNV4oA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/node/lightningcss/lightningcss-win32-x64-msvc": ["lightningcss-win32-x64-msvc@1.30.1", "", { "os": "win32", "cpu": "x64" }, "sha512-PVqXh48wh4T53F/1CCu8PIPCxLzWyCnn/9T5W1Jpmdy5h9Cwd+0YQS6/LwhHXSafuc61/xg9Lv5OrCby6a++jg=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/core/@emnapi/wasi-threads": ["@emnapi/wasi-threads@1.0.2", "", { "dependencies": { "tslib": "^2.4.0" }, "bundled": true }, "sha512-5n3nTJblwRi8LlXkJ9eBzu+kZR8Yxcc7ubakyQTFzPMtIhFpUBRbsnc2Dv88IZDIbCDlBiWrknhB4Lsz7mg6BA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/core/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
 | 
			
		||||
@@ -1018,48 +1040,40 @@
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@tybys/wasm-util/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager/@typescript-eslint/types": ["@typescript-eslint/types@8.36.0", "", {}, "sha512-xGms6l5cTJKQPZOKM75Dl9yBfNdGeLRsIyufewnxT4vZTrjC0ImQT4fj8QmtJK84F58uSh5HVBSANwcfiXxABQ=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/scope-manager/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.36.0", "", {}, "sha512-xGms6l5cTJKQPZOKM75Dl9yBfNdGeLRsIyufewnxT4vZTrjC0ImQT4fj8QmtJK84F58uSh5HVBSANwcfiXxABQ=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.36.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.36.0", "@typescript-eslint/tsconfig-utils": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.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 <5.9.0" } }, "sha512-JaS8bDVrfVJX4av0jLpe4ye0BpAaUW7+tnS4Y4ETa3q7NoZgzYbN9zDQTJ8kPb5fQ4n0hliAt9tA4Pfs2zA2Hg=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/typescript-estree/@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.36.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.36.0", "@typescript-eslint/types": "^8.36.0", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-JAhQFIABkWccQYeLMrHadu/fhpzmSQ1F1KXkpzqiVxA/iYI6UnRt2trqXHt1sYEcw1mxLnB9rKMsOxXPxowN/g=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/typescript-estree/@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.36.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-Nhh3TIEgN18mNbdXpd5Q8mSCBnrZQeY9V7Ca3dqYvNDStNIGRmJA6dmrIPMJ0kow3C7gcQbpsG2rPzy1Ks/AnA=="],
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.34.1", "", { "dependencies": { "@typescript-eslint/project-service": "8.34.1", "@typescript-eslint/tsconfig-utils": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.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 <5.9.0" } }, "sha512-rjCNqqYPuMUF5ODD+hWBNmOitjBWghkGKJg6hiCHzUvXRy6rK22Jd3rwbP2Xi+R7oYVvIKhokHVhH41BxPV5mA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/typescript-estree/minimatch": ["minimatch@9.0.5", "", { "dependencies": { "brace-expansion": "^2.0.1" } }, "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/parser/@typescript-eslint/typescript-estree/semver": ["semver@7.7.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree/@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.36.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.36.0", "@typescript-eslint/types": "^8.36.0", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-JAhQFIABkWccQYeLMrHadu/fhpzmSQ1F1KXkpzqiVxA/iYI6UnRt2trqXHt1sYEcw1mxLnB9rKMsOxXPxowN/g=="],
 | 
			
		||||
    "@typescript-eslint/scope-manager/@typescript-eslint/visitor-keys/eslint-visitor-keys": ["eslint-visitor-keys@4.2.0", "", {}, "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree/@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.36.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-Nhh3TIEgN18mNbdXpd5Q8mSCBnrZQeY9V7Ca3dqYvNDStNIGRmJA6dmrIPMJ0kow3C7gcQbpsG2rPzy1Ks/AnA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree/@typescript-eslint/types": ["@typescript-eslint/types@8.36.0", "", {}, "sha512-xGms6l5cTJKQPZOKM75Dl9yBfNdGeLRsIyufewnxT4vZTrjC0ImQT4fj8QmtJK84F58uSh5HVBSANwcfiXxABQ=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree/minimatch": ["minimatch@9.0.5", "", { "dependencies": { "brace-expansion": "^2.0.1" } }, "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/typescript-estree/semver": ["semver@7.7.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.36.0", "", { "dependencies": { "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.0" } }, "sha512-wCnapIKnDkN62fYtTGv2+RY8FlnBYA3tNm0fm91kc2BjPhV2vIjwwozJ7LToaLAyb1ca8BxrS7vT+Pvvf7RvqA=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.1" } }, "sha512-beu6o6QY4hJAgL1E8RaXNC071G4Kso2MGmJskCFQhRhg8VOH/FDbC8soP8NHN7e/Hdphwp8G8cE6OBzC8o41ZA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.36.0", "", {}, "sha512-xGms6l5cTJKQPZOKM75Dl9yBfNdGeLRsIyufewnxT4vZTrjC0ImQT4fj8QmtJK84F58uSh5HVBSANwcfiXxABQ=="],
 | 
			
		||||
    "@typescript-eslint/type-utils/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/typescript-estree/@typescript-eslint/visitor-keys/eslint-visitor-keys": ["eslint-visitor-keys@4.2.0", "", {}, "sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw=="],
 | 
			
		||||
 | 
			
		||||
    "@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-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.36.0", "", { "dependencies": { "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.0" } }, "sha512-wCnapIKnDkN62fYtTGv2+RY8FlnBYA3tNm0fm91kc2BjPhV2vIjwwozJ7LToaLAyb1ca8BxrS7vT+Pvvf7RvqA=="],
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/scope-manager": ["@typescript-eslint/scope-manager@8.34.1", "", { "dependencies": { "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.1" } }, "sha512-beu6o6QY4hJAgL1E8RaXNC071G4Kso2MGmJskCFQhRhg8VOH/FDbC8soP8NHN7e/Hdphwp8G8cE6OBzC8o41ZA=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.36.0", "", {}, "sha512-xGms6l5cTJKQPZOKM75Dl9yBfNdGeLRsIyufewnxT4vZTrjC0ImQT4fj8QmtJK84F58uSh5HVBSANwcfiXxABQ=="],
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/types": ["@typescript-eslint/types@8.34.1", "", {}, "sha512-rjLVbmE7HR18kDsjNIZQHxmv9RZwlgzavryL5Lnj2ujIRTeXlKtILHgRNmQ3j4daw7zd+mQgy+uyt6Zo6I0IGA=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.36.0", "", { "dependencies": { "@typescript-eslint/project-service": "8.36.0", "@typescript-eslint/tsconfig-utils": "8.36.0", "@typescript-eslint/types": "8.36.0", "@typescript-eslint/visitor-keys": "8.36.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 <5.9.0" } }, "sha512-JaS8bDVrfVJX4av0jLpe4ye0BpAaUW7+tnS4Y4ETa3q7NoZgzYbN9zDQTJ8kPb5fQ4n0hliAt9tA4Pfs2zA2Hg=="],
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/typescript-estree": ["@typescript-eslint/typescript-estree@8.34.1", "", { "dependencies": { "@typescript-eslint/project-service": "8.34.1", "@typescript-eslint/tsconfig-utils": "8.34.1", "@typescript-eslint/types": "8.34.1", "@typescript-eslint/visitor-keys": "8.34.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 <5.9.0" } }, "sha512-rjCNqqYPuMUF5ODD+hWBNmOitjBWghkGKJg6hiCHzUvXRy6rK22Jd3rwbP2Xi+R7oYVvIKhokHVhH41BxPV5mA=="],
 | 
			
		||||
 | 
			
		||||
    "@tailwindcss/oxide-wasm32-wasi/@emnapi/core/@emnapi/wasi-threads/tslib": ["tslib@2.8.1", "", { "bundled": true }, "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.36.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.36.0", "@typescript-eslint/types": "^8.36.0", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-JAhQFIABkWccQYeLMrHadu/fhpzmSQ1F1KXkpzqiVxA/iYI6UnRt2trqXHt1sYEcw1mxLnB9rKMsOxXPxowN/g=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.36.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-Nhh3TIEgN18mNbdXpd5Q8mSCBnrZQeY9V7Ca3dqYvNDStNIGRmJA6dmrIPMJ0kow3C7gcQbpsG2rPzy1Ks/AnA=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/typescript-estree/minimatch": ["minimatch@9.0.5", "", { "dependencies": { "brace-expansion": "^2.0.1" } }, "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow=="],
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/eslint-plugin/@typescript-eslint/utils/@typescript-eslint/typescript-estree/semver": ["semver@7.7.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA=="],
 | 
			
		||||
@@ -1068,10 +1082,6 @@
 | 
			
		||||
 | 
			
		||||
    "@typescript-eslint/type-utils/@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-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/project-service": ["@typescript-eslint/project-service@8.36.0", "", { "dependencies": { "@typescript-eslint/tsconfig-utils": "^8.36.0", "@typescript-eslint/types": "^8.36.0", "debug": "^4.3.4" }, "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-JAhQFIABkWccQYeLMrHadu/fhpzmSQ1F1KXkpzqiVxA/iYI6UnRt2trqXHt1sYEcw1mxLnB9rKMsOxXPxowN/g=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/typescript-estree/@typescript-eslint/tsconfig-utils": ["@typescript-eslint/tsconfig-utils@8.36.0", "", { "peerDependencies": { "typescript": ">=4.8.4 <5.9.0" } }, "sha512-Nhh3TIEgN18mNbdXpd5Q8mSCBnrZQeY9V7Ca3dqYvNDStNIGRmJA6dmrIPMJ0kow3C7gcQbpsG2rPzy1Ks/AnA=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/typescript-estree/minimatch": ["minimatch@9.0.5", "", { "dependencies": { "brace-expansion": "^2.0.1" } }, "sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow=="],
 | 
			
		||||
 | 
			
		||||
    "typescript-eslint/@typescript-eslint/utils/@typescript-eslint/typescript-estree/semver": ["semver@7.7.1", "", { "bin": { "semver": "bin/semver.js" } }, "sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA=="],
 | 
			
		||||
 
 | 
			
		||||
@@ -15,44 +15,44 @@
 | 
			
		||||
    "@radix-ui/react-select": "^2.2.5",
 | 
			
		||||
    "@radix-ui/react-separator": "^1.1.7",
 | 
			
		||||
    "@radix-ui/react-slot": "^1.2.3",
 | 
			
		||||
    "@tailwindcss/vite": "^4.1.11",
 | 
			
		||||
    "@tanstack/react-query": "^5.81.5",
 | 
			
		||||
    "@tailwindcss/vite": "^4.1.10",
 | 
			
		||||
    "@tanstack/react-query": "^5.80.7",
 | 
			
		||||
    "axios": "^1.10.0",
 | 
			
		||||
    "class-variance-authority": "^0.7.1",
 | 
			
		||||
    "clsx": "^2.1.1",
 | 
			
		||||
    "dompurify": "^3.2.6",
 | 
			
		||||
    "i18next": "^25.3.2",
 | 
			
		||||
    "i18next": "^25.2.1",
 | 
			
		||||
    "i18next-browser-languagedetector": "^8.2.0",
 | 
			
		||||
    "i18next-resources-to-backend": "^1.2.1",
 | 
			
		||||
    "input-otp": "^1.4.2",
 | 
			
		||||
    "lucide-react": "^0.525.0",
 | 
			
		||||
    "lucide-react": "^0.516.0",
 | 
			
		||||
    "next-themes": "^0.4.6",
 | 
			
		||||
    "react": "^19.0.0",
 | 
			
		||||
    "react-dom": "^19.0.0",
 | 
			
		||||
    "react-hook-form": "^7.60.0",
 | 
			
		||||
    "react-i18next": "^15.6.0",
 | 
			
		||||
    "react-hook-form": "^7.58.0",
 | 
			
		||||
    "react-i18next": "^15.5.3",
 | 
			
		||||
    "react-markdown": "^10.1.0",
 | 
			
		||||
    "react-router": "^7.6.3",
 | 
			
		||||
    "sonner": "^2.0.6",
 | 
			
		||||
    "react-router": "^7.6.2",
 | 
			
		||||
    "sonner": "^2.0.5",
 | 
			
		||||
    "tailwind-merge": "^3.3.1",
 | 
			
		||||
    "tailwindcss": "^4.1.11",
 | 
			
		||||
    "zod": "^3.25.76"
 | 
			
		||||
    "tailwindcss": "^4.1.10",
 | 
			
		||||
    "zod": "^3.25.67"
 | 
			
		||||
  },
 | 
			
		||||
  "devDependencies": {
 | 
			
		||||
    "@eslint/js": "^9.30.1",
 | 
			
		||||
    "@tanstack/eslint-plugin-query": "^5.81.2",
 | 
			
		||||
    "@types/node": "^24.0.12",
 | 
			
		||||
    "@eslint/js": "^9.29.0",
 | 
			
		||||
    "@tanstack/eslint-plugin-query": "^5.78.0",
 | 
			
		||||
    "@types/node": "^24.0.3",
 | 
			
		||||
    "@types/react": "^19.1.8",
 | 
			
		||||
    "@types/react-dom": "^19.1.6",
 | 
			
		||||
    "@vitejs/plugin-react": "^4.6.0",
 | 
			
		||||
    "eslint": "^9.30.1",
 | 
			
		||||
    "@vitejs/plugin-react": "^4.5.2",
 | 
			
		||||
    "eslint": "^9.29.0",
 | 
			
		||||
    "eslint-plugin-react-hooks": "^5.2.0",
 | 
			
		||||
    "eslint-plugin-react-refresh": "^0.4.19",
 | 
			
		||||
    "globals": "^16.3.0",
 | 
			
		||||
    "prettier": "3.6.2",
 | 
			
		||||
    "tw-animate-css": "^1.3.5",
 | 
			
		||||
    "globals": "^16.2.0",
 | 
			
		||||
    "prettier": "3.5.3",
 | 
			
		||||
    "tw-animate-css": "^1.3.4",
 | 
			
		||||
    "typescript": "~5.8.3",
 | 
			
		||||
    "typescript-eslint": "^8.36.0",
 | 
			
		||||
    "vite": "^7.0.3"
 | 
			
		||||
    "typescript-eslint": "^8.34.1",
 | 
			
		||||
    "vite": "^6.3.1"
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
@@ -33,9 +33,9 @@ export const LoginForm = (props: Props) => {
 | 
			
		||||
          control={form.control}
 | 
			
		||||
          name="username"
 | 
			
		||||
          render={({ field }) => (
 | 
			
		||||
            <FormItem className="mb-4 gap-0">
 | 
			
		||||
              <FormLabel className="mb-2">{t("loginUsername")}</FormLabel>
 | 
			
		||||
              <FormControl className="mb-1">
 | 
			
		||||
            <FormItem className="mb-4">
 | 
			
		||||
              <FormLabel>{t("loginUsername")}</FormLabel>
 | 
			
		||||
              <FormControl>
 | 
			
		||||
                <Input
 | 
			
		||||
                  placeholder={t("loginUsername")}
 | 
			
		||||
                  disabled={loading}
 | 
			
		||||
@@ -50,8 +50,8 @@ export const LoginForm = (props: Props) => {
 | 
			
		||||
          control={form.control}
 | 
			
		||||
          name="password"
 | 
			
		||||
          render={({ field }) => (
 | 
			
		||||
            <FormItem className="mb-4 gap-0">
 | 
			
		||||
              <div className="relative mb-1">
 | 
			
		||||
            <FormItem className="mb-4">
 | 
			
		||||
              <div className="relative">
 | 
			
		||||
                <FormLabel className="mb-2">{t("loginPassword")}</FormLabel>
 | 
			
		||||
                <FormControl>
 | 
			
		||||
                  <Input
 | 
			
		||||
@@ -61,14 +61,14 @@ export const LoginForm = (props: Props) => {
 | 
			
		||||
                    {...field}
 | 
			
		||||
                  />
 | 
			
		||||
                </FormControl>
 | 
			
		||||
                <FormMessage />
 | 
			
		||||
                <a
 | 
			
		||||
                  href="/forgot-password"
 | 
			
		||||
                  className="text-muted-foreground text-sm absolute right-0 bottom-[2.565rem]" // 2.565 is *just* perfect
 | 
			
		||||
                  className="text-muted-foreground text-sm absolute right-0 bottom-10"
 | 
			
		||||
                >
 | 
			
		||||
                  {t("forgotPasswordTitle")}
 | 
			
		||||
                </a>
 | 
			
		||||
              </div>
 | 
			
		||||
              <FormMessage />
 | 
			
		||||
            </FormItem>
 | 
			
		||||
          )}
 | 
			
		||||
        />
 | 
			
		||||
 
 | 
			
		||||
@@ -136,7 +136,7 @@ h4 {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
p {
 | 
			
		||||
  @apply leading-6;
 | 
			
		||||
  @apply leading-6 [&:not(:first-child)]:mt-6;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
blockquote {
 | 
			
		||||
 
 | 
			
		||||
@@ -27,8 +27,8 @@ export const languages = {
 | 
			
		||||
  "tr-TR": "Türkçe",
 | 
			
		||||
  "uk-UA": "Українська",
 | 
			
		||||
  "vi-VN": "Tiếng Việt",
 | 
			
		||||
  "zh-CN": "简体中文",
 | 
			
		||||
  "zh-TW": "繁體中文(台灣)",
 | 
			
		||||
  "zh-CN": "中文",
 | 
			
		||||
  "zh-TW": "中文",
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
export type SupportedLanguage = keyof typeof languages;
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "مرحبا بعودتك، ادخل باستخدام",
 | 
			
		||||
    "loginTitleSimple": "مرحبا بعودتك، سجل دخولك",
 | 
			
		||||
    "loginDivider": "أو",
 | 
			
		||||
    "loginTitle": "مرحبا بعودتك، قم بتسجيل الدخول باستخدام",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "Or",
 | 
			
		||||
    "loginUsername": "اسم المستخدم",
 | 
			
		||||
    "loginPassword": "كلمة المرور",
 | 
			
		||||
    "loginSubmit": "تسجيل الدخول",
 | 
			
		||||
@@ -10,8 +10,8 @@
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "تم تسجيل الدخول",
 | 
			
		||||
    "loginSuccessSubtitle": "مرحبا بعودتك!",
 | 
			
		||||
    "loginOauthFailTitle": "حدث خطأ",
 | 
			
		||||
    "loginOauthFailSubtitle": "أخفق الحصول على رابط OAuth",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailSubtitle": "فشل في الحصول على رابط OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "إعادة توجيه",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "إعادة توجيه إلى مزود OAuth الخاص بك",
 | 
			
		||||
    "continueRedirectingTitle": "إعادة توجيه...",
 | 
			
		||||
@@ -19,7 +19,7 @@
 | 
			
		||||
    "continueInvalidRedirectTitle": "إعادة توجيه غير صالحة",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "رابط إعادة التوجيه غير صالح",
 | 
			
		||||
    "continueInsecureRedirectTitle": "إعادة توجيه غير آمنة",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "أنت تحاول إعادة التوجيه من <code>https</code> إلى <code>http</code>، هل أنت متأكد أنك تريد المتابعة؟",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "متابعة",
 | 
			
		||||
    "continueSubtitle": "انقر الزر للمتابعة إلى التطبيق الخاص بك.",
 | 
			
		||||
    "logoutFailTitle": "فشل تسجيل الخروج",
 | 
			
		||||
@@ -32,7 +32,7 @@
 | 
			
		||||
    "notFoundTitle": "الصفحة غير موجودة",
 | 
			
		||||
    "notFoundSubtitle": "الصفحة التي تبحث عنها غير موجودة.",
 | 
			
		||||
    "notFoundButton": "انتقل إلى الرئيسية",
 | 
			
		||||
    "totpFailTitle": "أخفق التحقق من الرمز",
 | 
			
		||||
    "totpFailTitle": "فشل في التحقق من الرمز",
 | 
			
		||||
    "totpFailSubtitle": "الرجاء التحقق من الرمز الخاص بك وحاول مرة أخرى",
 | 
			
		||||
    "totpSuccessTitle": "تم التحقق",
 | 
			
		||||
    "totpSuccessSubtitle": "إعادة توجيه إلى تطبيقك",
 | 
			
		||||
@@ -42,13 +42,12 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "حاول مجددا",
 | 
			
		||||
    "untrustedRedirectTitle": "إعادة توجيه غير موثوقة",
 | 
			
		||||
    "untrustedRedirectSubtitle": "أنت تحاول إعادة التوجيه إلى نطاق لا يتطابق مع النطاق المكون الخاص بك (<code>{{domain}}</code>). هل أنت متأكد من أنك تريد المتابعة؟",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "إلغاء",
 | 
			
		||||
    "forgotPasswordTitle": "نسيت كلمة المرور؟",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
    "errorTitle": "حدث خطأ",
 | 
			
		||||
    "errorTitle": "An error occurred",
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information."
 | 
			
		||||
}
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Brugeren med brugernavnet <code>{{username}}</code> har ikke tilladelse til at tilgå ressourcen <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Brugeren med brugernavnet <code>{{username}}</code> har ikke tilladelse til at logge ind.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Brugeren med brugernavnet <code>{{username}}</code> er ikke i de grupper, som ressourcen <code>{{resource}}</code> kræver.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Din IP adresse <code>{{ip}}</code> er ikke autoriseret til at tilgå ressourcen <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Prøv igen",
 | 
			
		||||
    "untrustedRedirectTitle": "Usikker omdirigering",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Du forsøger at omdirigere til et domæne, der ikke matcher dit konfigurerede domæne (<code>{{domain}}</code>). Er du sikker på, at du vil fortsætte?",
 | 
			
		||||
 
 | 
			
		||||
@@ -7,7 +7,7 @@
 | 
			
		||||
    "loginSubmit": "Anmelden",
 | 
			
		||||
    "loginFailTitle": "Login fehlgeschlagen",
 | 
			
		||||
    "loginFailSubtitle": "Bitte überprüfe deinen Benutzernamen und Passwort",
 | 
			
		||||
    "loginFailRateLimit": "Zu viele fehlgeschlagene Loginversuche. Versuche es später erneut",
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "Angemeldet",
 | 
			
		||||
    "loginSuccessSubtitle": "Willkommen zurück!",
 | 
			
		||||
    "loginOauthFailTitle": "Ein Fehler ist aufgetreten",
 | 
			
		||||
@@ -19,7 +19,7 @@
 | 
			
		||||
    "continueInvalidRedirectTitle": "Ungültige Weiterleitung",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "Die Weiterleitungs-URL ist ungültig",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Unsichere Weiterleitung",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Sie versuchen von <code>https</code> auf <code>http</code> weiterzuleiten, was unsicher ist. Sind Sie sicher, dass Sie fortfahren möchten?",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Weiter",
 | 
			
		||||
    "continueSubtitle": "Klicken Sie auf den Button, um zur App zu gelangen.",
 | 
			
		||||
    "logoutFailTitle": "Abmelden fehlgeschlagen",
 | 
			
		||||
@@ -27,8 +27,8 @@
 | 
			
		||||
    "logoutSuccessTitle": "Abgemeldet",
 | 
			
		||||
    "logoutSuccessSubtitle": "Sie wurden abgemeldet",
 | 
			
		||||
    "logoutTitle": "Abmelden",
 | 
			
		||||
    "logoutUsernameSubtitle": "Sie sind derzeit als <code>{{username}}</code> angemeldet. Klicken Sie auf den Button unten, um sich abzumelden.",
 | 
			
		||||
    "logoutOauthSubtitle": "Sie sind derzeit als <code>{{username}}</code> über den OAuth-Anbieter {{provider}} angemeldet. Klicken Sie auf den Button unten, um sich abzumelden.",
 | 
			
		||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
			
		||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
			
		||||
    "notFoundTitle": "Seite nicht gefunden",
 | 
			
		||||
    "notFoundSubtitle": "Die gesuchte Seite existiert nicht.",
 | 
			
		||||
    "notFoundButton": "Nach Hause",
 | 
			
		||||
@@ -37,18 +37,17 @@
 | 
			
		||||
    "totpSuccessTitle": "Verifiziert",
 | 
			
		||||
    "totpSuccessSubtitle": "Leite zur App weiter",
 | 
			
		||||
    "totpTitle": "Geben Sie Ihren TOTP Code ein",
 | 
			
		||||
    "totpSubtitle": "Bitte geben Sie den Code aus Ihrer Authenticator-App ein.",
 | 
			
		||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
			
		||||
    "unauthorizedTitle": "Unautorisiert",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Der Benutzer mit Benutzername <code>{{username}}</code> ist nicht berechtigt, auf die Ressource <code>{{resource}}</code> zuzugreifen.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Der Benutzer mit Benutzername <code>{{username}}</code> ist nicht berechtigt, sich anzumelden.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Der Benutzer mit Benutzername <code>{{username}}</code> ist nicht in den Gruppen, die von der Ressource <code>{{resource}}</code> benötigt werden.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Ihre IP-Adresse <code>{{ip}}</code> ist nicht berechtigt, auf die Ressource <code>{{resource}}</code> zuzugreifen.",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Erneut versuchen",
 | 
			
		||||
    "untrustedRedirectTitle": "Nicht vertrauenswürdige Weiterleitung",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Sie versuchen auf eine Domain umzuleiten, die nicht mit Ihrer konfigurierten Domain übereinstimmt (<code>{{domain}}</code>). Sind Sie sicher, dass Sie fortfahren möchten?",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Abbrechen",
 | 
			
		||||
    "forgotPasswordTitle": "Passwort vergessen?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Fehler beim Laden der Authentifizierungsanbieter. Bitte überprüfen Sie Ihre Konfiguration.",
 | 
			
		||||
    "errorTitle": "Ein Fehler ist aufgetreten",
 | 
			
		||||
    "errorSubtitle": "Beim Versuch, diese Aktion auszuführen, ist ein Fehler aufgetreten. Bitte überprüfen Sie die Konsole für weitere Informationen."
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
    "errorTitle": "An error occurred",
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information."
 | 
			
		||||
}
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Ο χρήστης με όνομα χρήστη <code>{{username}}</code> δεν έχει άδεια πρόσβασης στον πόρο <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Ο χρήστης με όνομα χρήστη <code>{{username}}</code> δεν είναι εξουσιοδοτημένος να συνδεθεί.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Ο χρήστης με όνομα χρήστη <code>{{username}}</code> δεν είναι στις ομάδες που απαιτούνται από τον πόρο <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Η διεύθυνση IP σας <code>{{ip}}</code> δεν είναι εξουσιοδοτημένη να έχει πρόσβαση στον πόρο <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Προσπαθήστε ξανά",
 | 
			
		||||
    "untrustedRedirectTitle": "Μη έμπιστη ανακατεύθυνση",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Προσπαθείτε να ανακατευθύνετε σε ένα domain που δεν ταιριάζει με τον ρυθμισμένο domain σας (<code>{{domain}}</code>). Είστε βέβαιοι ότι θέλετε να συνεχίσετε;",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,54 +1,53 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "Bienvenido de vuelta, inicie sesión con",
 | 
			
		||||
    "loginTitleSimple": "Bienvenido de vuelta, por favor inicie sesión",
 | 
			
		||||
    "loginDivider": "O",
 | 
			
		||||
    "loginUsername": "Usuario",
 | 
			
		||||
    "loginPassword": "Contraseña",
 | 
			
		||||
    "loginSubmit": "Iniciar sesión",
 | 
			
		||||
    "loginFailTitle": "Fallo al iniciar sesión",
 | 
			
		||||
    "loginFailSubtitle": "Por favor revise su usuario y contraseña",
 | 
			
		||||
    "loginFailRateLimit": "Muchos inicios de sesión consecutivos fallidos. Por favor inténtelo más tarde",
 | 
			
		||||
    "loginSuccessTitle": "Sesión iniciada",
 | 
			
		||||
    "loginSuccessSubtitle": "¡Bienvenido de vuelta!",
 | 
			
		||||
    "loginOauthFailTitle": "Ocurrió un error",
 | 
			
		||||
    "loginTitle": "Bienvenido de nuevo, inicie sesión con",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "Or",
 | 
			
		||||
    "loginUsername": "Username",
 | 
			
		||||
    "loginPassword": "Password",
 | 
			
		||||
    "loginSubmit": "Login",
 | 
			
		||||
    "loginFailTitle": "Failed to log in",
 | 
			
		||||
    "loginFailSubtitle": "Please check your username and password",
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "Logged in",
 | 
			
		||||
    "loginSuccessSubtitle": "Welcome back!",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailSubtitle": "Error al obtener la URL de OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redireccionando",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redireccionando a tu proveedor de OAuth",
 | 
			
		||||
    "continueRedirectingTitle": "Redireccionando...",
 | 
			
		||||
    "continueRedirectingSubtitle": "Pronto será redirigido a la aplicación",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Redirección inválida",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "La URL de redirección es inválida",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Redirección insegura",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "Está intentando redirigir desde <code>https</code> a <code>http</code> lo cual no es seguro. ¿Está seguro que desea continuar?",
 | 
			
		||||
    "continueTitle": "Continuar",
 | 
			
		||||
    "continueSubtitle": "Haga clic en el botón para continuar hacia su aplicación.",
 | 
			
		||||
    "logoutFailTitle": "Fallo al cerrar sesión",
 | 
			
		||||
    "logoutFailSubtitle": "Por favor intente nuevamente",
 | 
			
		||||
    "logoutSuccessTitle": "Sesión cerrada",
 | 
			
		||||
    "logoutSuccessSubtitle": "Su sesión ha sido cerrada",
 | 
			
		||||
    "logoutTitle": "Cerrar sesión",
 | 
			
		||||
    "logoutUsernameSubtitle": "Actualmente está conectado como <code>{{username}}</code>. Haga clic en el botón de abajo para cerrar sesión.",
 | 
			
		||||
    "logoutOauthSubtitle": "Actualmente está conectado como <code>{{username}}</code> usando {{provider}} como su proveedor de OAuth. Haga clic en el botón de abajo para cerrar sesión.",
 | 
			
		||||
    "notFoundTitle": "Página no encontrada",
 | 
			
		||||
    "notFoundSubtitle": "La página que está buscando no existe.",
 | 
			
		||||
    "notFoundButton": "Volver al inicio",
 | 
			
		||||
    "totpFailTitle": "Error al verificar código",
 | 
			
		||||
    "totpFailSubtitle": "Por favor compruebe su código e inténtelo de nuevo",
 | 
			
		||||
    "totpSuccessTitle": "Verificado",
 | 
			
		||||
    "totpSuccessSubtitle": "Redirigiendo a su aplicación",
 | 
			
		||||
    "totpTitle": "Ingrese su código TOTP",
 | 
			
		||||
    "totpSubtitle": "Por favor introduzca el código de su aplicación de autenticación.",
 | 
			
		||||
    "unauthorizedTitle": "No autorizado",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "El usuario con nombre de usuario <code>{{username}}</code> no está autorizado para acceder al recurso <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "El usuario con nombre de usuario <code>{{username}}</code> no está autorizado a iniciar sesión.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "El usuario con nombre de usuario <code>{{username}}</code> no está en los grupos requeridos por el recurso <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Inténtelo de nuevo",
 | 
			
		||||
    "untrustedRedirectTitle": "Redirección no confiable",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Está intentando redirigir a un dominio que no coincide con su dominio configurado (<code>{{domain}}</code>). ¿Está seguro que desea continuar?",
 | 
			
		||||
    "cancelTitle": "Cancelar",
 | 
			
		||||
    "forgotPasswordTitle": "¿Olvidó su contraseña?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Error al cargar los proveedores de autenticación. Por favor revise su configuración.",
 | 
			
		||||
    "errorTitle": "Ha ocurrido un error",
 | 
			
		||||
    "errorSubtitle": "Ocurrió un error mientras se trataba de realizar esta acción. Por favor, revise la consola para más información."
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
    "logoutSuccessSubtitle": "You have been logged out",
 | 
			
		||||
    "logoutTitle": "Logout",
 | 
			
		||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
			
		||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
			
		||||
    "notFoundTitle": "Page not found",
 | 
			
		||||
    "notFoundSubtitle": "The page you are looking for does not exist.",
 | 
			
		||||
    "notFoundButton": "Go home",
 | 
			
		||||
    "totpFailTitle": "Failed to verify code",
 | 
			
		||||
    "totpFailSubtitle": "Please check your code and try again",
 | 
			
		||||
    "totpSuccessTitle": "Verified",
 | 
			
		||||
    "totpSuccessSubtitle": "Redirecting to your app",
 | 
			
		||||
    "totpTitle": "Enter your TOTP code",
 | 
			
		||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
			
		||||
    "unauthorizedTitle": "Unauthorized",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
    "errorTitle": "An error occurred",
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information."
 | 
			
		||||
}
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "Connecté",
 | 
			
		||||
    "loginSuccessSubtitle": "Bienvenue!",
 | 
			
		||||
    "loginOauthFailTitle": "Une erreur s'est produite",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailSubtitle": "Impossible d'obtenir l'URL OAuth",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirection",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirection vers votre fournisseur OAuth",
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Réessayer",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Opnieuw proberen",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,7 +1,7 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "Witaj ponownie, zaloguj się przez",
 | 
			
		||||
    "loginTitleSimple": "Witaj ponownie, zaloguj się",
 | 
			
		||||
    "loginDivider": "Lub",
 | 
			
		||||
    "loginDivider": "lub",
 | 
			
		||||
    "loginUsername": "Nazwa użytkownika",
 | 
			
		||||
    "loginPassword": "Hasło",
 | 
			
		||||
    "loginSubmit": "Zaloguj się",
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Użytkownik o nazwie użytkownika <code>{{username}}</code> nie ma uprawnień dostępu do zasobu <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Użytkownik o nazwie <code>{{username}}</code> nie jest upoważniony do zalogowania się.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Użytkownik o nazwie <code>{{username}}</code> nie należy do grup wymaganych przez zasób <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Twój adres IP <code>{{ip}}</code> nie ma autoryzacji do dostępu do zasobu <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Spróbuj ponownie",
 | 
			
		||||
    "untrustedRedirectTitle": "Niezaufane przekierowanie",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Próbujesz przekierować do domeny, która nie pasuje do Twojej skonfigurowanej domeny (<code>{{domain}}</code>). Czy na pewno chcesz kontynuować?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Tentar novamente",
 | 
			
		||||
    "untrustedRedirectTitle": "Redirecionamento não confiável",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "Пользователю <code>{{username}}</code> не разрешен доступ к <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "Пользователю <code>{{username}}</code> не разрешен вход.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "Пользователь <code>{{username}}</code> не состоит в группах, которым разрешен доступ к <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Ваш IP адрес <code>{{ip}}</code> не авторизован для доступа к ресурсу <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Повторить",
 | 
			
		||||
    "untrustedRedirectTitle": "Ненадежное перенаправление",
 | 
			
		||||
    "untrustedRedirectSubtitle": "Попытка перенаправить на домен, который не соответствует вашему заданному домену (<code>{{domain}}</code>). Уверены, что хотите продолжить?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -42,7 +42,6 @@
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
 
 | 
			
		||||
@@ -1,16 +1,16 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "欢迎回来,请使用以下方式登录",
 | 
			
		||||
    "loginTitleSimple": "欢迎回来,请登录",
 | 
			
		||||
    "loginTitle": "欢迎回来,请登录",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "或",
 | 
			
		||||
    "loginUsername": "用户名",
 | 
			
		||||
    "loginPassword": "密码",
 | 
			
		||||
    "loginSubmit": "登录",
 | 
			
		||||
    "loginFailTitle": "登录失败",
 | 
			
		||||
    "loginFailSubtitle": "请检查您的用户名和密码",
 | 
			
		||||
    "loginFailRateLimit": "您登录失败次数过多。请稍后再试",
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "已登录",
 | 
			
		||||
    "loginSuccessSubtitle": "欢迎回来!",
 | 
			
		||||
    "loginOauthFailTitle": "发生错误",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailSubtitle": "获取 OAuth URL 失败",
 | 
			
		||||
    "loginOauthSuccessTitle": "重定向中",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "重定向到您的 OAuth 提供商",
 | 
			
		||||
@@ -19,7 +19,7 @@
 | 
			
		||||
    "continueInvalidRedirectTitle": "无效的重定向",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "重定向URL无效",
 | 
			
		||||
    "continueInsecureRedirectTitle": "不安全的重定向",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "您正在尝试从<code>https</code>重定向到<code>http</code>可能存在风险。您确定要继续吗?",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "继续",
 | 
			
		||||
    "continueSubtitle": "点击按钮以继续您的应用。",
 | 
			
		||||
    "logoutFailTitle": "注销失败",
 | 
			
		||||
@@ -27,8 +27,8 @@
 | 
			
		||||
    "logoutSuccessTitle": "已登出",
 | 
			
		||||
    "logoutSuccessSubtitle": "您已登出",
 | 
			
		||||
    "logoutTitle": "登出",
 | 
			
		||||
    "logoutUsernameSubtitle": "您当前登录用户为<code>{{username}}</code>。点击下方按钮注销。",
 | 
			
		||||
    "logoutOauthSubtitle": "您当前以<code>{{username}}</code>登录,使用的是{{provider}} OAuth 提供商。点击下方按钮注销。",
 | 
			
		||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
			
		||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
			
		||||
    "notFoundTitle": "无法找到页面",
 | 
			
		||||
    "notFoundSubtitle": "您正在查找的页面不存在。",
 | 
			
		||||
    "notFoundButton": "回到主页",
 | 
			
		||||
@@ -37,18 +37,17 @@
 | 
			
		||||
    "totpSuccessTitle": "已验证",
 | 
			
		||||
    "totpSuccessSubtitle": "重定向到您的应用",
 | 
			
		||||
    "totpTitle": "输入您的 TOTP 代码",
 | 
			
		||||
    "totpSubtitle": "请输入您身份验证器应用中的代码。",
 | 
			
		||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
			
		||||
    "unauthorizedTitle": "未授权",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "用户名为<code>{{username}}</code>的用户无权访问资源<code>{{resource}}</code>。",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "用户名为<code>{{username}}</code>的用户无权登录。",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "用户名为<code>{{username}}</code>的用户不在资源<code>{{resource}}</code>所需的组中。",
 | 
			
		||||
    "unauthorizedIpSubtitle": "Your IP address <code>{{ip}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "重试",
 | 
			
		||||
    "untrustedRedirectTitle": "不可信的重定向",
 | 
			
		||||
    "untrustedRedirectSubtitle": "您正在尝试重定向到一个与您已配置的域名 (<code>{{domain}}</code>) 不匹配的域名。您确定要继续吗?",
 | 
			
		||||
    "cancelTitle": "取消",
 | 
			
		||||
    "forgotPasswordTitle": "忘记密码?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "加载身份验证提供程序失败,请检查您的配置。",
 | 
			
		||||
    "errorTitle": "发生了错误",
 | 
			
		||||
    "errorSubtitle": "执行此操作时发生错误,请检查控制台以获取更多信息。"
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
    "errorTitle": "An error occurred",
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information."
 | 
			
		||||
}
 | 
			
		||||
@@ -1,54 +1,53 @@
 | 
			
		||||
{
 | 
			
		||||
    "loginTitle": "歡迎回來,請用以下方式登入",
 | 
			
		||||
    "loginTitleSimple": "歡迎回來,請登入",
 | 
			
		||||
    "loginDivider": "或",
 | 
			
		||||
    "loginUsername": "帳號",
 | 
			
		||||
    "loginPassword": "密碼",
 | 
			
		||||
    "loginSubmit": "登入",
 | 
			
		||||
    "loginFailTitle": "登入失敗",
 | 
			
		||||
    "loginFailSubtitle": "請檢查您的帳號與密碼",
 | 
			
		||||
    "loginFailRateLimit": "登入失敗次數過多,請稍後再試",
 | 
			
		||||
    "loginSuccessTitle": "登入成功",
 | 
			
		||||
    "loginSuccessSubtitle": "歡迎回來!",
 | 
			
		||||
    "loginOauthFailTitle": "發生錯誤",
 | 
			
		||||
    "loginOauthFailSubtitle": "無法取得 OAuth 網址",
 | 
			
		||||
    "loginOauthSuccessTitle": "重新導向中",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "正在將您重新導向至 OAuth 供應商",
 | 
			
		||||
    "continueRedirectingTitle": "重新導向中...",
 | 
			
		||||
    "continueRedirectingSubtitle": "您即將被重新導向至應用程式",
 | 
			
		||||
    "continueInvalidRedirectTitle": "無效的重新導向",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "重新導向的網址無效",
 | 
			
		||||
    "continueInsecureRedirectTitle": "不安全的重新導向",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "您正嘗試從安全的 <code>https</code> 重新導向至不安全的 <code>http</code>。您確定要繼續嗎?",
 | 
			
		||||
    "continueTitle": "繼續",
 | 
			
		||||
    "continueSubtitle": "點擊按鈕以繼續前往您的應用程式。",
 | 
			
		||||
    "logoutFailTitle": "登出失敗",
 | 
			
		||||
    "logoutFailSubtitle": "請再試一次",
 | 
			
		||||
    "logoutSuccessTitle": "登出成功",
 | 
			
		||||
    "logoutSuccessSubtitle": "您已成功登出",
 | 
			
		||||
    "logoutTitle": "登出",
 | 
			
		||||
    "logoutUsernameSubtitle": "您目前以 <code>{{username}}</code> 的身分登入。點擊下方按鈕以登出。",
 | 
			
		||||
    "logoutOauthSubtitle": "您目前使用 {{provider}} OAuth 供應商並以 <code>{{username}}</code> 的身分登入。點擊下方按鈕以登出。",
 | 
			
		||||
    "notFoundTitle": "找不到頁面",
 | 
			
		||||
    "notFoundSubtitle": "您要尋找的頁面不存在。",
 | 
			
		||||
    "notFoundButton": "回到首頁",
 | 
			
		||||
    "totpFailTitle": "驗證失敗",
 | 
			
		||||
    "totpFailSubtitle": "請檢查您的驗證碼並再試一次",
 | 
			
		||||
    "totpSuccessTitle": "驗證成功",
 | 
			
		||||
    "totpSuccessSubtitle": "正在重新導向至您的應用程式",
 | 
			
		||||
    "totpTitle": "輸入您的 TOTP 驗證碼",
 | 
			
		||||
    "totpSubtitle": "請輸入您驗證器應用程式中的代碼。",
 | 
			
		||||
    "unauthorizedTitle": "未經授權",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "使用者 <code>{{username}}</code> 未被授權存取資源 <code>{{resource}}</code>。",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "使用者 <code>{{username}}</code> 未被授權登入。",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "使用者 <code>{{username}}</code> 不在存取資源 <code>{{resource}}</code> 所需的群組中。",
 | 
			
		||||
    "unauthorizedIpSubtitle": "您的 IP 位址 <code>{{ip}}</code> 未被授權存取資源 <code>{{resource}}</code>。",
 | 
			
		||||
    "unauthorizedButton": "再試一次",
 | 
			
		||||
    "untrustedRedirectTitle": "不受信任的重新導向",
 | 
			
		||||
    "untrustedRedirectSubtitle": "您正嘗試重新導向至的網域與您設定的網域 (<code>{{domain}}</code>) 不符。您確定要繼續嗎?",
 | 
			
		||||
    "cancelTitle": "取消",
 | 
			
		||||
    "forgotPasswordTitle": "忘記密碼?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "載入驗證供應商失敗。請檢查您的設定。",
 | 
			
		||||
    "errorTitle": "發生錯誤",
 | 
			
		||||
    "errorSubtitle": "執行此操作時發生錯誤。請檢查主控台以獲取更多資訊。"
 | 
			
		||||
    "loginTitle": "Welcome back, login with",
 | 
			
		||||
    "loginTitleSimple": "Welcome back, please login",
 | 
			
		||||
    "loginDivider": "Or",
 | 
			
		||||
    "loginUsername": "Username",
 | 
			
		||||
    "loginPassword": "Password",
 | 
			
		||||
    "loginSubmit": "Login",
 | 
			
		||||
    "loginFailTitle": "Failed to log in",
 | 
			
		||||
    "loginFailSubtitle": "Please check your username and password",
 | 
			
		||||
    "loginFailRateLimit": "You failed to login too many times. Please try again later",
 | 
			
		||||
    "loginSuccessTitle": "Logged in",
 | 
			
		||||
    "loginSuccessSubtitle": "Welcome back!",
 | 
			
		||||
    "loginOauthFailTitle": "An error occurred",
 | 
			
		||||
    "loginOauthFailSubtitle": "Failed to get OAuth URL",
 | 
			
		||||
    "loginOauthSuccessTitle": "Redirecting",
 | 
			
		||||
    "loginOauthSuccessSubtitle": "Redirecting to your OAuth provider",
 | 
			
		||||
    "continueRedirectingTitle": "Redirecting...",
 | 
			
		||||
    "continueRedirectingSubtitle": "You should be redirected to the app soon",
 | 
			
		||||
    "continueInvalidRedirectTitle": "Invalid redirect",
 | 
			
		||||
    "continueInvalidRedirectSubtitle": "The redirect URL is invalid",
 | 
			
		||||
    "continueInsecureRedirectTitle": "Insecure redirect",
 | 
			
		||||
    "continueInsecureRedirectSubtitle": "You are trying to redirect from <code>https</code> to <code>http</code> which is not secure. Are you sure you want to continue?",
 | 
			
		||||
    "continueTitle": "Continue",
 | 
			
		||||
    "continueSubtitle": "Click the button to continue to your app.",
 | 
			
		||||
    "logoutFailTitle": "Failed to log out",
 | 
			
		||||
    "logoutFailSubtitle": "Please try again",
 | 
			
		||||
    "logoutSuccessTitle": "Logged out",
 | 
			
		||||
    "logoutSuccessSubtitle": "You have been logged out",
 | 
			
		||||
    "logoutTitle": "Logout",
 | 
			
		||||
    "logoutUsernameSubtitle": "You are currently logged in as <code>{{username}}</code>. Click the button below to logout.",
 | 
			
		||||
    "logoutOauthSubtitle": "You are currently logged in as <code>{{username}}</code> using the {{provider}} OAuth provider. Click the button below to logout.",
 | 
			
		||||
    "notFoundTitle": "Page not found",
 | 
			
		||||
    "notFoundSubtitle": "The page you are looking for does not exist.",
 | 
			
		||||
    "notFoundButton": "Go home",
 | 
			
		||||
    "totpFailTitle": "Failed to verify code",
 | 
			
		||||
    "totpFailSubtitle": "Please check your code and try again",
 | 
			
		||||
    "totpSuccessTitle": "Verified",
 | 
			
		||||
    "totpSuccessSubtitle": "Redirecting to your app",
 | 
			
		||||
    "totpTitle": "Enter your TOTP code",
 | 
			
		||||
    "totpSubtitle": "Please enter the code from your authenticator app.",
 | 
			
		||||
    "unauthorizedTitle": "Unauthorized",
 | 
			
		||||
    "unauthorizedResourceSubtitle": "The user with username <code>{{username}}</code> is not authorized to access the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedLoginSubtitle": "The user with username <code>{{username}}</code> is not authorized to login.",
 | 
			
		||||
    "unauthorizedGroupsSubtitle": "The user with username <code>{{username}}</code> is not in the groups required by the resource <code>{{resource}}</code>.",
 | 
			
		||||
    "unauthorizedButton": "Try again",
 | 
			
		||||
    "untrustedRedirectTitle": "Untrusted redirect",
 | 
			
		||||
    "untrustedRedirectSubtitle": "You are trying to redirect to a domain that does not match your configured domain (<code>{{domain}}</code>). Are you sure you want to continue?",
 | 
			
		||||
    "cancelTitle": "Cancel",
 | 
			
		||||
    "forgotPasswordTitle": "Forgot your password?",
 | 
			
		||||
    "failedToFetchProvidersTitle": "Failed to load authentication providers. Please check your configuration.",
 | 
			
		||||
    "errorTitle": "An error occurred",
 | 
			
		||||
    "errorSubtitle": "An error occurred while trying to perform this action. Please check the console for more information."
 | 
			
		||||
}
 | 
			
		||||
@@ -162,9 +162,9 @@ export const LoginPage = () => {
 | 
			
		||||
          />
 | 
			
		||||
        )}
 | 
			
		||||
        {configuredProviders.length == 0 && (
 | 
			
		||||
          <p className="text-center text-red-600 max-w-sm">
 | 
			
		||||
          <h3 className="text-center text-xl text-red-600">
 | 
			
		||||
            {t("failedToFetchProvidersTitle")}
 | 
			
		||||
          </p>
 | 
			
		||||
          </h3>
 | 
			
		||||
        )}
 | 
			
		||||
      </CardContent>
 | 
			
		||||
    </Card>
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										11
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								go.mod
									
									
									
									
									
								
							@@ -4,29 +4,27 @@ go 1.23.2
 | 
			
		||||
 | 
			
		||||
require (
 | 
			
		||||
	github.com/gin-gonic/gin v1.10.1
 | 
			
		||||
	github.com/go-playground/validator/v10 v10.27.0
 | 
			
		||||
	github.com/go-playground/validator/v10 v10.26.0
 | 
			
		||||
	github.com/google/go-querystring v1.1.0
 | 
			
		||||
	github.com/google/uuid v1.6.0
 | 
			
		||||
	github.com/mdp/qrterminal/v3 v3.2.1
 | 
			
		||||
	github.com/rs/zerolog v1.34.0
 | 
			
		||||
	github.com/spf13/cobra v1.9.1
 | 
			
		||||
	github.com/spf13/viper v1.20.1
 | 
			
		||||
	github.com/traefik/paerser v0.2.2
 | 
			
		||||
	golang.org/x/crypto v0.39.0
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
require (
 | 
			
		||||
	github.com/Azure/go-ntlmssp v0.0.0-20221128193559-754e69321358 // indirect
 | 
			
		||||
	github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc // indirect
 | 
			
		||||
	github.com/charmbracelet/x/cellbuf v0.0.13 // indirect
 | 
			
		||||
	github.com/containerd/errdefs v1.0.0 // indirect
 | 
			
		||||
	github.com/containerd/errdefs/pkg v0.3.0 // indirect
 | 
			
		||||
	github.com/containerd/log v0.1.0 // indirect
 | 
			
		||||
	github.com/go-asn1-ber/asn1-ber v1.5.8-0.20250403174932-29230038a667 // indirect
 | 
			
		||||
	github.com/go-viper/mapstructure/v2 v2.3.0 // indirect
 | 
			
		||||
	github.com/go-viper/mapstructure/v2 v2.2.1 // indirect
 | 
			
		||||
	github.com/moby/sys/atomicwriter v0.1.0 // indirect
 | 
			
		||||
	github.com/moby/term v0.5.2 // indirect
 | 
			
		||||
	github.com/morikuni/aec v1.0.0 // indirect
 | 
			
		||||
	github.com/traefik/paerser v0.2.2 // indirect
 | 
			
		||||
	github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect
 | 
			
		||||
	go.opentelemetry.io/auto/sdk v1.1.0 // indirect
 | 
			
		||||
	go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.34.0 // indirect
 | 
			
		||||
@@ -53,7 +51,7 @@ require (
 | 
			
		||||
	github.com/charmbracelet/x/term v0.2.1 // indirect
 | 
			
		||||
	github.com/cloudwego/base64x v0.1.4 // indirect
 | 
			
		||||
	github.com/distribution/reference v0.6.0 // indirect
 | 
			
		||||
	github.com/docker/docker v28.3.1+incompatible
 | 
			
		||||
	github.com/docker/docker v28.2.2+incompatible
 | 
			
		||||
	github.com/docker/go-connections v0.5.0 // indirect
 | 
			
		||||
	github.com/docker/go-units v0.5.0 // indirect
 | 
			
		||||
	github.com/dustin/go-humanize v1.0.1 // indirect
 | 
			
		||||
@@ -62,7 +60,6 @@ require (
 | 
			
		||||
	github.com/fsnotify/fsnotify v1.8.0 // indirect
 | 
			
		||||
	github.com/gabriel-vasile/mimetype v1.4.8 // indirect
 | 
			
		||||
	github.com/gin-contrib/sse v1.0.0 // indirect
 | 
			
		||||
	github.com/go-ldap/ldap/v3 v3.4.11
 | 
			
		||||
	github.com/go-logr/logr v1.4.2 // indirect
 | 
			
		||||
	github.com/go-logr/stdr v1.2.2 // indirect
 | 
			
		||||
	github.com/go-playground/locales v0.14.1 // indirect
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										34
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								go.sum
									
									
									
									
									
								
							@@ -1,13 +1,9 @@
 | 
			
		||||
github.com/Azure/go-ansiterm v0.0.0-20250102033503-faa5f7b0171c h1:udKWzYgxTojEKWjV8V+WSxDXJ4NFATAsZjh8iIbsQIg=
 | 
			
		||||
github.com/Azure/go-ansiterm v0.0.0-20250102033503-faa5f7b0171c/go.mod h1:xomTg63KZ2rFqZQzSB4Vz2SUXa1BpHTVz9L5PTmPC4E=
 | 
			
		||||
github.com/Azure/go-ntlmssp v0.0.0-20221128193559-754e69321358 h1:mFRzDkZVAjdal+s7s0MwaRv9igoPqLRdzOLzw/8Xvq8=
 | 
			
		||||
github.com/Azure/go-ntlmssp v0.0.0-20221128193559-754e69321358/go.mod h1:chxPXzSsl7ZWRAuOIE23GDNzjWuZquvFlgA8xmpunjU=
 | 
			
		||||
github.com/MakeNowJust/heredoc v1.0.0 h1:cXCdzVdstXyiTqTvfqk9SDHpKNjxuom+DOlyEeQ4pzQ=
 | 
			
		||||
github.com/MakeNowJust/heredoc v1.0.0/go.mod h1:mG5amYoWBHf8vpLOuehzbGGw0EHxpZZ6lCpQ4fNJ8LE=
 | 
			
		||||
github.com/Microsoft/go-winio v0.4.14 h1:+hMXMk01us9KgxGb7ftKQt2Xpf5hH/yky+TDA+qxleU=
 | 
			
		||||
github.com/Microsoft/go-winio v0.4.14/go.mod h1:qXqCSQ3Xa7+6tgxaGTIe4Kpcdsi+P8jBhyzoq1bpyYA=
 | 
			
		||||
github.com/alexbrainman/sspi v0.0.0-20231016080023-1a75b4708caa h1:LHTHcTQiSGT7VVbI0o4wBRNQIgn917usHWOd6VAffYI=
 | 
			
		||||
github.com/alexbrainman/sspi v0.0.0-20231016080023-1a75b4708caa/go.mod h1:cEWa1LVoE5KvSD9ONXsZrj0z6KqySlCCNKHlLzbqAt4=
 | 
			
		||||
github.com/atotto/clipboard v0.1.4 h1:EH0zSVneZPSuFR11BlR9YppQTVDbh5+16AmcJi4g1z4=
 | 
			
		||||
github.com/atotto/clipboard v0.1.4/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI=
 | 
			
		||||
github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k=
 | 
			
		||||
@@ -72,8 +68,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
 | 
			
		||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/distribution/reference v0.6.0 h1:0IXCQ5g4/QMHHkarYzh5l+u8T3t73zM5QvfrDyIgxBk=
 | 
			
		||||
github.com/distribution/reference v0.6.0/go.mod h1:BbU0aIcezP1/5jX/8MP0YiH4SdvB5Y4f/wlDRiLyi3E=
 | 
			
		||||
github.com/docker/docker v28.3.1+incompatible h1:20+BmuA9FXlCX4ByQ0vYJcUEnOmRM6XljDnFWR+jCyY=
 | 
			
		||||
github.com/docker/docker v28.3.1+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
 | 
			
		||||
github.com/docker/docker v28.2.2+incompatible h1:CjwRSksz8Yo4+RmQ339Dp/D2tGO5JxwYeqtMOEe0LDw=
 | 
			
		||||
github.com/docker/docker v28.2.2+incompatible/go.mod h1:eEKB0N0r5NX/I1kEveEz05bcu8tLC/8azJZsviup8Sk=
 | 
			
		||||
github.com/docker/go-connections v0.5.0 h1:USnMq7hx7gwdVZq1L49hLXaFtUdTADjXGp+uj1Br63c=
 | 
			
		||||
github.com/docker/go-connections v0.5.0/go.mod h1:ov60Kzw0kKElRwhNs9UlUHAE/F9Fe6GLaXnqyDdmEXc=
 | 
			
		||||
github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4=
 | 
			
		||||
@@ -94,10 +90,6 @@ github.com/gin-contrib/sse v1.0.0 h1:y3bT1mUWUxDpW4JLQg/HnTqV4rozuW4tC9eFKTxYI9E
 | 
			
		||||
github.com/gin-contrib/sse v1.0.0/go.mod h1:zNuFdwarAygJBht0NTKiSi3jRf6RbqeILZ9Sp6Slhe0=
 | 
			
		||||
github.com/gin-gonic/gin v1.10.1 h1:T0ujvqyCSqRopADpgPgiTT63DUQVSfojyME59Ei63pQ=
 | 
			
		||||
github.com/gin-gonic/gin v1.10.1/go.mod h1:4PMNQiOhvDRa013RKVbsiNwoyezlm2rm0uX/T7kzp5Y=
 | 
			
		||||
github.com/go-asn1-ber/asn1-ber v1.5.8-0.20250403174932-29230038a667 h1:BP4M0CvQ4S3TGls2FvczZtj5Re/2ZzkV9VwqPHH/3Bo=
 | 
			
		||||
github.com/go-asn1-ber/asn1-ber v1.5.8-0.20250403174932-29230038a667/go.mod h1:hEBeB/ic+5LoWskz+yKT7vGhhPYkProFKoKdwZRWMe0=
 | 
			
		||||
github.com/go-ldap/ldap/v3 v3.4.11 h1:4k0Yxweg+a3OyBLjdYn5OKglv18JNvfDykSoI8bW0gU=
 | 
			
		||||
github.com/go-ldap/ldap/v3 v3.4.11/go.mod h1:bY7t0FLK8OAVpp/vV6sSlpz3EQDGcQwc8pF0ujLgKvM=
 | 
			
		||||
github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
 | 
			
		||||
github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY=
 | 
			
		||||
github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
 | 
			
		||||
@@ -109,10 +101,10 @@ github.com/go-playground/locales v0.14.1 h1:EWaQ/wswjilfKLTECiXz7Rh+3BjFhfDFKv/o
 | 
			
		||||
github.com/go-playground/locales v0.14.1/go.mod h1:hxrqLVvrK65+Rwrd5Fc6F2O76J/NuW9t0sjnWqG1slY=
 | 
			
		||||
github.com/go-playground/universal-translator v0.18.1 h1:Bcnm0ZwsGyWbCzImXv+pAJnYK9S473LQFuzCbDbfSFY=
 | 
			
		||||
github.com/go-playground/universal-translator v0.18.1/go.mod h1:xekY+UJKNuX9WP91TpwSH2VMlDf28Uj24BCp08ZFTUY=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.27.0 h1:w8+XrWVMhGkxOaaowyKH35gFydVHOvC0/uWoy2Fzwn4=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.27.0/go.mod h1:I5QpIEbmr8On7W0TktmJAumgzX4CA1XNl4ZmDuVHKKo=
 | 
			
		||||
github.com/go-viper/mapstructure/v2 v2.3.0 h1:27XbWsHIqhbdR5TIC911OfYvgSaW93HM+dX7970Q7jk=
 | 
			
		||||
github.com/go-viper/mapstructure/v2 v2.3.0/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.26.0 h1:SP05Nqhjcvz81uJaRfEV0YBSSSGMc/iMaVtFbr3Sw2k=
 | 
			
		||||
github.com/go-playground/validator/v10 v10.26.0/go.mod h1:I5QpIEbmr8On7W0TktmJAumgzX4CA1XNl4ZmDuVHKKo=
 | 
			
		||||
github.com/go-viper/mapstructure/v2 v2.2.1 h1:ZAaOCxANMuZx5RCeg0mBdEZk7DZasvvZIxtHqx8aGss=
 | 
			
		||||
github.com/go-viper/mapstructure/v2 v2.2.1/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM=
 | 
			
		||||
github.com/goccy/go-json v0.10.4 h1:JSwxQzIqKfmFX1swYPpUThQZp/Ka4wzJdK0LWVytLPM=
 | 
			
		||||
github.com/goccy/go-json v0.10.4/go.mod h1:oq7eo15ShAhp70Anwd5lgX2pLfOS3QCiwU/PULtXL6M=
 | 
			
		||||
github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA=
 | 
			
		||||
@@ -134,22 +126,8 @@ github.com/gorilla/sessions v1.4.0 h1:kpIYOp/oi6MG/p5PgxApU8srsSw9tuFbt46Lt7auzq
 | 
			
		||||
github.com/gorilla/sessions v1.4.0/go.mod h1:FLWm50oby91+hl7p/wRxDth9bWSuk0qVL2emc7lT5ik=
 | 
			
		||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.25.1 h1:VNqngBF40hVlDloBruUehVYC3ArSgIyScOAyMRqBxRg=
 | 
			
		||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.25.1/go.mod h1:RBRO7fro65R6tjKzYgLAFo0t1QEXY1Dp+i/bvpRiqiQ=
 | 
			
		||||
github.com/hashicorp/go-uuid v1.0.3 h1:2gKiV6YVmrJ1i2CKKa9obLvRieoRGviZFL26PcT/Co8=
 | 
			
		||||
github.com/hashicorp/go-uuid v1.0.3/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro=
 | 
			
		||||
github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8=
 | 
			
		||||
github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw=
 | 
			
		||||
github.com/jcmturner/aescts/v2 v2.0.0 h1:9YKLH6ey7H4eDBXW8khjYslgyqG2xZikXP0EQFKrle8=
 | 
			
		||||
github.com/jcmturner/aescts/v2 v2.0.0/go.mod h1:AiaICIRyfYg35RUkr8yESTqvSy7csK90qZ5xfvvsoNs=
 | 
			
		||||
github.com/jcmturner/dnsutils/v2 v2.0.0 h1:lltnkeZGL0wILNvrNiVCR6Ro5PGU/SeBvVO/8c/iPbo=
 | 
			
		||||
github.com/jcmturner/dnsutils/v2 v2.0.0/go.mod h1:b0TnjGOvI/n42bZa+hmXL+kFJZsFT7G4t3HTlQ184QM=
 | 
			
		||||
github.com/jcmturner/gofork v1.7.6 h1:QH0l3hzAU1tfT3rZCnW5zXl+orbkNMMRGJfdJjHVETg=
 | 
			
		||||
github.com/jcmturner/gofork v1.7.6/go.mod h1:1622LH6i/EZqLloHfE7IeZ0uEJwMSUyQ/nDd82IeqRo=
 | 
			
		||||
github.com/jcmturner/goidentity/v6 v6.0.1 h1:VKnZd2oEIMorCTsFBnJWbExfNN7yZr3EhJAxwOkZg6o=
 | 
			
		||||
github.com/jcmturner/goidentity/v6 v6.0.1/go.mod h1:X1YW3bgtvwAXju7V3LCIMpY0Gbxyjn/mY9zx4tFonSg=
 | 
			
		||||
github.com/jcmturner/gokrb5/v8 v8.4.4 h1:x1Sv4HaTpepFkXbt2IkL29DXRf8sOfZXo8eRKh687T8=
 | 
			
		||||
github.com/jcmturner/gokrb5/v8 v8.4.4/go.mod h1:1btQEpgT6k+unzCwX1KdWMEwPPkkgBtP+F6aCACiMrs=
 | 
			
		||||
github.com/jcmturner/rpc/v2 v2.0.3 h1:7FXXj8Ti1IaVFpSAziCZWNzbNuZmnvw/i6CqLNdWfZY=
 | 
			
		||||
github.com/jcmturner/rpc/v2 v2.0.3/go.mod h1:VUJYCIDm3PVOEHw8sgt091/20OJjskO/YJki3ELg/Hc=
 | 
			
		||||
github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM=
 | 
			
		||||
github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo=
 | 
			
		||||
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
package server
 | 
			
		||||
package api
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"fmt"
 | 
			
		||||
@@ -15,13 +15,20 @@ import (
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type Server struct {
 | 
			
		||||
	Config   types.ServerConfig
 | 
			
		||||
	Handlers *handlers.Handlers
 | 
			
		||||
	Router   *gin.Engine
 | 
			
		||||
func NewAPI(config types.APIConfig, handlers *handlers.Handlers) *API {
 | 
			
		||||
	return &API{
 | 
			
		||||
		Config:   config,
 | 
			
		||||
		Handlers: handlers,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewServer(config types.ServerConfig, handlers *handlers.Handlers) (*Server, error) {
 | 
			
		||||
type API struct {
 | 
			
		||||
	Config   types.APIConfig
 | 
			
		||||
	Router   *gin.Engine
 | 
			
		||||
	Handlers *handlers.Handlers
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (api *API) Init() {
 | 
			
		||||
	// Disable gin logs
 | 
			
		||||
	gin.SetMode(gin.ReleaseMode)
 | 
			
		||||
 | 
			
		||||
@@ -35,7 +42,7 @@ func NewServer(config types.ServerConfig, handlers *handlers.Handlers) (*Server,
 | 
			
		||||
	dist, err := fs.Sub(assets.Assets, "dist")
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to get UI assets")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create file server
 | 
			
		||||
@@ -62,38 +69,41 @@ func NewServer(config types.ServerConfig, handlers *handlers.Handlers) (*Server,
 | 
			
		||||
		}
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Proxy routes
 | 
			
		||||
	router.GET("/api/auth/:proxy", handlers.AuthHandler)
 | 
			
		||||
 | 
			
		||||
	// Auth routes
 | 
			
		||||
	router.POST("/api/login", handlers.LoginHandler)
 | 
			
		||||
	router.POST("/api/totp", handlers.TotpHandler)
 | 
			
		||||
	router.POST("/api/logout", handlers.LogoutHandler)
 | 
			
		||||
 | 
			
		||||
	// Context routes
 | 
			
		||||
	router.GET("/api/app", handlers.AppHandler)
 | 
			
		||||
	router.GET("/api/user", handlers.UserHandler)
 | 
			
		||||
 | 
			
		||||
	// OAuth routes
 | 
			
		||||
	router.GET("/api/oauth/url/:provider", handlers.OauthUrlHandler)
 | 
			
		||||
	router.GET("/api/oauth/callback/:provider", handlers.OauthCallbackHandler)
 | 
			
		||||
 | 
			
		||||
	// App routes
 | 
			
		||||
	router.GET("/api/healthcheck", handlers.HealthcheckHandler)
 | 
			
		||||
 | 
			
		||||
	// Return the server
 | 
			
		||||
	return &Server{
 | 
			
		||||
		Config:   config,
 | 
			
		||||
		Handlers: handlers,
 | 
			
		||||
		Router:   router,
 | 
			
		||||
	}, nil
 | 
			
		||||
	// Set router
 | 
			
		||||
	api.Router = router
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (s *Server) Start() error {
 | 
			
		||||
	// Run server
 | 
			
		||||
	log.Info().Str("address", s.Config.Address).Int("port", s.Config.Port).Msg("Starting server")
 | 
			
		||||
func (api *API) SetupRoutes() {
 | 
			
		||||
	// Proxy
 | 
			
		||||
	api.Router.GET("/api/auth/:proxy", api.Handlers.AuthHandler)
 | 
			
		||||
 | 
			
		||||
	return s.Router.Run(fmt.Sprintf("%s:%d", s.Config.Address, s.Config.Port))
 | 
			
		||||
	// Auth
 | 
			
		||||
	api.Router.POST("/api/login", api.Handlers.LoginHandler)
 | 
			
		||||
	api.Router.POST("/api/totp", api.Handlers.TotpHandler)
 | 
			
		||||
	api.Router.POST("/api/logout", api.Handlers.LogoutHandler)
 | 
			
		||||
 | 
			
		||||
	// Context
 | 
			
		||||
	api.Router.GET("/api/app", api.Handlers.AppHandler)
 | 
			
		||||
	api.Router.GET("/api/user", api.Handlers.UserHandler)
 | 
			
		||||
 | 
			
		||||
	// OAuth
 | 
			
		||||
	api.Router.GET("/api/oauth/url/:provider", api.Handlers.OauthUrlHandler)
 | 
			
		||||
	api.Router.GET("/api/oauth/callback/:provider", api.Handlers.OauthCallbackHandler)
 | 
			
		||||
 | 
			
		||||
	// App
 | 
			
		||||
	api.Router.GET("/api/healthcheck", api.Handlers.HealthcheckHandler)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (api *API) Run() {
 | 
			
		||||
	log.Info().Str("address", api.Config.Address).Int("port", api.Config.Port).Msg("Starting server")
 | 
			
		||||
 | 
			
		||||
	// Run server
 | 
			
		||||
	err := api.Router.Run(fmt.Sprintf("%s:%d", api.Config.Address, api.Config.Port))
 | 
			
		||||
 | 
			
		||||
	// Check for errors
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to start server")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// zerolog is a middleware for gin that logs requests using zerolog
 | 
			
		||||
							
								
								
									
										322
									
								
								internal/api/api_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										322
									
								
								internal/api/api_test.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,322 @@
 | 
			
		||||
package api_test
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"encoding/json"
 | 
			
		||||
	"io"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"net/http/httptest"
 | 
			
		||||
	"reflect"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"testing"
 | 
			
		||||
	"tinyauth/internal/api"
 | 
			
		||||
	"tinyauth/internal/auth"
 | 
			
		||||
	"tinyauth/internal/docker"
 | 
			
		||||
	"tinyauth/internal/handlers"
 | 
			
		||||
	"tinyauth/internal/hooks"
 | 
			
		||||
	"tinyauth/internal/providers"
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
 | 
			
		||||
	"github.com/magiconair/properties/assert"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Simple API config for tests
 | 
			
		||||
var apiConfig = types.APIConfig{
 | 
			
		||||
	Port:    8080,
 | 
			
		||||
	Address: "0.0.0.0",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Simple handlers config for tests
 | 
			
		||||
var handlersConfig = types.HandlersConfig{
 | 
			
		||||
	AppURL:                "http://localhost:8080",
 | 
			
		||||
	Domain:                "localhost",
 | 
			
		||||
	DisableContinue:       false,
 | 
			
		||||
	CookieSecure:          false,
 | 
			
		||||
	Title:                 "Tinyauth",
 | 
			
		||||
	GenericName:           "Generic",
 | 
			
		||||
	ForgotPasswordMessage: "Some message",
 | 
			
		||||
	CsrfCookieName:        "tinyauth-csrf",
 | 
			
		||||
	RedirectCookieName:    "tinyauth-redirect",
 | 
			
		||||
	BackgroundImage:       "https://example.com/image.png",
 | 
			
		||||
	OAuthAutoRedirect:     "none",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Simple auth config for tests
 | 
			
		||||
var authConfig = types.AuthConfig{
 | 
			
		||||
	Users:             types.Users{},
 | 
			
		||||
	OauthWhitelist:    "",
 | 
			
		||||
	Secret:            "super-secret-api-thing-for-tests", // It is 32 chars long
 | 
			
		||||
	CookieSecure:      false,
 | 
			
		||||
	SessionExpiry:     3600,
 | 
			
		||||
	LoginTimeout:      0,
 | 
			
		||||
	LoginMaxRetries:   0,
 | 
			
		||||
	SessionCookieName: "tinyauth-session",
 | 
			
		||||
	Domain:            "localhost",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Simple hooks config for tests
 | 
			
		||||
var hooksConfig = types.HooksConfig{
 | 
			
		||||
	Domain: "localhost",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Cookie
 | 
			
		||||
var cookie string
 | 
			
		||||
 | 
			
		||||
// User
 | 
			
		||||
var user = types.User{
 | 
			
		||||
	Username: "user",
 | 
			
		||||
	Password: "$2a$10$AvGHLTYv3xiRJ0xV9xs3XeVIlkGTygI9nqIamFYB5Xu.5.0UWF7B6", // pass
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// We need all this to be able to test the API
 | 
			
		||||
func getAPI(t *testing.T) *api.API {
 | 
			
		||||
	// Create docker service
 | 
			
		||||
	docker := docker.NewDocker()
 | 
			
		||||
 | 
			
		||||
	// Initialize docker
 | 
			
		||||
	err := docker.Init()
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Failed to initialize docker: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create auth service
 | 
			
		||||
	authConfig.Users = types.Users{
 | 
			
		||||
		{
 | 
			
		||||
			Username: user.Username,
 | 
			
		||||
			Password: user.Password,
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
	auth := auth.NewAuth(authConfig, docker)
 | 
			
		||||
 | 
			
		||||
	// Create providers service
 | 
			
		||||
	providers := providers.NewProviders(types.OAuthConfig{})
 | 
			
		||||
 | 
			
		||||
	// Initialize providers
 | 
			
		||||
	providers.Init()
 | 
			
		||||
 | 
			
		||||
	// Create hooks service
 | 
			
		||||
	hooks := hooks.NewHooks(hooksConfig, auth, providers)
 | 
			
		||||
 | 
			
		||||
	// Create handlers service
 | 
			
		||||
	handlers := handlers.NewHandlers(handlersConfig, auth, hooks, providers, docker)
 | 
			
		||||
 | 
			
		||||
	// Create API
 | 
			
		||||
	api := api.NewAPI(apiConfig, handlers)
 | 
			
		||||
 | 
			
		||||
	// Setup routes
 | 
			
		||||
	api.Init()
 | 
			
		||||
	api.SetupRoutes()
 | 
			
		||||
 | 
			
		||||
	return api
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test login (we will need this for the other tests)
 | 
			
		||||
func TestLogin(t *testing.T) {
 | 
			
		||||
	t.Log("Testing login")
 | 
			
		||||
 | 
			
		||||
	// Get API
 | 
			
		||||
	api := getAPI(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	user := types.LoginRequest{
 | 
			
		||||
		Username: "user",
 | 
			
		||||
		Password: "pass",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	json, err := json.Marshal(user)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error marshalling json: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("POST", "/api/login", strings.NewReader(string(json)))
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	api.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Get the cookie
 | 
			
		||||
	cookie = recorder.Result().Cookies()[0].Value
 | 
			
		||||
 | 
			
		||||
	// Check if the cookie is set
 | 
			
		||||
	if cookie == "" {
 | 
			
		||||
		t.Fatalf("Cookie not set")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test app context
 | 
			
		||||
func TestAppContext(t *testing.T) {
 | 
			
		||||
	t.Log("Testing app context")
 | 
			
		||||
 | 
			
		||||
	// Get API
 | 
			
		||||
	api := getAPI(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("GET", "/api/app", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	api.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Read the body of the response
 | 
			
		||||
	body, err := io.ReadAll(recorder.Body)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error getting body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Unmarshal the body into the user struct
 | 
			
		||||
	var app types.AppContext
 | 
			
		||||
 | 
			
		||||
	err = json.Unmarshal(body, &app)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error unmarshalling body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create tests values
 | 
			
		||||
	expected := types.AppContext{
 | 
			
		||||
		Status:                200,
 | 
			
		||||
		Message:               "OK",
 | 
			
		||||
		ConfiguredProviders:   []string{"username"},
 | 
			
		||||
		DisableContinue:       false,
 | 
			
		||||
		Title:                 "Tinyauth",
 | 
			
		||||
		GenericName:           "Generic",
 | 
			
		||||
		ForgotPasswordMessage: "Some message",
 | 
			
		||||
		BackgroundImage:       "https://example.com/image.png",
 | 
			
		||||
		OAuthAutoRedirect:     "none",
 | 
			
		||||
		Domain:                "localhost",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// We should get the username back
 | 
			
		||||
	if !reflect.DeepEqual(app, expected) {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, app)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test user context
 | 
			
		||||
func TestUserContext(t *testing.T) {
 | 
			
		||||
	t.Log("Testing user context")
 | 
			
		||||
 | 
			
		||||
	// Get API
 | 
			
		||||
	api := getAPI(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("GET", "/api/user", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth-session",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	api.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Read the body of the response
 | 
			
		||||
	body, err := io.ReadAll(recorder.Body)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error getting body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Unmarshal the body into the user struct
 | 
			
		||||
	type User struct {
 | 
			
		||||
		Username string `json:"username"`
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var user User
 | 
			
		||||
 | 
			
		||||
	err = json.Unmarshal(body, &user)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error unmarshalling body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// We should get the username back
 | 
			
		||||
	if user.Username != "user" {
 | 
			
		||||
		t.Fatalf("Expected user, got %s", user.Username)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test logout
 | 
			
		||||
func TestLogout(t *testing.T) {
 | 
			
		||||
	t.Log("Testing logout")
 | 
			
		||||
 | 
			
		||||
	// Get API
 | 
			
		||||
	api := getAPI(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("POST", "/api/logout", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	api.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Check if the cookie is different (means go sessions flushed it)
 | 
			
		||||
	if recorder.Result().Cookies()[0].Value == cookie {
 | 
			
		||||
		t.Fatalf("Cookie not flushed")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// TODO: Testing for the oauth stuff
 | 
			
		||||
@@ -7,7 +7,6 @@ import (
 | 
			
		||||
	"sync"
 | 
			
		||||
	"time"
 | 
			
		||||
	"tinyauth/internal/docker"
 | 
			
		||||
	"tinyauth/internal/ldap"
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
@@ -17,151 +16,52 @@ import (
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func NewAuth(config types.AuthConfig, docker *docker.Docker) *Auth {
 | 
			
		||||
	return &Auth{
 | 
			
		||||
		Config:        config,
 | 
			
		||||
		Docker:        docker,
 | 
			
		||||
		LoginAttempts: make(map[string]*types.LoginAttempt),
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Auth struct {
 | 
			
		||||
	Config        types.AuthConfig
 | 
			
		||||
	Docker        *docker.Docker
 | 
			
		||||
	LoginAttempts map[string]*types.LoginAttempt
 | 
			
		||||
	LoginMutex    sync.RWMutex
 | 
			
		||||
	Store         *sessions.CookieStore
 | 
			
		||||
	LDAP          *ldap.LDAP
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewAuth(config types.AuthConfig, docker *docker.Docker, ldap *ldap.LDAP) *Auth {
 | 
			
		||||
func (auth *Auth) GetSession(c *gin.Context) (*sessions.Session, error) {
 | 
			
		||||
	// Create cookie store
 | 
			
		||||
	store := sessions.NewCookieStore([]byte(config.HMACSecret), []byte(config.EncryptionSecret))
 | 
			
		||||
	store := sessions.NewCookieStore([]byte(auth.Config.Secret))
 | 
			
		||||
 | 
			
		||||
	// Configure cookie store
 | 
			
		||||
	store.Options = &sessions.Options{
 | 
			
		||||
		Path:     "/",
 | 
			
		||||
		MaxAge:   config.SessionExpiry,
 | 
			
		||||
		Secure:   config.CookieSecure,
 | 
			
		||||
		MaxAge:   auth.Config.SessionExpiry,
 | 
			
		||||
		Secure:   auth.Config.CookieSecure,
 | 
			
		||||
		HttpOnly: true,
 | 
			
		||||
		Domain:   fmt.Sprintf(".%s", config.Domain),
 | 
			
		||||
		Domain:   fmt.Sprintf(".%s", auth.Config.Domain),
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return &Auth{
 | 
			
		||||
		Config:        config,
 | 
			
		||||
		Docker:        docker,
 | 
			
		||||
		LoginAttempts: make(map[string]*types.LoginAttempt),
 | 
			
		||||
		Store:         store,
 | 
			
		||||
		LDAP:          ldap,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) GetSession(c *gin.Context) (*sessions.Session, error) {
 | 
			
		||||
	// Get session
 | 
			
		||||
	session, err := auth.Store.Get(c.Request, auth.Config.SessionCookieName)
 | 
			
		||||
 | 
			
		||||
	session, err := store.Get(c.Request, auth.Config.SessionCookieName)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Warn().Err(err).Msg("Invalid session, clearing cookie and retrying")
 | 
			
		||||
 | 
			
		||||
		// Delete the session cookie if there is an error
 | 
			
		||||
		c.SetCookie(auth.Config.SessionCookieName, "", -1, "/", fmt.Sprintf(".%s", auth.Config.Domain), auth.Config.CookieSecure, true)
 | 
			
		||||
 | 
			
		||||
		// Try to get the session again
 | 
			
		||||
		session, err = auth.Store.Get(c.Request, auth.Config.SessionCookieName)
 | 
			
		||||
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			// If we still can't get the session, log the error and return nil
 | 
			
		||||
			log.Error().Err(err).Msg("Failed to get session")
 | 
			
		||||
			return nil, err
 | 
			
		||||
		}
 | 
			
		||||
		log.Error().Err(err).Msg("Failed to get session")
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return session, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) SearchUser(username string) types.UserSearch {
 | 
			
		||||
func (auth *Auth) GetUser(username string) *types.User {
 | 
			
		||||
	// Loop through users and return the user if the username matches
 | 
			
		||||
	log.Debug().Str("username", username).Msg("Searching for user")
 | 
			
		||||
 | 
			
		||||
	if auth.GetLocalUser(username).Username != "" {
 | 
			
		||||
		log.Debug().Str("username", username).Msg("Found local user")
 | 
			
		||||
 | 
			
		||||
		// If user found, return a user with the username and type "local"
 | 
			
		||||
		return types.UserSearch{
 | 
			
		||||
			Username: username,
 | 
			
		||||
			Type:     "local",
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If no user found, check LDAP
 | 
			
		||||
	if auth.LDAP != nil {
 | 
			
		||||
		log.Debug().Str("username", username).Msg("Checking LDAP for user")
 | 
			
		||||
 | 
			
		||||
		userDN, err := auth.LDAP.Search(username)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Warn().Err(err).Str("username", username).Msg("Failed to find user in LDAP")
 | 
			
		||||
			return types.UserSearch{}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// If user found in LDAP, return a user with the DN as username
 | 
			
		||||
		return types.UserSearch{
 | 
			
		||||
			Username: userDN,
 | 
			
		||||
			Type:     "ldap",
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return types.UserSearch{}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) VerifyUser(search types.UserSearch, password string) bool {
 | 
			
		||||
	// Authenticate the user based on the type
 | 
			
		||||
	switch search.Type {
 | 
			
		||||
	case "local":
 | 
			
		||||
		// Get local user
 | 
			
		||||
		user := auth.GetLocalUser(search.Username)
 | 
			
		||||
 | 
			
		||||
		// Check if password is correct
 | 
			
		||||
		return auth.CheckPassword(user, password)
 | 
			
		||||
	case "ldap":
 | 
			
		||||
		// If LDAP is configured, bind to the LDAP server with the user DN and password
 | 
			
		||||
		if auth.LDAP != nil {
 | 
			
		||||
			log.Debug().Str("username", search.Username).Msg("Binding to LDAP for user authentication")
 | 
			
		||||
 | 
			
		||||
			// Bind to the LDAP server
 | 
			
		||||
			err := auth.LDAP.Bind(search.Username, password)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				log.Warn().Err(err).Str("username", search.Username).Msg("Failed to bind to LDAP")
 | 
			
		||||
				return false
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			// If bind is successful, rebind with the LDAP bind user
 | 
			
		||||
			err = auth.LDAP.Bind(auth.LDAP.Config.BindDN, auth.LDAP.Config.BindPassword)
 | 
			
		||||
			if err != nil {
 | 
			
		||||
				log.Error().Err(err).Msg("Failed to rebind with service account after user authentication")
 | 
			
		||||
				// Consider closing the connection or creating a new one
 | 
			
		||||
				return false
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			log.Debug().Str("username", search.Username).Msg("LDAP authentication successful")
 | 
			
		||||
 | 
			
		||||
			// Return true if the bind was successful
 | 
			
		||||
			return true
 | 
			
		||||
		}
 | 
			
		||||
	default:
 | 
			
		||||
		log.Warn().Str("type", search.Type).Msg("Unknown user type for authentication")
 | 
			
		||||
		return false
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If no user found or authentication failed, return false
 | 
			
		||||
	log.Warn().Str("username", search.Username).Msg("User authentication failed")
 | 
			
		||||
	return false
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) GetLocalUser(username string) types.User {
 | 
			
		||||
	// Loop through users and return the user if the username matches
 | 
			
		||||
	log.Debug().Str("username", username).Msg("Searching for local user")
 | 
			
		||||
 | 
			
		||||
	for _, user := range auth.Config.Users {
 | 
			
		||||
		if user.Username == username {
 | 
			
		||||
			return user
 | 
			
		||||
			return &user
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If no user found, return an empty user
 | 
			
		||||
	log.Warn().Str("username", username).Msg("Local user not found")
 | 
			
		||||
	return types.User{}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) CheckPassword(user types.User, password string) bool {
 | 
			
		||||
@@ -233,8 +133,8 @@ func (auth *Auth) RecordLoginAttempt(identifier string, success bool) {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) EmailWhitelisted(email string) bool {
 | 
			
		||||
	return utils.CheckFilter(auth.Config.OauthWhitelist, email)
 | 
			
		||||
func (auth *Auth) EmailWhitelisted(emailSrc string) bool {
 | 
			
		||||
	return utils.CheckWhitelist(auth.Config.OauthWhitelist, emailSrc)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) CreateSessionCookie(c *gin.Context, data *types.SessionCookie) error {
 | 
			
		||||
@@ -361,20 +261,20 @@ func (auth *Auth) GetSessionCookie(c *gin.Context) (types.SessionCookie, error)
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) UserAuthConfigured() bool {
 | 
			
		||||
	// If there are users, return true
 | 
			
		||||
	return len(auth.Config.Users) > 0 || auth.LDAP != nil
 | 
			
		||||
	return len(auth.Config.Users) > 0
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) ResourceAllowed(c *gin.Context, context types.UserContext, labels types.Labels) bool {
 | 
			
		||||
	// Check if oauth is allowed
 | 
			
		||||
	if context.OAuth {
 | 
			
		||||
		log.Debug().Msg("Checking OAuth whitelist")
 | 
			
		||||
		return utils.CheckFilter(labels.OAuth.Whitelist, context.Email)
 | 
			
		||||
		return utils.CheckWhitelist(labels.OAuth.Whitelist, context.Email)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check users
 | 
			
		||||
	log.Debug().Msg("Checking users")
 | 
			
		||||
 | 
			
		||||
	return utils.CheckFilter(labels.Users, context.Username)
 | 
			
		||||
	return utils.CheckWhitelist(labels.Users, context.Username)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) OAuthGroup(c *gin.Context, context types.UserContext, labels types.Labels) bool {
 | 
			
		||||
@@ -394,7 +294,7 @@ func (auth *Auth) OAuthGroup(c *gin.Context, context types.UserContext, labels t
 | 
			
		||||
 | 
			
		||||
	// For every group check if it is in the required groups
 | 
			
		||||
	for _, group := range oauthGroups {
 | 
			
		||||
		if utils.CheckFilter(labels.OAuth.Groups, group) {
 | 
			
		||||
		if utils.CheckWhitelist(labels.OAuth.Groups, group) {
 | 
			
		||||
			log.Debug().Str("group", group).Msg("Group is in required groups")
 | 
			
		||||
			return true
 | 
			
		||||
		}
 | 
			
		||||
@@ -452,7 +352,10 @@ func (auth *Auth) GetBasicAuth(c *gin.Context) *types.User {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) CheckIP(labels types.Labels, ip string) bool {
 | 
			
		||||
func (auth *Auth) CheckIP(c *gin.Context, labels types.Labels) bool {
 | 
			
		||||
	// Get the IP address from the request
 | 
			
		||||
	ip := c.ClientIP()
 | 
			
		||||
 | 
			
		||||
	// Check if the IP is in block list
 | 
			
		||||
	for _, blocked := range labels.IP.Block {
 | 
			
		||||
		res, err := utils.FilterIP(blocked, ip)
 | 
			
		||||
@@ -489,22 +392,3 @@ func (auth *Auth) CheckIP(labels types.Labels, ip string) bool {
 | 
			
		||||
 | 
			
		||||
	return true
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) BypassedIP(labels types.Labels, ip string) bool {
 | 
			
		||||
	// For every IP in the bypass list, check if the IP matches
 | 
			
		||||
	for _, bypassed := range labels.IP.Bypass {
 | 
			
		||||
		res, err := utils.FilterIP(bypassed, ip)
 | 
			
		||||
		if err != nil {
 | 
			
		||||
			log.Warn().Err(err).Str("item", bypassed).Msg("Invalid IP/CIDR in bypass list")
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
		if res {
 | 
			
		||||
			log.Debug().Str("ip", ip).Str("item", bypassed).Msg("IP is in bypass list, allowing access")
 | 
			
		||||
			return true
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	log.Debug().Str("ip", ip).Msg("IP not in bypass list, continuing with authentication")
 | 
			
		||||
 | 
			
		||||
	return false
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -18,7 +18,7 @@ func TestLoginRateLimiting(t *testing.T) {
 | 
			
		||||
	// Initialize a new auth service with 3 max retries and 5 seconds timeout
 | 
			
		||||
	config.LoginMaxRetries = 3
 | 
			
		||||
	config.LoginTimeout = 5
 | 
			
		||||
	authService := auth.NewAuth(config, &docker.Docker{}, nil)
 | 
			
		||||
	authService := auth.NewAuth(config, &docker.Docker{})
 | 
			
		||||
 | 
			
		||||
	// Test identifier
 | 
			
		||||
	identifier := "test_user"
 | 
			
		||||
@@ -62,7 +62,7 @@ func TestLoginRateLimiting(t *testing.T) {
 | 
			
		||||
	// Reinitialize auth service with a shorter timeout for testing
 | 
			
		||||
	config.LoginTimeout = 1
 | 
			
		||||
	config.LoginMaxRetries = 3
 | 
			
		||||
	authService = auth.NewAuth(config, &docker.Docker{}, nil)
 | 
			
		||||
	authService = auth.NewAuth(config, &docker.Docker{})
 | 
			
		||||
 | 
			
		||||
	// Add enough failed attempts to lock the account
 | 
			
		||||
	for i := 0; i < 3; i++ {
 | 
			
		||||
@@ -87,7 +87,7 @@ func TestLoginRateLimiting(t *testing.T) {
 | 
			
		||||
	t.Log("Testing disabled rate limiting")
 | 
			
		||||
	config.LoginMaxRetries = 0
 | 
			
		||||
	config.LoginTimeout = 0
 | 
			
		||||
	authService = auth.NewAuth(config, &docker.Docker{}, nil)
 | 
			
		||||
	authService = auth.NewAuth(config, &docker.Docker{})
 | 
			
		||||
 | 
			
		||||
	for i := 0; i < 10; i++ {
 | 
			
		||||
		authService.RecordLoginAttempt(identifier, false)
 | 
			
		||||
@@ -103,7 +103,7 @@ func TestConcurrentLoginAttempts(t *testing.T) {
 | 
			
		||||
	// Initialize a new auth service with 2 max retries and 5 seconds timeout
 | 
			
		||||
	config.LoginMaxRetries = 2
 | 
			
		||||
	config.LoginTimeout = 5
 | 
			
		||||
	authService := auth.NewAuth(config, &docker.Docker{}, nil)
 | 
			
		||||
	authService := auth.NewAuth(config, &docker.Docker{})
 | 
			
		||||
 | 
			
		||||
	// Test multiple identifiers
 | 
			
		||||
	identifiers := []string{"user1", "user2", "user3"}
 | 
			
		||||
 
 | 
			
		||||
@@ -11,30 +11,35 @@ import (
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func NewDocker() *Docker {
 | 
			
		||||
	return &Docker{}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Docker struct {
 | 
			
		||||
	Client  *client.Client
 | 
			
		||||
	Context context.Context
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewDocker() (*Docker, error) {
 | 
			
		||||
func (docker *Docker) Init() error {
 | 
			
		||||
	// Create a new docker client
 | 
			
		||||
	client, err := client.NewClientWithOpts(client.FromEnv)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create the context
 | 
			
		||||
	ctx := context.Background()
 | 
			
		||||
	docker.Context = context.Background()
 | 
			
		||||
 | 
			
		||||
	// Negotiate API version
 | 
			
		||||
	client.NegotiateAPIVersion(ctx)
 | 
			
		||||
	client.NegotiateAPIVersion(docker.Context)
 | 
			
		||||
 | 
			
		||||
	return &Docker{
 | 
			
		||||
		Client:  client,
 | 
			
		||||
		Context: ctx,
 | 
			
		||||
	}, nil
 | 
			
		||||
	// Set client
 | 
			
		||||
	docker.Client = client
 | 
			
		||||
 | 
			
		||||
	// Done
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (docker *Docker) GetContainers() ([]container.Summary, error) {
 | 
			
		||||
@@ -69,7 +74,7 @@ func (docker *Docker) DockerConnected() bool {
 | 
			
		||||
	return err == nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (docker *Docker) GetLabels(app string, domain string) (types.Labels, error) {
 | 
			
		||||
func (docker *Docker) GetLabels(id string, domain string) (types.Labels, error) {
 | 
			
		||||
	// Check if we have access to the Docker API
 | 
			
		||||
	isConnected := docker.DockerConnected()
 | 
			
		||||
 | 
			
		||||
@@ -112,16 +117,9 @@ func (docker *Docker) GetLabels(app string, domain string) (types.Labels, error)
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Check if the container matches the ID or domain
 | 
			
		||||
		for _, lDomain := range labels.Domain {
 | 
			
		||||
			if lDomain == domain {
 | 
			
		||||
				log.Debug().Str("id", inspect.ID).Msg("Found matching container by domain")
 | 
			
		||||
				return labels, nil
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if strings.TrimPrefix(inspect.Name, "/") == app {
 | 
			
		||||
			log.Debug().Str("id", inspect.ID).Msg("Found matching container by name")
 | 
			
		||||
		// Check if the labels match the id or the domain
 | 
			
		||||
		if strings.TrimPrefix(inspect.Name, "/") == id || labels.Domain == domain {
 | 
			
		||||
			log.Debug().Str("id", inspect.ID).Msg("Found matching container")
 | 
			
		||||
			return labels, nil
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 
 | 
			
		||||
@@ -18,14 +18,6 @@ import (
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type Handlers struct {
 | 
			
		||||
	Config    types.HandlersConfig
 | 
			
		||||
	Auth      *auth.Auth
 | 
			
		||||
	Hooks     *hooks.Hooks
 | 
			
		||||
	Providers *providers.Providers
 | 
			
		||||
	Docker    *docker.Docker
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewHandlers(config types.HandlersConfig, auth *auth.Auth, hooks *hooks.Hooks, providers *providers.Providers, docker *docker.Docker) *Handlers {
 | 
			
		||||
	return &Handlers{
 | 
			
		||||
		Config:    config,
 | 
			
		||||
@@ -36,6 +28,14 @@ func NewHandlers(config types.HandlersConfig, auth *auth.Auth, hooks *hooks.Hook
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Handlers struct {
 | 
			
		||||
	Config    types.HandlersConfig
 | 
			
		||||
	Auth      *auth.Auth
 | 
			
		||||
	Hooks     *hooks.Hooks
 | 
			
		||||
	Providers *providers.Providers
 | 
			
		||||
	Docker    *docker.Docker
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (h *Handlers) AuthHandler(c *gin.Context) {
 | 
			
		||||
	// Create struct for proxy
 | 
			
		||||
	var proxy types.Proxy
 | 
			
		||||
@@ -96,29 +96,11 @@ func (h *Handlers) AuthHandler(c *gin.Context) {
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Get client IP
 | 
			
		||||
	ip := c.ClientIP()
 | 
			
		||||
 | 
			
		||||
	// Check if the IP is in bypass list
 | 
			
		||||
	if h.Auth.BypassedIP(labels, ip) {
 | 
			
		||||
		headersParsed := utils.ParseHeaders(labels.Headers)
 | 
			
		||||
		for key, value := range headersParsed {
 | 
			
		||||
			log.Debug().Str("key", key).Msg("Setting header")
 | 
			
		||||
			c.Header(key, value)
 | 
			
		||||
		}
 | 
			
		||||
		if labels.Basic.Username != "" && utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File) != "" {
 | 
			
		||||
			log.Debug().Str("username", labels.Basic.Username).Msg("Setting basic auth headers")
 | 
			
		||||
			c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Basic.Username, utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File))))
 | 
			
		||||
		}
 | 
			
		||||
		c.JSON(200, gin.H{
 | 
			
		||||
			"status":  200,
 | 
			
		||||
			"message": "Authenticated",
 | 
			
		||||
		})
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if the IP is allowed/blocked
 | 
			
		||||
	if !h.Auth.CheckIP(labels, ip) {
 | 
			
		||||
	ip := c.ClientIP()
 | 
			
		||||
	if !h.Auth.CheckIP(c, labels) {
 | 
			
		||||
		log.Warn().Str("ip", ip).Msg("IP not allowed")
 | 
			
		||||
 | 
			
		||||
		if proxy.Proxy == "nginx" || !isBrowser {
 | 
			
		||||
			c.JSON(403, gin.H{
 | 
			
		||||
				"status":  403,
 | 
			
		||||
@@ -172,9 +154,9 @@ func (h *Handlers) AuthHandler(c *gin.Context) {
 | 
			
		||||
			log.Debug().Str("key", key).Msg("Setting header")
 | 
			
		||||
			c.Header(key, value)
 | 
			
		||||
		}
 | 
			
		||||
		if labels.Basic.Username != "" && utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File) != "" {
 | 
			
		||||
			log.Debug().Str("username", labels.Basic.Username).Msg("Setting basic auth headers")
 | 
			
		||||
			c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Basic.Username, utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File))))
 | 
			
		||||
		if labels.Basic.User != "" && labels.Basic.Password != "" {
 | 
			
		||||
			log.Debug().Str("username", labels.Basic.User).Msg("Setting basic auth headers")
 | 
			
		||||
			c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Basic.User, labels.Basic.Password)))
 | 
			
		||||
		}
 | 
			
		||||
		c.JSON(200, gin.H{
 | 
			
		||||
			"status":  200,
 | 
			
		||||
@@ -301,9 +283,9 @@ func (h *Handlers) AuthHandler(c *gin.Context) {
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Set basic auth headers if configured
 | 
			
		||||
		if labels.Basic.Username != "" && utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File) != "" {
 | 
			
		||||
			log.Debug().Str("username", labels.Basic.Username).Msg("Setting basic auth headers")
 | 
			
		||||
			c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Basic.Username, utils.GetSecret(labels.Basic.Password.Plain, labels.Basic.Password.File))))
 | 
			
		||||
		if labels.Basic.User != "" && labels.Basic.Password != "" {
 | 
			
		||||
			log.Debug().Str("username", labels.Basic.User).Msg("Setting basic auth headers")
 | 
			
		||||
			c.Header("Authorization", fmt.Sprintf("Basic %s", utils.GetBasicAuth(labels.Basic.User, labels.Basic.Password)))
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// The user is allowed to access the app
 | 
			
		||||
@@ -380,13 +362,11 @@ func (h *Handlers) LoginHandler(c *gin.Context) {
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Search for a user based on username
 | 
			
		||||
	userSearch := h.Auth.SearchUser(login.Username)
 | 
			
		||||
 | 
			
		||||
	log.Debug().Interface("userSearch", userSearch).Msg("Searching for user")
 | 
			
		||||
	// Get user based on username
 | 
			
		||||
	user := h.Auth.GetUser(login.Username)
 | 
			
		||||
 | 
			
		||||
	// User does not exist
 | 
			
		||||
	if userSearch.Type == "" {
 | 
			
		||||
	if user == nil {
 | 
			
		||||
		log.Debug().Str("username", login.Username).Msg("User not found")
 | 
			
		||||
		// Record failed login attempt
 | 
			
		||||
		h.Auth.RecordLoginAttempt(rateIdentifier, false)
 | 
			
		||||
@@ -400,7 +380,7 @@ func (h *Handlers) LoginHandler(c *gin.Context) {
 | 
			
		||||
	log.Debug().Msg("Got user")
 | 
			
		||||
 | 
			
		||||
	// Check if password is correct
 | 
			
		||||
	if !h.Auth.VerifyUser(userSearch, login.Password) {
 | 
			
		||||
	if !h.Auth.CheckPassword(*user, login.Password) {
 | 
			
		||||
		log.Debug().Str("username", login.Username).Msg("Password incorrect")
 | 
			
		||||
		// Record failed login attempt
 | 
			
		||||
		h.Auth.RecordLoginAttempt(rateIdentifier, false)
 | 
			
		||||
@@ -416,34 +396,28 @@ func (h *Handlers) LoginHandler(c *gin.Context) {
 | 
			
		||||
	// Record successful login attempt (will reset failed attempt counter)
 | 
			
		||||
	h.Auth.RecordLoginAttempt(rateIdentifier, true)
 | 
			
		||||
 | 
			
		||||
	// Check if user is using TOTP
 | 
			
		||||
	if userSearch.Type == "local" {
 | 
			
		||||
		// Get local user
 | 
			
		||||
		localUser := h.Auth.GetLocalUser(login.Username)
 | 
			
		||||
	// Check if user has totp enabled
 | 
			
		||||
	if user.TotpSecret != "" {
 | 
			
		||||
		log.Debug().Msg("Totp enabled")
 | 
			
		||||
 | 
			
		||||
		// Check if TOTP is enabled
 | 
			
		||||
		if localUser.TotpSecret != "" {
 | 
			
		||||
			log.Debug().Msg("Totp enabled")
 | 
			
		||||
		// Set totp pending cookie
 | 
			
		||||
		h.Auth.CreateSessionCookie(c, &types.SessionCookie{
 | 
			
		||||
			Username:    login.Username,
 | 
			
		||||
			Name:        utils.Capitalize(login.Username),
 | 
			
		||||
			Email:       fmt.Sprintf("%s@%s", strings.ToLower(login.Username), h.Config.Domain),
 | 
			
		||||
			Provider:    "username",
 | 
			
		||||
			TotpPending: true,
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
			// Set totp pending cookie
 | 
			
		||||
			h.Auth.CreateSessionCookie(c, &types.SessionCookie{
 | 
			
		||||
				Username:    login.Username,
 | 
			
		||||
				Name:        utils.Capitalize(login.Username),
 | 
			
		||||
				Email:       fmt.Sprintf("%s@%s", strings.ToLower(login.Username), h.Config.Domain),
 | 
			
		||||
				Provider:    "username",
 | 
			
		||||
				TotpPending: true,
 | 
			
		||||
			})
 | 
			
		||||
		// Return totp required
 | 
			
		||||
		c.JSON(200, gin.H{
 | 
			
		||||
			"status":      200,
 | 
			
		||||
			"message":     "Waiting for totp",
 | 
			
		||||
			"totpPending": true,
 | 
			
		||||
		})
 | 
			
		||||
 | 
			
		||||
			// Return totp required
 | 
			
		||||
			c.JSON(200, gin.H{
 | 
			
		||||
				"status":      200,
 | 
			
		||||
				"message":     "Waiting for totp",
 | 
			
		||||
				"totpPending": true,
 | 
			
		||||
			})
 | 
			
		||||
 | 
			
		||||
			// Stop further processing
 | 
			
		||||
			return
 | 
			
		||||
		}
 | 
			
		||||
		// Stop further processing
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create session cookie with username as provider
 | 
			
		||||
@@ -495,7 +469,17 @@ func (h *Handlers) TotpHandler(c *gin.Context) {
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Get user
 | 
			
		||||
	user := h.Auth.GetLocalUser(userContext.Username)
 | 
			
		||||
	user := h.Auth.GetUser(userContext.Username)
 | 
			
		||||
 | 
			
		||||
	// Check if user exists
 | 
			
		||||
	if user == nil {
 | 
			
		||||
		log.Debug().Msg("User not found")
 | 
			
		||||
		c.JSON(401, gin.H{
 | 
			
		||||
			"status":  401,
 | 
			
		||||
			"message": "Unauthorized",
 | 
			
		||||
		})
 | 
			
		||||
		return
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if totp is correct
 | 
			
		||||
	ok := totp.Validate(totpReq.Code, user.TotpSecret)
 | 
			
		||||
 
 | 
			
		||||
@@ -12,12 +12,6 @@ import (
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type Hooks struct {
 | 
			
		||||
	Config    types.HooksConfig
 | 
			
		||||
	Auth      *auth.Auth
 | 
			
		||||
	Providers *providers.Providers
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewHooks(config types.HooksConfig, auth *auth.Auth, providers *providers.Providers) *Hooks {
 | 
			
		||||
	return &Hooks{
 | 
			
		||||
		Config:    config,
 | 
			
		||||
@@ -26,6 +20,12 @@ func NewHooks(config types.HooksConfig, auth *auth.Auth, providers *providers.Pr
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Hooks struct {
 | 
			
		||||
	Config    types.HooksConfig
 | 
			
		||||
	Auth      *auth.Auth
 | 
			
		||||
	Providers *providers.Providers
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (hooks *Hooks) UseUserContext(c *gin.Context) types.UserContext {
 | 
			
		||||
	// Get session cookie and basic auth
 | 
			
		||||
	cookie, err := hooks.Auth.GetSessionCookie(c)
 | 
			
		||||
@@ -35,49 +35,30 @@ func (hooks *Hooks) UseUserContext(c *gin.Context) types.UserContext {
 | 
			
		||||
	if basic != nil {
 | 
			
		||||
		log.Debug().Msg("Got basic auth")
 | 
			
		||||
 | 
			
		||||
		// Search for a user based on username
 | 
			
		||||
		userSearch := hooks.Auth.SearchUser(basic.Username)
 | 
			
		||||
		// Get user
 | 
			
		||||
		user := hooks.Auth.GetUser(basic.Username)
 | 
			
		||||
 | 
			
		||||
		if userSearch.Type == "" {
 | 
			
		||||
		// Check we have a user
 | 
			
		||||
		if user == nil {
 | 
			
		||||
			log.Error().Str("username", basic.Username).Msg("User does not exist")
 | 
			
		||||
 | 
			
		||||
			// Return empty context
 | 
			
		||||
			return types.UserContext{}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Verify the user
 | 
			
		||||
		if !hooks.Auth.VerifyUser(userSearch, basic.Password) {
 | 
			
		||||
			log.Error().Str("username", basic.Username).Msg("Password incorrect")
 | 
			
		||||
 | 
			
		||||
			// Return empty context
 | 
			
		||||
			return types.UserContext{}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		// Get the user type
 | 
			
		||||
		if userSearch.Type == "ldap" {
 | 
			
		||||
			log.Debug().Msg("User is LDAP")
 | 
			
		||||
 | 
			
		||||
		// Check if the user has a correct password
 | 
			
		||||
		if hooks.Auth.CheckPassword(*user, basic.Password) {
 | 
			
		||||
			// Return user context since we are logged in with basic auth
 | 
			
		||||
			return types.UserContext{
 | 
			
		||||
				Username:    basic.Username,
 | 
			
		||||
				Name:        utils.Capitalize(basic.Username),
 | 
			
		||||
				Email:       fmt.Sprintf("%s@%s", strings.ToLower(basic.Username), hooks.Config.Domain),
 | 
			
		||||
				IsLoggedIn:  true,
 | 
			
		||||
				Provider:    "basic",
 | 
			
		||||
				TotpEnabled: false,
 | 
			
		||||
				TotpEnabled: user.TotpSecret != "",
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		user := hooks.Auth.GetLocalUser(basic.Username)
 | 
			
		||||
 | 
			
		||||
		return types.UserContext{
 | 
			
		||||
			Username:    basic.Username,
 | 
			
		||||
			Name:        utils.Capitalize(basic.Username),
 | 
			
		||||
			Email:       fmt.Sprintf("%s@%s", strings.ToLower(basic.Username), hooks.Config.Domain),
 | 
			
		||||
			IsLoggedIn:  true,
 | 
			
		||||
			Provider:    "basic",
 | 
			
		||||
			TotpEnabled: user.TotpSecret != "",
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check cookie error after basic auth
 | 
			
		||||
@@ -104,25 +85,18 @@ func (hooks *Hooks) UseUserContext(c *gin.Context) types.UserContext {
 | 
			
		||||
	if cookie.Provider == "username" {
 | 
			
		||||
		log.Debug().Msg("Provider is username")
 | 
			
		||||
 | 
			
		||||
		// Search for the user with the username
 | 
			
		||||
		userSearch := hooks.Auth.SearchUser(cookie.Username)
 | 
			
		||||
		// Check if user exists
 | 
			
		||||
		if hooks.Auth.GetUser(cookie.Username) != nil {
 | 
			
		||||
			log.Debug().Msg("User exists")
 | 
			
		||||
 | 
			
		||||
		if userSearch.Type == "" {
 | 
			
		||||
			log.Error().Str("username", cookie.Username).Msg("User does not exist")
 | 
			
		||||
 | 
			
		||||
			// Return empty context
 | 
			
		||||
			return types.UserContext{}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Debug().Str("type", userSearch.Type).Msg("User exists")
 | 
			
		||||
 | 
			
		||||
		// It exists so we are logged in
 | 
			
		||||
		return types.UserContext{
 | 
			
		||||
			Username:   cookie.Username,
 | 
			
		||||
			Name:       cookie.Name,
 | 
			
		||||
			Email:      cookie.Email,
 | 
			
		||||
			IsLoggedIn: true,
 | 
			
		||||
			Provider:   "username",
 | 
			
		||||
			// It exists so we are logged in
 | 
			
		||||
			return types.UserContext{
 | 
			
		||||
				Username:   cookie.Username,
 | 
			
		||||
				Name:       cookie.Name,
 | 
			
		||||
				Email:      cookie.Email,
 | 
			
		||||
				IsLoggedIn: true,
 | 
			
		||||
				Provider:   "username",
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,77 +0,0 @@
 | 
			
		||||
package ldap
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"crypto/tls"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
 | 
			
		||||
	ldapgo "github.com/go-ldap/ldap/v3"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type LDAP struct {
 | 
			
		||||
	Config types.LdapConfig
 | 
			
		||||
	Conn   *ldapgo.Conn
 | 
			
		||||
	BaseDN string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewLDAP(config types.LdapConfig) (*LDAP, error) {
 | 
			
		||||
	// Connect to the LDAP server
 | 
			
		||||
	conn, err := ldapgo.DialURL(config.Address, ldapgo.DialWithTLSConfig(&tls.Config{
 | 
			
		||||
		InsecureSkipVerify: config.Insecure,
 | 
			
		||||
		MinVersion:         tls.VersionTLS12,
 | 
			
		||||
	}))
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Bind to the LDAP server with the provided credentials
 | 
			
		||||
	err = conn.Bind(config.BindDN, config.BindPassword)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return &LDAP{
 | 
			
		||||
		Config: config,
 | 
			
		||||
		Conn:   conn,
 | 
			
		||||
		BaseDN: config.BaseDN,
 | 
			
		||||
	}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (l *LDAP) Search(username string) (string, error) {
 | 
			
		||||
	// Escape the username to prevent LDAP injection
 | 
			
		||||
	escapedUsername := ldapgo.EscapeFilter(username)
 | 
			
		||||
	filter := fmt.Sprintf(l.Config.SearchFilter, escapedUsername)
 | 
			
		||||
 | 
			
		||||
	// Create a search request to find the user by username
 | 
			
		||||
	searchRequest := ldapgo.NewSearchRequest(
 | 
			
		||||
		l.BaseDN,
 | 
			
		||||
		ldapgo.ScopeWholeSubtree, ldapgo.NeverDerefAliases, 0, 0, false,
 | 
			
		||||
		filter,
 | 
			
		||||
		[]string{"dn"},
 | 
			
		||||
		nil,
 | 
			
		||||
	)
 | 
			
		||||
 | 
			
		||||
	// Perform the search
 | 
			
		||||
	searchResult, err := l.Conn.Search(searchRequest)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return "", err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if len(searchResult.Entries) != 1 {
 | 
			
		||||
		return "", fmt.Errorf("err multiple or no entries found for user %s", username)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// User found, return the distinguished name (DN)
 | 
			
		||||
	userDN := searchResult.Entries[0].DN
 | 
			
		||||
 | 
			
		||||
	return userDN, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (l *LDAP) Bind(userDN string, password string) error {
 | 
			
		||||
	// Bind to the LDAP server with the user's DN and password
 | 
			
		||||
	err := l.Conn.Bind(userDN, password)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return err
 | 
			
		||||
	}
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
@@ -10,24 +10,32 @@ import (
 | 
			
		||||
	"golang.org/x/oauth2"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
type OAuth struct {
 | 
			
		||||
	Config   oauth2.Config
 | 
			
		||||
	Context  context.Context
 | 
			
		||||
	Token    *oauth2.Token
 | 
			
		||||
	Verifier string
 | 
			
		||||
func NewOAuth(config oauth2.Config, insecureSkipVerify bool) *OAuth {
 | 
			
		||||
	return &OAuth{
 | 
			
		||||
		Config:             config,
 | 
			
		||||
		InsecureSkipVerify: insecureSkipVerify,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewOAuth(config oauth2.Config, insecureSkipVerify bool) *OAuth {
 | 
			
		||||
type OAuth struct {
 | 
			
		||||
	Config             oauth2.Config
 | 
			
		||||
	Context            context.Context
 | 
			
		||||
	Token              *oauth2.Token
 | 
			
		||||
	Verifier           string
 | 
			
		||||
	InsecureSkipVerify bool
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (oauth *OAuth) Init() {
 | 
			
		||||
	// Create transport with TLS
 | 
			
		||||
	transport := &http.Transport{
 | 
			
		||||
		TLSClientConfig: &tls.Config{
 | 
			
		||||
			InsecureSkipVerify: insecureSkipVerify,
 | 
			
		||||
			InsecureSkipVerify: oauth.InsecureSkipVerify,
 | 
			
		||||
			MinVersion:         tls.VersionTLS12,
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create a new context
 | 
			
		||||
	ctx := context.Background()
 | 
			
		||||
	oauth.Context = context.Background()
 | 
			
		||||
 | 
			
		||||
	// Create the HTTP client with the transport
 | 
			
		||||
	httpClient := &http.Client{
 | 
			
		||||
@@ -35,16 +43,9 @@ func NewOAuth(config oauth2.Config, insecureSkipVerify bool) *OAuth {
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the HTTP client in the context
 | 
			
		||||
	ctx = context.WithValue(ctx, oauth2.HTTPClient, httpClient)
 | 
			
		||||
 | 
			
		||||
	oauth.Context = context.WithValue(oauth.Context, oauth2.HTTPClient, httpClient)
 | 
			
		||||
	// Create the verifier
 | 
			
		||||
	verifier := oauth2.GenerateVerifier()
 | 
			
		||||
 | 
			
		||||
	return &OAuth{
 | 
			
		||||
		Config:   config,
 | 
			
		||||
		Context:  ctx,
 | 
			
		||||
		Verifier: verifier,
 | 
			
		||||
	}
 | 
			
		||||
	oauth.Verifier = oauth2.GenerateVerifier()
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (oauth *OAuth) GetAuthURL(state string) string {
 | 
			
		||||
 
 | 
			
		||||
@@ -11,6 +11,12 @@ import (
 | 
			
		||||
	"golang.org/x/oauth2/endpoints"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func NewProviders(config types.OAuthConfig) *Providers {
 | 
			
		||||
	return &Providers{
 | 
			
		||||
		Config: config,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Providers struct {
 | 
			
		||||
	Config  types.OAuthConfig
 | 
			
		||||
	Github  *oauth.OAuth
 | 
			
		||||
@@ -18,57 +24,60 @@ type Providers struct {
 | 
			
		||||
	Generic *oauth.OAuth
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func NewProviders(config types.OAuthConfig) *Providers {
 | 
			
		||||
	providers := &Providers{
 | 
			
		||||
		Config: config,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
func (providers *Providers) Init() {
 | 
			
		||||
	// If we have a client id and secret for github, initialize the oauth provider
 | 
			
		||||
	if config.GithubClientId != "" && config.GithubClientSecret != "" {
 | 
			
		||||
	if providers.Config.GithubClientId != "" && providers.Config.GithubClientSecret != "" {
 | 
			
		||||
		log.Info().Msg("Initializing Github OAuth")
 | 
			
		||||
 | 
			
		||||
		// Create a new oauth provider with the github config
 | 
			
		||||
		providers.Github = oauth.NewOAuth(oauth2.Config{
 | 
			
		||||
			ClientID:     config.GithubClientId,
 | 
			
		||||
			ClientSecret: config.GithubClientSecret,
 | 
			
		||||
			RedirectURL:  fmt.Sprintf("%s/api/oauth/callback/github", config.AppURL),
 | 
			
		||||
			ClientID:     providers.Config.GithubClientId,
 | 
			
		||||
			ClientSecret: providers.Config.GithubClientSecret,
 | 
			
		||||
			RedirectURL:  fmt.Sprintf("%s/api/oauth/callback/github", providers.Config.AppURL),
 | 
			
		||||
			Scopes:       GithubScopes(),
 | 
			
		||||
			Endpoint:     endpoints.GitHub,
 | 
			
		||||
		}, false)
 | 
			
		||||
 | 
			
		||||
		// Initialize the oauth provider
 | 
			
		||||
		providers.Github.Init()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If we have a client id and secret for google, initialize the oauth provider
 | 
			
		||||
	if config.GoogleClientId != "" && config.GoogleClientSecret != "" {
 | 
			
		||||
	if providers.Config.GoogleClientId != "" && providers.Config.GoogleClientSecret != "" {
 | 
			
		||||
		log.Info().Msg("Initializing Google OAuth")
 | 
			
		||||
 | 
			
		||||
		// Create a new oauth provider with the google config
 | 
			
		||||
		providers.Google = oauth.NewOAuth(oauth2.Config{
 | 
			
		||||
			ClientID:     config.GoogleClientId,
 | 
			
		||||
			ClientSecret: config.GoogleClientSecret,
 | 
			
		||||
			RedirectURL:  fmt.Sprintf("%s/api/oauth/callback/google", config.AppURL),
 | 
			
		||||
			ClientID:     providers.Config.GoogleClientId,
 | 
			
		||||
			ClientSecret: providers.Config.GoogleClientSecret,
 | 
			
		||||
			RedirectURL:  fmt.Sprintf("%s/api/oauth/callback/google", providers.Config.AppURL),
 | 
			
		||||
			Scopes:       GoogleScopes(),
 | 
			
		||||
			Endpoint:     endpoints.Google,
 | 
			
		||||
		}, false)
 | 
			
		||||
 | 
			
		||||
		// Initialize the oauth provider
 | 
			
		||||
		providers.Google.Init()
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// If we have a client id and secret for generic oauth, initialize the oauth provider
 | 
			
		||||
	if config.GenericClientId != "" && config.GenericClientSecret != "" {
 | 
			
		||||
	if providers.Config.GenericClientId != "" && providers.Config.GenericClientSecret != "" {
 | 
			
		||||
		log.Info().Msg("Initializing Generic OAuth")
 | 
			
		||||
 | 
			
		||||
		// Create a new oauth provider with the generic config
 | 
			
		||||
		providers.Generic = oauth.NewOAuth(oauth2.Config{
 | 
			
		||||
			ClientID:     config.GenericClientId,
 | 
			
		||||
			ClientSecret: config.GenericClientSecret,
 | 
			
		||||
			RedirectURL:  fmt.Sprintf("%s/api/oauth/callback/generic", config.AppURL),
 | 
			
		||||
			Scopes:       config.GenericScopes,
 | 
			
		||||
			ClientID:     providers.Config.GenericClientId,
 | 
			
		||||
			ClientSecret: providers.Config.GenericClientSecret,
 | 
			
		||||
			RedirectURL:  fmt.Sprintf("%s/api/oauth/callback/generic", providers.Config.AppURL),
 | 
			
		||||
			Scopes:       providers.Config.GenericScopes,
 | 
			
		||||
			Endpoint: oauth2.Endpoint{
 | 
			
		||||
				AuthURL:  config.GenericAuthURL,
 | 
			
		||||
				TokenURL: config.GenericTokenURL,
 | 
			
		||||
				AuthURL:  providers.Config.GenericAuthURL,
 | 
			
		||||
				TokenURL: providers.Config.GenericTokenURL,
 | 
			
		||||
			},
 | 
			
		||||
		}, config.GenericSkipSSL)
 | 
			
		||||
	}
 | 
			
		||||
		}, providers.Config.GenericSkipSSL)
 | 
			
		||||
 | 
			
		||||
	return providers
 | 
			
		||||
		// Initialize the oauth provider
 | 
			
		||||
		providers.Generic.Init()
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (providers *Providers) GetProvider(provider string) *oauth.OAuth {
 | 
			
		||||
 
 | 
			
		||||
@@ -1,521 +0,0 @@
 | 
			
		||||
package server_test
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"encoding/json"
 | 
			
		||||
	"io"
 | 
			
		||||
	"net/http"
 | 
			
		||||
	"net/http/httptest"
 | 
			
		||||
	"reflect"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"testing"
 | 
			
		||||
	"time"
 | 
			
		||||
	"tinyauth/internal/auth"
 | 
			
		||||
	"tinyauth/internal/docker"
 | 
			
		||||
	"tinyauth/internal/handlers"
 | 
			
		||||
	"tinyauth/internal/hooks"
 | 
			
		||||
	"tinyauth/internal/providers"
 | 
			
		||||
	"tinyauth/internal/server"
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
 | 
			
		||||
	"github.com/magiconair/properties/assert"
 | 
			
		||||
	"github.com/pquerna/otp/totp"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
// Simple server config for tests
 | 
			
		||||
var serverConfig = types.ServerConfig{
 | 
			
		||||
	Port:    8080,
 | 
			
		||||
	Address: "0.0.0.0",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Simple handlers config for tests
 | 
			
		||||
var handlersConfig = types.HandlersConfig{
 | 
			
		||||
	AppURL:                "http://localhost:8080",
 | 
			
		||||
	Domain:                "localhost",
 | 
			
		||||
	DisableContinue:       false,
 | 
			
		||||
	CookieSecure:          false,
 | 
			
		||||
	Title:                 "Tinyauth",
 | 
			
		||||
	GenericName:           "Generic",
 | 
			
		||||
	ForgotPasswordMessage: "Message",
 | 
			
		||||
	CsrfCookieName:        "tinyauth-csrf",
 | 
			
		||||
	RedirectCookieName:    "tinyauth-redirect",
 | 
			
		||||
	BackgroundImage:       "https://example.com/image.png",
 | 
			
		||||
	OAuthAutoRedirect:     "none",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Simple auth config for tests
 | 
			
		||||
var authConfig = types.AuthConfig{
 | 
			
		||||
	Users:             types.Users{},
 | 
			
		||||
	OauthWhitelist:    "",
 | 
			
		||||
	HMACSecret:        "4bZ9K.*:;zH=,9zG!meUxu.B5-S[7.V.", // Complex on purpose
 | 
			
		||||
	EncryptionSecret:  "\\:!R(u[Sbv6ZLm.7es)H|OqH4y}0u\\rj",
 | 
			
		||||
	CookieSecure:      false,
 | 
			
		||||
	SessionExpiry:     3600,
 | 
			
		||||
	LoginTimeout:      0,
 | 
			
		||||
	LoginMaxRetries:   0,
 | 
			
		||||
	SessionCookieName: "tinyauth-session",
 | 
			
		||||
	Domain:            "localhost",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Simple hooks config for tests
 | 
			
		||||
var hooksConfig = types.HooksConfig{
 | 
			
		||||
	Domain: "localhost",
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Cookie
 | 
			
		||||
var cookie = "MTc1MTkyMzM5MnxiME9aTzlGQjZMNEJMdDZMc0lHMk9zcXQyME9SR1ZnUmlaYWZNcWplek5vcVNpdkdHRTZqb09YWkVUYUN6NEt4MkEyOGEyX2hFQWZEUEYtbllDX0h5eDBCb3VyT2phQlRpZWFfRFdTMGw2WUg2VWw4RGdNbEhQclotOUJjblJGaWFQcmhyaWFna0dXRWNud2c1akg5eEpLZ3JzS0pfWktscVZyckZFR1VDX0R5QjFOT0hzMTNKb18ySEMxZlluSWNxa1ByM0VhSzNyMkRtdDNORWJXVGFYSnMzWjFGa0lrZlhSTWduRmttMHhQUXN4UFhNbHFXY0lBWjBnUWpKU0xXMHRubjlKbjV0LXBGdjk0MmpJX0xMX1ZYblVJVW9LWUJoWmpNanVXNkNjamhYWlR2V29rY0RNYWkxY2lMQnpqLUI2cHMyYTZkWWgtWnlFdGN0amh2WURUeUNGT3ZLS1FJVUFIb0NWR1RPMlRtY2c9PXwerwFtb9urOXnwA02qXbLeorMloaK_paQd0in4BAesmg=="
 | 
			
		||||
 | 
			
		||||
// User
 | 
			
		||||
var user = types.User{
 | 
			
		||||
	Username: "user",
 | 
			
		||||
	Password: "$2a$10$AvGHLTYv3xiRJ0xV9xs3XeVIlkGTygI9nqIamFYB5Xu.5.0UWF7B6", // pass
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Initialize the server for tests
 | 
			
		||||
func getServer(t *testing.T) *server.Server {
 | 
			
		||||
	// Create docker service
 | 
			
		||||
	docker, err := docker.NewDocker()
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Failed to initialize docker: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create auth service
 | 
			
		||||
	authConfig.Users = types.Users{
 | 
			
		||||
		{
 | 
			
		||||
			Username:   user.Username,
 | 
			
		||||
			Password:   user.Password,
 | 
			
		||||
			TotpSecret: user.TotpSecret,
 | 
			
		||||
		},
 | 
			
		||||
	}
 | 
			
		||||
	auth := auth.NewAuth(authConfig, docker, nil)
 | 
			
		||||
 | 
			
		||||
	// Create providers service
 | 
			
		||||
	providers := providers.NewProviders(types.OAuthConfig{})
 | 
			
		||||
 | 
			
		||||
	// Create hooks service
 | 
			
		||||
	hooks := hooks.NewHooks(hooksConfig, auth, providers)
 | 
			
		||||
 | 
			
		||||
	// Create handlers service
 | 
			
		||||
	handlers := handlers.NewHandlers(handlersConfig, auth, hooks, providers, docker)
 | 
			
		||||
 | 
			
		||||
	// Create server
 | 
			
		||||
	srv, err := server.NewServer(serverConfig, handlers)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Failed to create server: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Return the server
 | 
			
		||||
	return srv
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test login
 | 
			
		||||
func TestLogin(t *testing.T) {
 | 
			
		||||
	t.Log("Testing login")
 | 
			
		||||
 | 
			
		||||
	// Get server
 | 
			
		||||
	srv := getServer(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	user := types.LoginRequest{
 | 
			
		||||
		Username: "user",
 | 
			
		||||
		Password: "pass",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	json, err := json.Marshal(user)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error marshalling json: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("POST", "/api/login", strings.NewReader(string(json)))
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Get the result cookie
 | 
			
		||||
	cookies := recorder.Result().Cookies()
 | 
			
		||||
 | 
			
		||||
	// Check if the cookie is set
 | 
			
		||||
	if len(cookies) == 0 {
 | 
			
		||||
		t.Fatalf("Cookie not set")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie for further tests
 | 
			
		||||
	cookie = cookies[0].Value
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test app context
 | 
			
		||||
func TestAppContext(t *testing.T) {
 | 
			
		||||
	t.Log("Testing app context")
 | 
			
		||||
 | 
			
		||||
	// Get server
 | 
			
		||||
	srv := getServer(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("GET", "/api/app", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Read the body of the response
 | 
			
		||||
	body, err := io.ReadAll(recorder.Body)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error getting body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Unmarshal the body into the user struct
 | 
			
		||||
	var app types.AppContext
 | 
			
		||||
 | 
			
		||||
	err = json.Unmarshal(body, &app)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error unmarshalling body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create tests values
 | 
			
		||||
	expected := types.AppContext{
 | 
			
		||||
		Status:                200,
 | 
			
		||||
		Message:               "OK",
 | 
			
		||||
		ConfiguredProviders:   []string{"username"},
 | 
			
		||||
		DisableContinue:       false,
 | 
			
		||||
		Title:                 "Tinyauth",
 | 
			
		||||
		GenericName:           "Generic",
 | 
			
		||||
		ForgotPasswordMessage: "Message",
 | 
			
		||||
		BackgroundImage:       "https://example.com/image.png",
 | 
			
		||||
		OAuthAutoRedirect:     "none",
 | 
			
		||||
		Domain:                "localhost",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// We should get the username back
 | 
			
		||||
	if !reflect.DeepEqual(app, expected) {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, app)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test user context
 | 
			
		||||
func TestUserContext(t *testing.T) {
 | 
			
		||||
	// Refresh the cookie
 | 
			
		||||
	TestLogin(t)
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing user context")
 | 
			
		||||
 | 
			
		||||
	// Get server
 | 
			
		||||
	srv := getServer(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("GET", "/api/user", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth-session",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Read the body of the response
 | 
			
		||||
	body, err := io.ReadAll(recorder.Body)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error getting body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Unmarshal the body into the user struct
 | 
			
		||||
	type User struct {
 | 
			
		||||
		Username string `json:"username"`
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var user User
 | 
			
		||||
 | 
			
		||||
	err = json.Unmarshal(body, &user)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error unmarshalling body: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// We should get the username back
 | 
			
		||||
	if user.Username != "user" {
 | 
			
		||||
		t.Fatalf("Expected user, got %s", user.Username)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test logout
 | 
			
		||||
func TestLogout(t *testing.T) {
 | 
			
		||||
	// Refresh the cookie
 | 
			
		||||
	TestLogin(t)
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing logout")
 | 
			
		||||
 | 
			
		||||
	// Get server
 | 
			
		||||
	srv := getServer(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("POST", "/api/logout", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth-session",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Check if the cookie is different (means the cookie is gone)
 | 
			
		||||
	if recorder.Result().Cookies()[0].Value == cookie {
 | 
			
		||||
		t.Fatalf("Cookie not flushed")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test auth endpoint
 | 
			
		||||
func TestAuth(t *testing.T) {
 | 
			
		||||
	// Refresh the cookie
 | 
			
		||||
	TestLogin(t)
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing auth endpoint")
 | 
			
		||||
 | 
			
		||||
	// Get server
 | 
			
		||||
	srv := getServer(t)
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("GET", "/api/auth/traefik", nil)
 | 
			
		||||
 | 
			
		||||
	// Set the accept header
 | 
			
		||||
	req.Header.Set("Accept", "text/html")
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusTemporaryRedirect)
 | 
			
		||||
 | 
			
		||||
	// Recreate recorder
 | 
			
		||||
	recorder = httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Recreate the request
 | 
			
		||||
	req, err = http.NewRequest("GET", "/api/auth/traefik", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Test with the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth-session",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request again
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Recreate recorder
 | 
			
		||||
	recorder = httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Recreate the request
 | 
			
		||||
	req, err = http.NewRequest("GET", "/api/auth/nginx", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Serve the request again
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusUnauthorized)
 | 
			
		||||
 | 
			
		||||
	// Recreate recorder
 | 
			
		||||
	recorder = httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Recreate the request
 | 
			
		||||
	req, err = http.NewRequest("GET", "/api/auth/nginx", nil)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Test with the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth-session",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request again
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func TestTOTP(t *testing.T) {
 | 
			
		||||
	t.Log("Testing TOTP")
 | 
			
		||||
 | 
			
		||||
	// Generate totp secret
 | 
			
		||||
	key, err := totp.Generate(totp.GenerateOpts{
 | 
			
		||||
		Issuer:      "Tinyauth",
 | 
			
		||||
		AccountName: user.Username,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Failed to generate TOTP secret: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create secret
 | 
			
		||||
	secret := key.Secret()
 | 
			
		||||
 | 
			
		||||
	// Set the user's TOTP secret
 | 
			
		||||
	user.TotpSecret = secret
 | 
			
		||||
 | 
			
		||||
	// Get server
 | 
			
		||||
	srv := getServer(t)
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	user := types.LoginRequest{
 | 
			
		||||
		Username: "user",
 | 
			
		||||
		Password: "pass",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	loginJson, err := json.Marshal(user)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error marshalling json: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder := httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err := http.NewRequest("POST", "/api/login", strings.NewReader(string(loginJson)))
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
 | 
			
		||||
	// Set the cookie for next test
 | 
			
		||||
	cookie = recorder.Result().Cookies()[0].Value
 | 
			
		||||
 | 
			
		||||
	// Create TOTP code
 | 
			
		||||
	code, err := totp.GenerateCode(secret, time.Now())
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Failed to generate TOTP code: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create TOTP request
 | 
			
		||||
	totpRequest := types.TotpRequest{
 | 
			
		||||
		Code: code,
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Marshal the TOTP request
 | 
			
		||||
	totpJson, err := json.Marshal(totpRequest)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error marshalling TOTP request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Create recorder
 | 
			
		||||
	recorder = httptest.NewRecorder()
 | 
			
		||||
 | 
			
		||||
	// Create request
 | 
			
		||||
	req, err = http.NewRequest("POST", "/api/totp", strings.NewReader(string(totpJson)))
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error creating request: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Set the cookie
 | 
			
		||||
	req.AddCookie(&http.Cookie{
 | 
			
		||||
		Name:  "tinyauth-session",
 | 
			
		||||
		Value: cookie,
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Serve the request
 | 
			
		||||
	srv.Router.ServeHTTP(recorder, req)
 | 
			
		||||
 | 
			
		||||
	// Assert
 | 
			
		||||
	assert.Equal(t, recorder.Code, http.StatusOK)
 | 
			
		||||
}
 | 
			
		||||
@@ -36,12 +36,6 @@ type Config struct {
 | 
			
		||||
	LoginMaxRetries         int    `mapstructure:"login-max-retries"`
 | 
			
		||||
	FogotPasswordMessage    string `mapstructure:"forgot-password-message" validate:"required"`
 | 
			
		||||
	BackgroundImage         string `mapstructure:"background-image" validate:"required"`
 | 
			
		||||
	LdapAddress             string `mapstructure:"ldap-address"`
 | 
			
		||||
	LdapBindDN              string `mapstructure:"ldap-bind-dn"`
 | 
			
		||||
	LdapBindPassword        string `mapstructure:"ldap-bind-password"`
 | 
			
		||||
	LdapBaseDN              string `mapstructure:"ldap-base-dn"`
 | 
			
		||||
	LdapInsecure            bool   `mapstructure:"ldap-insecure"`
 | 
			
		||||
	LdapSearchFilter        string `mapstructure:"ldap-search-filter"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Server configuration
 | 
			
		||||
@@ -75,8 +69,8 @@ type OAuthConfig struct {
 | 
			
		||||
	AppURL              string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// ServerConfig is the configuration for the server
 | 
			
		||||
type ServerConfig struct {
 | 
			
		||||
// APIConfig is the configuration for the API
 | 
			
		||||
type APIConfig struct {
 | 
			
		||||
	Port    int
 | 
			
		||||
	Address string
 | 
			
		||||
}
 | 
			
		||||
@@ -86,13 +80,12 @@ type AuthConfig struct {
 | 
			
		||||
	Users             Users
 | 
			
		||||
	OauthWhitelist    string
 | 
			
		||||
	SessionExpiry     int
 | 
			
		||||
	Secret            string
 | 
			
		||||
	CookieSecure      bool
 | 
			
		||||
	Domain            string
 | 
			
		||||
	LoginTimeout      int
 | 
			
		||||
	LoginMaxRetries   int
 | 
			
		||||
	SessionCookieName string
 | 
			
		||||
	HMACSecret        string
 | 
			
		||||
	EncryptionSecret  string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// HooksConfig is the configuration for the hooks service
 | 
			
		||||
@@ -108,21 +101,14 @@ type OAuthLabels struct {
 | 
			
		||||
 | 
			
		||||
// Basic auth labels for a tinyauth protected container
 | 
			
		||||
type BasicLabels struct {
 | 
			
		||||
	Username string
 | 
			
		||||
	Password PassowrdLabels
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// PassowrdLabels is a struct that contains the password labels for a tinyauth protected container
 | 
			
		||||
type PassowrdLabels struct {
 | 
			
		||||
	Plain string
 | 
			
		||||
	File  string
 | 
			
		||||
	User     string
 | 
			
		||||
	Password string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// IP labels for a tinyauth protected container
 | 
			
		||||
type IPLabels struct {
 | 
			
		||||
	Allow  []string
 | 
			
		||||
	Block  []string
 | 
			
		||||
	Bypass []string
 | 
			
		||||
	Allow []string
 | 
			
		||||
	Block []string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Labels is a struct that contains the labels for a tinyauth protected container
 | 
			
		||||
@@ -130,18 +116,8 @@ type Labels struct {
 | 
			
		||||
	Users   string
 | 
			
		||||
	Allowed string
 | 
			
		||||
	Headers []string
 | 
			
		||||
	Domain  []string
 | 
			
		||||
	Domain  string
 | 
			
		||||
	Basic   BasicLabels
 | 
			
		||||
	OAuth   OAuthLabels
 | 
			
		||||
	IP      IPLabels
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Ldap config is a struct that contains the configuration for the LDAP service
 | 
			
		||||
type LdapConfig struct {
 | 
			
		||||
	Address      string
 | 
			
		||||
	BindDN       string
 | 
			
		||||
	BindPassword string
 | 
			
		||||
	BaseDN       string
 | 
			
		||||
	Insecure     bool
 | 
			
		||||
	SearchFilter string
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -12,12 +12,6 @@ type User struct {
 | 
			
		||||
	TotpSecret string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// UserSearch is the response of the get user
 | 
			
		||||
type UserSearch struct {
 | 
			
		||||
	Username string
 | 
			
		||||
	Type     string // "local", "ldap" or empty
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Users is a list of users
 | 
			
		||||
type Users []User
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -1,11 +1,8 @@
 | 
			
		||||
package utils
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"bytes"
 | 
			
		||||
	"crypto/sha256"
 | 
			
		||||
	"encoding/base64"
 | 
			
		||||
	"errors"
 | 
			
		||||
	"io"
 | 
			
		||||
	"net"
 | 
			
		||||
	"net/url"
 | 
			
		||||
	"os"
 | 
			
		||||
@@ -14,7 +11,6 @@ import (
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
 | 
			
		||||
	"github.com/traefik/paerser/parser"
 | 
			
		||||
	"golang.org/x/crypto/hkdf"
 | 
			
		||||
 | 
			
		||||
	"github.com/google/uuid"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
@@ -188,7 +184,7 @@ func ParseHeaders(headers []string) map[string]string {
 | 
			
		||||
	// Loop through the headers
 | 
			
		||||
	for _, header := range headers {
 | 
			
		||||
		split := strings.SplitN(header, "=", 2)
 | 
			
		||||
		if len(split) != 2 || strings.TrimSpace(split[0]) == "" || strings.TrimSpace(split[1]) == "" {
 | 
			
		||||
		if len(split) != 2 {
 | 
			
		||||
			log.Warn().Str("header", header).Msg("Invalid header format, skipping")
 | 
			
		||||
			continue
 | 
			
		||||
		}
 | 
			
		||||
@@ -292,17 +288,17 @@ func ParseSecretFile(contents string) string {
 | 
			
		||||
	return ""
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Check if a string matches a regex or if it is included in a comma separated list
 | 
			
		||||
func CheckFilter(filter string, str string) bool {
 | 
			
		||||
	// Check if the filter is empty
 | 
			
		||||
	if len(strings.TrimSpace(filter)) == 0 {
 | 
			
		||||
// Check if a string matches a regex or a whitelist
 | 
			
		||||
func CheckWhitelist(whitelist string, str string) bool {
 | 
			
		||||
	// Check if the whitelist is empty
 | 
			
		||||
	if len(strings.TrimSpace(whitelist)) == 0 {
 | 
			
		||||
		return true
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if the filter is a regex
 | 
			
		||||
	if strings.HasPrefix(filter, "/") && strings.HasSuffix(filter, "/") {
 | 
			
		||||
	// Check if the whitelist is a regex
 | 
			
		||||
	if strings.HasPrefix(whitelist, "/") && strings.HasSuffix(whitelist, "/") {
 | 
			
		||||
		// Create regex
 | 
			
		||||
		re, err := regexp.Compile(filter[1 : len(filter)-1])
 | 
			
		||||
		re, err := regexp.Compile(whitelist[1 : len(whitelist)-1])
 | 
			
		||||
 | 
			
		||||
		// Check if there was an error
 | 
			
		||||
		if err != nil {
 | 
			
		||||
@@ -316,11 +312,11 @@ func CheckFilter(filter string, str string) bool {
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Split the filter by comma
 | 
			
		||||
	filterSplit := strings.Split(filter, ",")
 | 
			
		||||
	// Split the whitelist by comma
 | 
			
		||||
	whitelistSplit := strings.Split(whitelist, ",")
 | 
			
		||||
 | 
			
		||||
	// Loop through the filter items
 | 
			
		||||
	for _, item := range filterSplit {
 | 
			
		||||
	// Loop through the whitelist
 | 
			
		||||
	for _, item := range whitelistSplit {
 | 
			
		||||
		// Check if the item matches with the string
 | 
			
		||||
		if strings.TrimSpace(item) == str {
 | 
			
		||||
			return true
 | 
			
		||||
@@ -409,32 +405,3 @@ func FilterIP(filter string, ip string) (bool, error) {
 | 
			
		||||
	// If the filter is not a CIDR range or a single IP, return false
 | 
			
		||||
	return false, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func DeriveKey(secret string, info string) (string, error) {
 | 
			
		||||
	// Create hashing function
 | 
			
		||||
	hash := sha256.New
 | 
			
		||||
 | 
			
		||||
	// Create a new key using the secret and info
 | 
			
		||||
	hkdf := hkdf.New(hash, []byte(secret), nil, []byte(info)) // I am not using a salt because I just want two different keys from one secret, maybe bad practice
 | 
			
		||||
 | 
			
		||||
	// Create a new key
 | 
			
		||||
	key := make([]byte, 24)
 | 
			
		||||
 | 
			
		||||
	// Read the key from the HKDF
 | 
			
		||||
	_, err := io.ReadFull(hkdf, key)
 | 
			
		||||
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return "", err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Verify the key is not empty
 | 
			
		||||
	if bytes.Equal(key, make([]byte, 24)) {
 | 
			
		||||
		return "", errors.New("derived key is empty")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Encode the key to base64
 | 
			
		||||
	encodedKey := base64.StdEncoding.EncodeToString(key)
 | 
			
		||||
 | 
			
		||||
	// Return the key as a base64 encoded string
 | 
			
		||||
	return encodedKey, nil
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -315,6 +315,25 @@ func TestGetLabels(t *testing.T) {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test the filter function
 | 
			
		||||
func TestFilter(t *testing.T) {
 | 
			
		||||
	t.Log("Testing filter helper")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	data := []string{"", "val1", "", "val2", "", "val3", ""}
 | 
			
		||||
	expected := []string{"val1", "val2", "val3"}
 | 
			
		||||
 | 
			
		||||
	// Test the filter function
 | 
			
		||||
	result := utils.Filter(data, func(val string) bool {
 | 
			
		||||
		return val != ""
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if !reflect.DeepEqual(expected, result) {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test parse user
 | 
			
		||||
func TestParseUser(t *testing.T) {
 | 
			
		||||
	t.Log("Testing parse user with a valid user")
 | 
			
		||||
@@ -377,77 +396,108 @@ func TestParseUser(t *testing.T) {
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test the check filter function
 | 
			
		||||
func TestCheckFilter(t *testing.T) {
 | 
			
		||||
	t.Log("Testing check filter with a comma separated list")
 | 
			
		||||
// Test the whitelist function
 | 
			
		||||
func TestCheckWhitelist(t *testing.T) {
 | 
			
		||||
	t.Log("Testing check whitelist with a comma whitelist")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter := "user1,user2,user3"
 | 
			
		||||
	whitelist := "user1,user2,user3"
 | 
			
		||||
	str := "user1"
 | 
			
		||||
	expected := true
 | 
			
		||||
 | 
			
		||||
	// Test the check filter function
 | 
			
		||||
	result := utils.CheckFilter(filter, str)
 | 
			
		||||
	// Test the check whitelist function
 | 
			
		||||
	result := utils.CheckWhitelist(whitelist, str)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing check filter with a regex filter")
 | 
			
		||||
	t.Log("Testing check whitelist with a regex whitelist")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = "/^user[0-9]+$/"
 | 
			
		||||
	whitelist = "/^user[0-9]+$/"
 | 
			
		||||
	str = "user1"
 | 
			
		||||
	expected = true
 | 
			
		||||
 | 
			
		||||
	// Test the check filter function
 | 
			
		||||
	result = utils.CheckFilter(filter, str)
 | 
			
		||||
	// Test the check whitelist function
 | 
			
		||||
	result = utils.CheckWhitelist(whitelist, str)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing check filter with an empty filter")
 | 
			
		||||
	t.Log("Testing check whitelist with an empty whitelist")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = ""
 | 
			
		||||
	whitelist = ""
 | 
			
		||||
	str = "user1"
 | 
			
		||||
	expected = true
 | 
			
		||||
 | 
			
		||||
	// Test the check filter function
 | 
			
		||||
	result = utils.CheckFilter(filter, str)
 | 
			
		||||
	// Test the check whitelist function
 | 
			
		||||
	result = utils.CheckWhitelist(whitelist, str)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing check filter with an invalid regex filter")
 | 
			
		||||
	t.Log("Testing check whitelist with an invalid regex whitelist")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = "/^user[0-9+$/"
 | 
			
		||||
	whitelist = "/^user[0-9+$/"
 | 
			
		||||
	str = "user1"
 | 
			
		||||
	expected = false
 | 
			
		||||
 | 
			
		||||
	// Test the check filter function
 | 
			
		||||
	result = utils.CheckFilter(filter, str)
 | 
			
		||||
	// Test the check whitelist function
 | 
			
		||||
	result = utils.CheckWhitelist(whitelist, str)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing check filter with a non matching list")
 | 
			
		||||
	t.Log("Testing check whitelist with a non matching whitelist")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = "user1,user2,user3"
 | 
			
		||||
	whitelist = "user1,user2,user3"
 | 
			
		||||
	str = "user4"
 | 
			
		||||
	expected = false
 | 
			
		||||
 | 
			
		||||
	// Test the check filter function
 | 
			
		||||
	result = utils.CheckFilter(filter, str)
 | 
			
		||||
	// Test the check whitelist function
 | 
			
		||||
	result = utils.CheckWhitelist(whitelist, str)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test capitalize
 | 
			
		||||
func TestCapitalize(t *testing.T) {
 | 
			
		||||
	t.Log("Testing capitalize with a valid string")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	str := "test"
 | 
			
		||||
	expected := "Test"
 | 
			
		||||
 | 
			
		||||
	// Test the capitalize function
 | 
			
		||||
	result := utils.Capitalize(str)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing capitalize with an empty string")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	str = ""
 | 
			
		||||
	expected = ""
 | 
			
		||||
 | 
			
		||||
	// Test the capitalize function
 | 
			
		||||
	result = utils.Capitalize(str)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
@@ -485,170 +535,3 @@ func TestSanitizeHeader(t *testing.T) {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test the parse headers function
 | 
			
		||||
func TestParseHeaders(t *testing.T) {
 | 
			
		||||
	t.Log("Testing parse headers with a valid string")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	headers := []string{"X-Hea\x00der1=value1", "X-Header2=value\n2"}
 | 
			
		||||
	expected := map[string]string{
 | 
			
		||||
		"X-Header1": "value1",
 | 
			
		||||
		"X-Header2": "value2",
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Test the parse headers function
 | 
			
		||||
	result := utils.ParseHeaders(headers)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if !reflect.DeepEqual(expected, result) {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing parse headers with an invalid string")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	headers = []string{"X-Header1=", "X-Header2", "=value", "X-Header3=value3"}
 | 
			
		||||
	expected = map[string]string{"X-Header3": "value3"}
 | 
			
		||||
 | 
			
		||||
	// Test the parse headers function
 | 
			
		||||
	result = utils.ParseHeaders(headers)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if !reflect.DeepEqual(expected, result) {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test the parse secret file function
 | 
			
		||||
func TestParseSecretFile(t *testing.T) {
 | 
			
		||||
	t.Log("Testing parse secret file with a valid file")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	content := "\n\n    \n\n\n  secret   \n\n    \n  "
 | 
			
		||||
	expected := "secret"
 | 
			
		||||
 | 
			
		||||
	// Test the parse secret file function
 | 
			
		||||
	result := utils.ParseSecretFile(content)
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test the filter IP function
 | 
			
		||||
func TestFilterIP(t *testing.T) {
 | 
			
		||||
	t.Log("Testing filter IP with an IP and a valid CIDR")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	ip := "10.10.10.10"
 | 
			
		||||
	filter := "10.10.10.0/24"
 | 
			
		||||
	expected := true
 | 
			
		||||
 | 
			
		||||
	// Test the filter IP function
 | 
			
		||||
	result, err := utils.FilterIP(filter, ip)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error filtering IP: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing filter IP with an IP and a valid IP")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = "10.10.10.10"
 | 
			
		||||
	expected = true
 | 
			
		||||
 | 
			
		||||
	// Test the filter IP function
 | 
			
		||||
	result, err = utils.FilterIP(filter, ip)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error filtering IP: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing filter IP with an IP and an non matching CIDR")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = "10.10.15.0/24"
 | 
			
		||||
	expected = false
 | 
			
		||||
 | 
			
		||||
	// Test the filter IP function
 | 
			
		||||
	result, err = utils.FilterIP(filter, ip)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error filtering IP: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing filter IP with a non matching IP and a valid CIDR")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = "10.10.10.11"
 | 
			
		||||
	expected = false
 | 
			
		||||
 | 
			
		||||
	// Test the filter IP function
 | 
			
		||||
	result, err = utils.FilterIP(filter, ip)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error filtering IP: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	t.Log("Testing filter IP with an IP and an invalid CIDR")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	filter = "10.../83"
 | 
			
		||||
 | 
			
		||||
	// Test the filter IP function
 | 
			
		||||
	_, err = utils.FilterIP(filter, ip)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err == nil {
 | 
			
		||||
		t.Fatalf("Expected error filtering IP")
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Test the derive key function
 | 
			
		||||
func TestDeriveKey(t *testing.T) {
 | 
			
		||||
	t.Log("Testing the derive key function")
 | 
			
		||||
 | 
			
		||||
	// Create variables
 | 
			
		||||
	master := "master"
 | 
			
		||||
	info := "info"
 | 
			
		||||
	expected := "gdrdU/fXzclYjiSXRexEatVgV13qQmKl"
 | 
			
		||||
 | 
			
		||||
	// Test the derive key function
 | 
			
		||||
	result, err := utils.DeriveKey(master, info)
 | 
			
		||||
 | 
			
		||||
	// Check if there was an error
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		t.Fatalf("Error deriving key: %v", err)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// Check if the result is equal to the expected
 | 
			
		||||
	if result != expected {
 | 
			
		||||
		t.Fatalf("Expected %v, got %v", expected, result)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user