mirror of
				https://github.com/steveiliop56/tinyauth.git
				synced 2025-11-03 23:55:44 +00:00 
			
		
		
		
	Compare commits
	
		
			9 Commits
		
	
	
		
			v0.2.0-bet
			...
			v0.3.0-bet
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					b901744e03 | ||
| 
						 | 
					61a7400cf1 | ||
| 
						 | 
					40ab77cdd5 | ||
| 
						 | 
					403787e56c | ||
| 
						 | 
					d3e52c925d | ||
| 
						 | 
					a4c717ba34 | ||
| 
						 | 
					5e73d06fcc | ||
| 
						 | 
					2988b5f22f | ||
| 
						 | 
					a28e55ae4c | 
@@ -46,4 +46,4 @@ COPY --from=builder /tinyauth/tinyauth ./
 | 
			
		||||
 | 
			
		||||
EXPOSE 3000
 | 
			
		||||
 | 
			
		||||
CMD ["./tinyauth"]
 | 
			
		||||
ENTRYPOINT ["./tinyauth"]
 | 
			
		||||
							
								
								
									
										48
									
								
								cmd/root.go
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								cmd/root.go
									
									
									
									
									
								
							@@ -3,14 +3,14 @@ package cmd
 | 
			
		||||
import (
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
	"time"
 | 
			
		||||
	cmd "tinyauth/cmd/user"
 | 
			
		||||
	"tinyauth/internal/api"
 | 
			
		||||
	"tinyauth/internal/assets"
 | 
			
		||||
	"tinyauth/internal/auth"
 | 
			
		||||
	"tinyauth/internal/hooks"
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
	"tinyauth/internal/utils"
 | 
			
		||||
 | 
			
		||||
	"github.com/go-playground/validator/v10"
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"github.com/spf13/viper"
 | 
			
		||||
@@ -21,10 +21,6 @@ var rootCmd = &cobra.Command{
 | 
			
		||||
	Short: "An extremely simple traefik forward auth proxy.",
 | 
			
		||||
	Long: `Tinyauth is an extremely simple traefik forward-auth login screen that makes securing your apps easy.`,
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		// Logger
 | 
			
		||||
		log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stderr, TimeFormat: time.RFC3339}).With().Timestamp().Logger()
 | 
			
		||||
		log.Info().Str("version", assets.Version).Msg("Starting tinyauth")
 | 
			
		||||
 | 
			
		||||
		// Get config
 | 
			
		||||
		log.Info().Msg("Parsing config")
 | 
			
		||||
		var config types.Config
 | 
			
		||||
@@ -45,32 +41,51 @@ var rootCmd = &cobra.Command{
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		users := config.Users
 | 
			
		||||
		usersString := config.Users
 | 
			
		||||
 | 
			
		||||
		if config.UsersFile != "" {
 | 
			
		||||
			log.Info().Msg("Reading users from file")
 | 
			
		||||
			usersFromFile, readErr := utils.GetUsersFromFile(config.UsersFile)
 | 
			
		||||
			HandleError(readErr, "Failed to read users from file")
 | 
			
		||||
			usersFromFileParsed := strings.Join(strings.Split(usersFromFile, "\n"), ",")
 | 
			
		||||
			if users != "" {
 | 
			
		||||
				users = users + "," + usersFromFileParsed
 | 
			
		||||
			if usersString != "" {
 | 
			
		||||
				usersString = usersString + "," + usersFromFileParsed
 | 
			
		||||
			} else {
 | 
			
		||||
				users = usersFromFileParsed
 | 
			
		||||
				usersString = usersFromFileParsed
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		userList, createErr := utils.ParseUsers(users)
 | 
			
		||||
		HandleError(createErr, "Failed to parse users")
 | 
			
		||||
		users, parseErr := utils.ParseUsers(usersString)
 | 
			
		||||
		HandleError(parseErr, "Failed to parse users")
 | 
			
		||||
 | 
			
		||||
		// Start server
 | 
			
		||||
		log.Info().Msg("Starting server")
 | 
			
		||||
		api.Run(config, userList)
 | 
			
		||||
		// Create auth service
 | 
			
		||||
		auth := auth.NewAuth(users)
 | 
			
		||||
		
 | 
			
		||||
		// Create hooks service
 | 
			
		||||
		hooks := hooks.NewHooks(auth)
 | 
			
		||||
		
 | 
			
		||||
		// Create API
 | 
			
		||||
		api := api.NewAPI(types.APIConfig{
 | 
			
		||||
			Port: config.Port,
 | 
			
		||||
			Address: config.Address,
 | 
			
		||||
			Secret: config.Secret,
 | 
			
		||||
			AppURL: config.AppURL,
 | 
			
		||||
			CookieSecure: config.CookieSecure,
 | 
			
		||||
		}, hooks, auth)
 | 
			
		||||
 | 
			
		||||
		// Setup routes
 | 
			
		||||
		api.Init()
 | 
			
		||||
		api.SetupRoutes()
 | 
			
		||||
 | 
			
		||||
		// Start
 | 
			
		||||
		api.Run()
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func Execute() {
 | 
			
		||||
	err := rootCmd.Execute()
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		log.Fatal().Err(err).Msg("Failed to execute command")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
@@ -83,6 +98,7 @@ func HandleError(err error, msg string) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	rootCmd.AddCommand(cmd.UserCmd())
 | 
			
		||||
	viper.AutomaticEnv()
 | 
			
		||||
	rootCmd.Flags().IntP("port", "p", 3000, "Port to run the server on.")
 | 
			
		||||
	rootCmd.Flags().String("address", "0.0.0.0", "Address to bind the server to.")
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										83
									
								
								cmd/user/create/create.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										83
									
								
								cmd/user/create/create.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,83 @@
 | 
			
		||||
package create
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"fmt"
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var interactive bool
 | 
			
		||||
var username string
 | 
			
		||||
var password string
 | 
			
		||||
var docker bool
 | 
			
		||||
 | 
			
		||||
var CreateCmd = &cobra.Command{
 | 
			
		||||
	Use:  "create",
 | 
			
		||||
	Short: "Create a user",
 | 
			
		||||
	Long: `Create a user either interactively or by passing flags.`,
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		if interactive {
 | 
			
		||||
			form := huh.NewForm(
 | 
			
		||||
				huh.NewGroup(
 | 
			
		||||
					huh.NewInput().Title("Username").Value(&username).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("username cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewInput().Title("Password").Value(&password).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("password cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewSelect[bool]().Title("Format the output for docker?").Options(huh.NewOption("Yes", true), huh.NewOption("No", false)).Value(&docker),
 | 
			
		||||
				),
 | 
			
		||||
			)
 | 
			
		||||
 | 
			
		||||
			var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
 | 
			
		||||
			formErr := form.WithTheme(baseTheme).Run()
 | 
			
		||||
 | 
			
		||||
			if formErr != nil {
 | 
			
		||||
				log.Fatal().Err(formErr).Msg("Form failed")
 | 
			
		||||
				os.Exit(1)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if username == "" || password == "" {
 | 
			
		||||
			log.Error().Msg("Username and password cannot be empty")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Info().Str("username", username).Str("password", password).Bool("docker", docker).Msg("Creating user")
 | 
			
		||||
 | 
			
		||||
		passwordByte, passwordErr := bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost)
 | 
			
		||||
 | 
			
		||||
		if passwordErr != nil {
 | 
			
		||||
			log.Fatal().Err(passwordErr).Msg("Failed to hash password")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		passwordString := string(passwordByte)
 | 
			
		||||
 | 
			
		||||
		if docker {
 | 
			
		||||
			passwordString = strings.ReplaceAll(passwordString, "$", "$$")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		log.Info().Str("user", fmt.Sprintf("%s:%s", username, passwordString)).Msg("User created")
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	CreateCmd.Flags().BoolVarP(&interactive, "interactive", "i", false, "Create a user interactively")
 | 
			
		||||
	CreateCmd.Flags().BoolVarP(&docker, "docker", "d", false, "Format output for docker")
 | 
			
		||||
	CreateCmd.Flags().StringVarP(&username, "username", "u", "", "Username")
 | 
			
		||||
	CreateCmd.Flags().StringVarP(&password, "password", "p", "", "Password")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										19
									
								
								cmd/user/user.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								cmd/user/user.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,19 @@
 | 
			
		||||
package cmd
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"tinyauth/cmd/user/create"
 | 
			
		||||
	"tinyauth/cmd/user/verify"
 | 
			
		||||
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func UserCmd() *cobra.Command {
 | 
			
		||||
	userCmd := &cobra.Command{
 | 
			
		||||
		Use:  "user",
 | 
			
		||||
		Short: "User utilities",
 | 
			
		||||
		Long: `Utilities for creating and verifying tinyauth compatible users.`,
 | 
			
		||||
	}
 | 
			
		||||
	userCmd.AddCommand(create.CreateCmd)
 | 
			
		||||
	userCmd.AddCommand(verify.VerifyCmd)
 | 
			
		||||
	return userCmd
 | 
			
		||||
}	
 | 
			
		||||
							
								
								
									
										96
									
								
								cmd/user/verify/verify.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								cmd/user/verify/verify.go
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,96 @@
 | 
			
		||||
package verify
 | 
			
		||||
 | 
			
		||||
import (
 | 
			
		||||
	"errors"
 | 
			
		||||
	"os"
 | 
			
		||||
	"strings"
 | 
			
		||||
 | 
			
		||||
	"github.com/charmbracelet/huh"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
	"github.com/spf13/cobra"
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
var interactive bool
 | 
			
		||||
var username string
 | 
			
		||||
var password string
 | 
			
		||||
var docker bool
 | 
			
		||||
var user string
 | 
			
		||||
 | 
			
		||||
var VerifyCmd = &cobra.Command{
 | 
			
		||||
	Use: "verify",
 | 
			
		||||
	Short: "Verify a user is set up correctly",
 | 
			
		||||
	Long: `Verify a user is set up correctly meaning that it has a correct password.`,
 | 
			
		||||
	Run: func(cmd *cobra.Command, args []string) {
 | 
			
		||||
		if interactive {
 | 
			
		||||
			form := huh.NewForm(
 | 
			
		||||
				huh.NewGroup(
 | 
			
		||||
					huh.NewInput().Title("User (user:hash)").Value(&user).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("user cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewInput().Title("Username").Value(&username).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("username cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewInput().Title("Password").Value(&password).Validate((func(s string) error {
 | 
			
		||||
						if s == "" {
 | 
			
		||||
							return errors.New("password cannot be empty")
 | 
			
		||||
						}
 | 
			
		||||
						return nil
 | 
			
		||||
					})),
 | 
			
		||||
					huh.NewSelect[bool]().Title("Is the user formatted for docker?").Options(huh.NewOption("Yes", true), huh.NewOption("No", false)).Value(&docker),
 | 
			
		||||
				),
 | 
			
		||||
			)
 | 
			
		||||
 | 
			
		||||
			var baseTheme *huh.Theme = huh.ThemeBase()
 | 
			
		||||
 | 
			
		||||
			formErr := form.WithTheme(baseTheme).Run()
 | 
			
		||||
 | 
			
		||||
			if formErr != nil {
 | 
			
		||||
				log.Fatal().Err(formErr).Msg("Form failed")
 | 
			
		||||
				os.Exit(1)
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if username == "" || password == "" || user == "" { 
 | 
			
		||||
			log.Error().Msg("Username, password and user cannot be empty")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
		log.Info().Str("user", user).Str("username", username).Str("password", password).Bool("docker", docker).Msg("Verifying user")
 | 
			
		||||
 | 
			
		||||
		userSplit := strings.Split(user, ":")
 | 
			
		||||
 | 
			
		||||
		if userSplit[1] == "" {
 | 
			
		||||
			log.Error().Msg("User is not formatted correctly")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if docker {
 | 
			
		||||
			userSplit[1] = strings.ReplaceAll(userSplit[1], "$$", "$")
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		verifyErr := bcrypt.CompareHashAndPassword([]byte(userSplit[1]), []byte(password))
 | 
			
		||||
 | 
			
		||||
		if verifyErr != nil || username != userSplit[0] {
 | 
			
		||||
			log.Error().Msg("Username or password incorrect")
 | 
			
		||||
			os.Exit(1)
 | 
			
		||||
		} else {
 | 
			
		||||
			log.Info().Msg("Verification successful")
 | 
			
		||||
		}
 | 
			
		||||
	},
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func init() {
 | 
			
		||||
	VerifyCmd.Flags().BoolVarP(&interactive, "interactive", "i", false, "Create a user interactively")
 | 
			
		||||
	VerifyCmd.Flags().BoolVar(&docker, "docker", false, "Is the user formatted for docker?")
 | 
			
		||||
	VerifyCmd.Flags().StringVar(&username, "username", "", "Username")
 | 
			
		||||
	VerifyCmd.Flags().StringVar(&password, "password", "", "Password")
 | 
			
		||||
	VerifyCmd.Flags().StringVar(&user, "user", "", "Hash (user:hash combination)")
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										33
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								go.mod
									
									
									
									
									
								
							@@ -3,23 +3,38 @@ module tinyauth
 | 
			
		||||
go 1.23.2
 | 
			
		||||
 | 
			
		||||
require (
 | 
			
		||||
	github.com/gin-contrib/sessions v1.0.2
 | 
			
		||||
	github.com/gin-gonic/gin v1.10.0
 | 
			
		||||
	github.com/go-playground/validator/v10 v10.24.0
 | 
			
		||||
	github.com/google/go-querystring v1.1.0
 | 
			
		||||
	github.com/rs/zerolog v1.33.0
 | 
			
		||||
	github.com/spf13/cobra v1.8.1
 | 
			
		||||
	github.com/spf13/viper v1.19.0
 | 
			
		||||
	golang.org/x/crypto v0.32.0
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
require (
 | 
			
		||||
	github.com/atotto/clipboard v0.1.4 // indirect
 | 
			
		||||
	github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect
 | 
			
		||||
	github.com/bytedance/sonic v1.12.7 // indirect
 | 
			
		||||
	github.com/bytedance/sonic/loader v0.2.3 // indirect
 | 
			
		||||
	github.com/catppuccin/go v0.2.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/bubbles v0.20.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/bubbletea v1.1.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/huh v0.6.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/lipgloss v0.13.0 // indirect
 | 
			
		||||
	github.com/charmbracelet/x/ansi v0.2.3 // indirect
 | 
			
		||||
	github.com/charmbracelet/x/exp/strings v0.0.0-20240722160745-212f7b056ed0 // indirect
 | 
			
		||||
	github.com/charmbracelet/x/term v0.2.0 // indirect
 | 
			
		||||
	github.com/cloudwego/base64x v0.1.4 // indirect
 | 
			
		||||
	github.com/dustin/go-humanize v1.0.1 // indirect
 | 
			
		||||
	github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f // indirect
 | 
			
		||||
	github.com/fsnotify/fsnotify v1.7.0 // indirect
 | 
			
		||||
	github.com/gabriel-vasile/mimetype v1.4.8 // indirect
 | 
			
		||||
	github.com/gin-contrib/sessions v1.0.2 // indirect
 | 
			
		||||
	github.com/gin-contrib/sse v1.0.0 // indirect
 | 
			
		||||
	github.com/go-playground/locales v0.14.1 // indirect
 | 
			
		||||
	github.com/go-playground/universal-translator v0.18.1 // indirect
 | 
			
		||||
	github.com/go-playground/validator/v10 v10.24.0 // indirect
 | 
			
		||||
	github.com/goccy/go-json v0.10.4 // indirect
 | 
			
		||||
	github.com/google/go-querystring v1.1.0 // indirect
 | 
			
		||||
	github.com/gorilla/context v1.1.2 // indirect
 | 
			
		||||
	github.com/gorilla/securecookie v1.1.2 // indirect
 | 
			
		||||
	github.com/gorilla/sessions v1.2.2 // indirect
 | 
			
		||||
@@ -28,30 +43,36 @@ require (
 | 
			
		||||
	github.com/json-iterator/go v1.1.12 // indirect
 | 
			
		||||
	github.com/klauspost/cpuid/v2 v2.2.9 // indirect
 | 
			
		||||
	github.com/leodido/go-urn v1.4.0 // indirect
 | 
			
		||||
	github.com/lucasb-eyer/go-colorful v1.2.0 // indirect
 | 
			
		||||
	github.com/magiconair/properties v1.8.7 // indirect
 | 
			
		||||
	github.com/mattn/go-colorable v0.1.14 // indirect
 | 
			
		||||
	github.com/mattn/go-isatty v0.0.20 // indirect
 | 
			
		||||
	github.com/mattn/go-localereader v0.0.1 // indirect
 | 
			
		||||
	github.com/mattn/go-runewidth v0.0.16 // indirect
 | 
			
		||||
	github.com/mitchellh/hashstructure/v2 v2.0.2 // indirect
 | 
			
		||||
	github.com/mitchellh/mapstructure v1.5.0 // indirect
 | 
			
		||||
	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
 | 
			
		||||
	github.com/modern-go/reflect2 v1.0.2 // indirect
 | 
			
		||||
	github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 // indirect
 | 
			
		||||
	github.com/muesli/cancelreader v0.2.2 // indirect
 | 
			
		||||
	github.com/muesli/termenv v0.15.3-0.20240618155329-98d742f6907a // indirect
 | 
			
		||||
	github.com/pelletier/go-toml/v2 v2.2.3 // indirect
 | 
			
		||||
	github.com/rs/zerolog v1.33.0 // indirect
 | 
			
		||||
	github.com/rivo/uniseg v0.4.7 // indirect
 | 
			
		||||
	github.com/sagikazarmark/locafero v0.4.0 // indirect
 | 
			
		||||
	github.com/sagikazarmark/slog-shim v0.1.0 // indirect
 | 
			
		||||
	github.com/sourcegraph/conc v0.3.0 // indirect
 | 
			
		||||
	github.com/spf13/afero v1.11.0 // indirect
 | 
			
		||||
	github.com/spf13/cast v1.6.0 // indirect
 | 
			
		||||
	github.com/spf13/pflag v1.0.5 // indirect
 | 
			
		||||
	github.com/spf13/viper v1.19.0 // indirect
 | 
			
		||||
	github.com/subosito/gotenv v1.6.0 // indirect
 | 
			
		||||
	github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
 | 
			
		||||
	github.com/ugorji/go/codec v1.2.12 // indirect
 | 
			
		||||
	go.uber.org/atomic v1.9.0 // indirect
 | 
			
		||||
	go.uber.org/multierr v1.9.0 // indirect
 | 
			
		||||
	golang.org/x/arch v0.13.0 // indirect
 | 
			
		||||
	golang.org/x/crypto v0.32.0 // indirect
 | 
			
		||||
	golang.org/x/exp v0.0.0-20230905200255-921286631fa9 // indirect
 | 
			
		||||
	golang.org/x/net v0.34.0 // indirect
 | 
			
		||||
	golang.org/x/sync v0.10.0 // indirect
 | 
			
		||||
	golang.org/x/sys v0.29.0 // indirect
 | 
			
		||||
	golang.org/x/text v0.21.0 // indirect
 | 
			
		||||
	google.golang.org/protobuf v1.36.3 // indirect
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										68
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										68
									
								
								go.sum
									
									
									
									
									
								
							@@ -1,16 +1,43 @@
 | 
			
		||||
github.com/atotto/clipboard v0.1.4 h1:EH0zSVneZPSuFR11BlR9YppQTVDbh5+16AmcJi4g1z4=
 | 
			
		||||
github.com/atotto/clipboard v0.1.4/go.mod h1:ZY9tmq7sm5xIbd9bOK4onWV4S6X0u6GY7Vn0Yu86PYI=
 | 
			
		||||
github.com/aymanbagabas/go-osc52/v2 v2.0.1 h1:HwpRHbFMcZLEVr42D4p7XBqjyuxQH5SMiErDT4WkJ2k=
 | 
			
		||||
github.com/aymanbagabas/go-osc52/v2 v2.0.1/go.mod h1:uYgXzlJ7ZpABp8OJ+exZzJJhRNQ2ASbcXHWsFqH8hp8=
 | 
			
		||||
github.com/bytedance/sonic v1.12.7 h1:CQU8pxOy9HToxhndH0Kx/S1qU/CuS9GnKYrGioDcU1Q=
 | 
			
		||||
github.com/bytedance/sonic v1.12.7/go.mod h1:tnbal4mxOMju17EGfknm2XyYcpyCnIROYOEYuemj13I=
 | 
			
		||||
github.com/bytedance/sonic/loader v0.1.1/go.mod h1:ncP89zfokxS5LZrJxl5z0UJcsk4M4yY2JpfqGeCtNLU=
 | 
			
		||||
github.com/bytedance/sonic/loader v0.2.3 h1:yctD0Q3v2NOGfSWPLPvG2ggA2kV6TS6s4wioyEqssH0=
 | 
			
		||||
github.com/bytedance/sonic/loader v0.2.3/go.mod h1:N8A3vUdtUebEY2/VQC0MyhYeKUFosQU6FxH2JmUe6VI=
 | 
			
		||||
github.com/catppuccin/go v0.2.0 h1:ktBeIrIP42b/8FGiScP9sgrWOss3lw0Z5SktRoithGA=
 | 
			
		||||
github.com/catppuccin/go v0.2.0/go.mod h1:8IHJuMGaUUjQM82qBrGNBv7LFq6JI3NnQCF6MOlZjpc=
 | 
			
		||||
github.com/charmbracelet/bubbles v0.20.0 h1:jSZu6qD8cRQ6k9OMfR1WlM+ruM8fkPWkHvQWD9LIutE=
 | 
			
		||||
github.com/charmbracelet/bubbles v0.20.0/go.mod h1:39slydyswPy+uVOHZ5x/GjwVAFkCsV8IIVy+4MhzwwU=
 | 
			
		||||
github.com/charmbracelet/bubbletea v1.1.0 h1:FjAl9eAL3HBCHenhz/ZPjkKdScmaS5SK69JAK2YJK9c=
 | 
			
		||||
github.com/charmbracelet/bubbletea v1.1.0/go.mod h1:9Ogk0HrdbHolIKHdjfFpyXJmiCzGwy+FesYkZr7hYU4=
 | 
			
		||||
github.com/charmbracelet/huh v0.6.0 h1:mZM8VvZGuE0hoDXq6XLxRtgfWyTI3b2jZNKh0xWmax8=
 | 
			
		||||
github.com/charmbracelet/huh v0.6.0/go.mod h1:GGNKeWCeNzKpEOh/OJD8WBwTQjV3prFAtQPpLv+AVwU=
 | 
			
		||||
github.com/charmbracelet/lipgloss v0.13.0 h1:4X3PPeoWEDCMvzDvGmTajSyYPcZM4+y8sCA/SsA3cjw=
 | 
			
		||||
github.com/charmbracelet/lipgloss v0.13.0/go.mod h1:nw4zy0SBX/F/eAO1cWdcvy6qnkDUxr8Lw7dvFrAIbbY=
 | 
			
		||||
github.com/charmbracelet/x/ansi v0.2.3 h1:VfFN0NUpcjBRd4DnKfRaIRo53KRgey/nhOoEqosGDEY=
 | 
			
		||||
github.com/charmbracelet/x/ansi v0.2.3/go.mod h1:dk73KoMTT5AX5BsX0KrqhsTqAnhZZoCBjs7dGWp4Ktw=
 | 
			
		||||
github.com/charmbracelet/x/exp/strings v0.0.0-20240722160745-212f7b056ed0 h1:qko3AQ4gK1MTS/de7F5hPGx6/k1u0w4TeYmBFwzYVP4=
 | 
			
		||||
github.com/charmbracelet/x/exp/strings v0.0.0-20240722160745-212f7b056ed0/go.mod h1:pBhA0ybfXv6hDjQUZ7hk1lVxBiUbupdw5R31yPUViVQ=
 | 
			
		||||
github.com/charmbracelet/x/term v0.2.0 h1:cNB9Ot9q8I711MyZ7myUR5HFWL/lc3OpU8jZ4hwm0x0=
 | 
			
		||||
github.com/charmbracelet/x/term v0.2.0/go.mod h1:GVxgxAbjUrmpvIINHIQnJJKpMlHiZ4cktEQCN6GWyF0=
 | 
			
		||||
github.com/cloudwego/base64x v0.1.4 h1:jwCgWpFanWmN8xoIUHa2rtzmkd5J2plF/dnLS6Xd/0Y=
 | 
			
		||||
github.com/cloudwego/base64x v0.1.4/go.mod h1:0zlkT4Wn5C6NdauXdJRhSKRlJvmclQ1hhJgA0rcu/8w=
 | 
			
		||||
github.com/cloudwego/iasm v0.2.0/go.mod h1:8rXZaNYT2n95jn+zTI1sDr+IgcD2GVs0nlbbQPiEFhY=
 | 
			
		||||
github.com/coreos/go-systemd/v22 v22.5.0/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc=
 | 
			
		||||
github.com/cpuguy83/go-md2man/v2 v2.0.4/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
 | 
			
		||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
 | 
			
		||||
github.com/dustin/go-humanize v1.0.1 h1:GzkhY7T5VNhEkwH0PVJgjz+fX1rhBrR7pRT3mDkpeCY=
 | 
			
		||||
github.com/dustin/go-humanize v1.0.1/go.mod h1:Mu1zIs6XwVuF/gI1OepvI0qD18qycQx+mFykh5fBlto=
 | 
			
		||||
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f h1:Y/CXytFA4m6baUTXGLOoWe4PQhGxaX0KpnayAqC48p4=
 | 
			
		||||
github.com/erikgeiser/coninput v0.0.0-20211004153227-1c3628e74d0f/go.mod h1:vw97MGsxSvLiUE2X8qFplwetxpGLQrlU1Q9AUEIzCaM=
 | 
			
		||||
github.com/frankban/quicktest v1.14.6 h1:7Xjx+VpznH+oBnejlPUj8oUpdxnVs4f8XU8WnHkI4W8=
 | 
			
		||||
github.com/frankban/quicktest v1.14.6/go.mod h1:4ptaffx2x8+WTWXmUCuVU6aPUX1/Mz7zb5vbUoiM6w0=
 | 
			
		||||
github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA=
 | 
			
		||||
github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM=
 | 
			
		||||
github.com/gabriel-vasile/mimetype v1.4.8 h1:FfZ3gj38NjllZIeJAmMhr+qKL8Wu+nOoI3GqacKw1NM=
 | 
			
		||||
@@ -33,11 +60,13 @@ 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=
 | 
			
		||||
github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
 | 
			
		||||
github.com/google/go-cmp v0.5.5 h1:Khx7svrCpmxxtHBq5j2mp/xVjsi8hQMfNLvJFAlrGgU=
 | 
			
		||||
github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
 | 
			
		||||
github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38=
 | 
			
		||||
github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
 | 
			
		||||
github.com/google/go-querystring v1.1.0 h1:AnCroh3fv4ZBgVIf1Iwtovgjaw/GiKJo8M8yD/fhyJ8=
 | 
			
		||||
github.com/google/go-querystring v1.1.0/go.mod h1:Kcdr2DB4koayq7X8pmAG4sNG59So17icRSOU623lUBU=
 | 
			
		||||
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
 | 
			
		||||
github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0=
 | 
			
		||||
github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
 | 
			
		||||
github.com/gorilla/context v1.1.2 h1:WRkNAv2uoa03QNIc1A6u4O7DAGMUVoopZhkiXWA2V1o=
 | 
			
		||||
github.com/gorilla/context v1.1.2/go.mod h1:KDPwT9i/MeWHiLl90fuTgrt4/wPcv75vFAZLaOOcbxM=
 | 
			
		||||
github.com/gorilla/securecookie v1.1.2 h1:YCIWL56dvtr73r6715mJs5ZvhtnY73hBvEF8kXD8ePA=
 | 
			
		||||
@@ -54,8 +83,14 @@ github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa02
 | 
			
		||||
github.com/klauspost/cpuid/v2 v2.2.9 h1:66ze0taIn2H33fBvCkXuv9BmCwDfafmiIVpKV9kKGuY=
 | 
			
		||||
github.com/klauspost/cpuid/v2 v2.2.9/go.mod h1:rqkxqrZ1EhYM9G+hXH7YdowN5R5RGN6NK4QwQ3WMXF8=
 | 
			
		||||
github.com/knz/go-libedit v1.10.1/go.mod h1:MZTVkCWyz0oBc7JOWP3wNAzd002ZbM/5hgShxwh4x8M=
 | 
			
		||||
github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE=
 | 
			
		||||
github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk=
 | 
			
		||||
github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY=
 | 
			
		||||
github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE=
 | 
			
		||||
github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ=
 | 
			
		||||
github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI=
 | 
			
		||||
github.com/lucasb-eyer/go-colorful v1.2.0 h1:1nnpGOrhyZZuNyfu1QjKiUICQ74+3FNCN69Aj6K7nkY=
 | 
			
		||||
github.com/lucasb-eyer/go-colorful v1.2.0/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0=
 | 
			
		||||
github.com/magiconair/properties v1.8.7 h1:IeQXZAiQcpL9mgcAe1Nu6cX9LLw6ExEHKjN0VQdvPDY=
 | 
			
		||||
github.com/magiconair/properties v1.8.7/go.mod h1:Dhd985XPs7jluiymwWYZ0G4Z61jb3vdS329zhj2hYo0=
 | 
			
		||||
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
 | 
			
		||||
@@ -65,6 +100,12 @@ github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/
 | 
			
		||||
github.com/mattn/go-isatty v0.0.19/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
 | 
			
		||||
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
 | 
			
		||||
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
 | 
			
		||||
github.com/mattn/go-localereader v0.0.1 h1:ygSAOl7ZXTx4RdPYinUpg6W99U8jWvWi9Ye2JC/oIi4=
 | 
			
		||||
github.com/mattn/go-localereader v0.0.1/go.mod h1:8fBrzywKY7BI3czFoHkuzRoWE9C+EiG4R1k4Cjx5p88=
 | 
			
		||||
github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc=
 | 
			
		||||
github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
 | 
			
		||||
github.com/mitchellh/hashstructure/v2 v2.0.2 h1:vGKWl0YJqUNxE8d+h8f6NJLcCJrgbhC4NcD46KavDd4=
 | 
			
		||||
github.com/mitchellh/hashstructure/v2 v2.0.2/go.mod h1:MG3aRVU/N29oo/V/IhBX8GR/zz4kQkprJgF2EVszyDE=
 | 
			
		||||
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
 | 
			
		||||
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
 | 
			
		||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
 | 
			
		||||
@@ -72,11 +113,23 @@ github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w
 | 
			
		||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
 | 
			
		||||
github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M=
 | 
			
		||||
github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk=
 | 
			
		||||
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6 h1:ZK8zHtRHOkbHy6Mmr5D264iyp3TiX5OmNcI5cIARiQI=
 | 
			
		||||
github.com/muesli/ansi v0.0.0-20230316100256-276c6243b2f6/go.mod h1:CJlz5H+gyd6CUWT45Oy4q24RdLyn7Md9Vj2/ldJBSIo=
 | 
			
		||||
github.com/muesli/cancelreader v0.2.2 h1:3I4Kt4BQjOR54NavqnDogx/MIoWBFa0StPA8ELUXHmA=
 | 
			
		||||
github.com/muesli/cancelreader v0.2.2/go.mod h1:3XuTXfFS2VjM+HTLZY9Ak0l6eUKfijIfMUZ4EgX0QYo=
 | 
			
		||||
github.com/muesli/termenv v0.15.3-0.20240618155329-98d742f6907a h1:2MaM6YC3mGu54x+RKAA6JiFFHlHDY1UbkxqppT7wYOg=
 | 
			
		||||
github.com/muesli/termenv v0.15.3-0.20240618155329-98d742f6907a/go.mod h1:hxSnBBYLK21Vtq/PHd0S2FYCxBXzBua8ov5s1RobyRQ=
 | 
			
		||||
github.com/pelletier/go-toml/v2 v2.2.3 h1:YmeHyLY8mFWbdkNWwpr+qIL2bEqT0o95WSdkNHvL12M=
 | 
			
		||||
github.com/pelletier/go-toml/v2 v2.2.3/go.mod h1:MfCQTFTvCcUyyvvwm1+G6H/jORL20Xlb6rzQu9GuUkc=
 | 
			
		||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 h1:Jamvg5psRIccs7FGNTlIRMkT8wgtp5eCXdBlqhYGL6U=
 | 
			
		||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
 | 
			
		||||
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
 | 
			
		||||
github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ=
 | 
			
		||||
github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
 | 
			
		||||
github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8=
 | 
			
		||||
github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs=
 | 
			
		||||
github.com/rs/xid v1.5.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg=
 | 
			
		||||
github.com/rs/zerolog v1.33.0 h1:1cU2KZkvPxNyfgEmhHAz/1A9Bz+llsdYzklWFzgp0r8=
 | 
			
		||||
github.com/rs/zerolog v1.33.0/go.mod h1:/7mN4D5sKwJLZQ2b/znpjC3/GQWY/xaDXUM0kKWRHss=
 | 
			
		||||
@@ -127,6 +180,9 @@ golang.org/x/exp v0.0.0-20230905200255-921286631fa9 h1:GoHiUyI/Tp2nVkLI2mCxVkOjs
 | 
			
		||||
golang.org/x/exp v0.0.0-20230905200255-921286631fa9/go.mod h1:S2oDrQGGwySpoQPVqRShND87VCbxmc6bL1Yd2oYrm6k=
 | 
			
		||||
golang.org/x/net v0.34.0 h1:Mb7Mrk043xzHgnRM88suvJFwzVrRfHEHJEl5/71CKw0=
 | 
			
		||||
golang.org/x/net v0.34.0/go.mod h1:di0qlW3YNM5oh6GqDGQr92MyTozJPmybPK4Ev/Gm31k=
 | 
			
		||||
golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ=
 | 
			
		||||
golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
 | 
			
		||||
golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
golang.org/x/sys v0.12.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
 | 
			
		||||
@@ -134,12 +190,12 @@ golang.org/x/sys v0.29.0 h1:TPYlXGxvx1MGTn2GiZDhnjPA9wZzZeGKHHmKhHYvgaU=
 | 
			
		||||
golang.org/x/sys v0.29.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
 | 
			
		||||
golang.org/x/text v0.21.0 h1:zyQAAkrwaneQ066sspRyJaG9VNi/YJ1NfzcGB3hZ/qo=
 | 
			
		||||
golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4=
 | 
			
		||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
 | 
			
		||||
google.golang.org/protobuf v1.36.3 h1:82DV7MYdb8anAVi3qge1wSnMDrnKK7ebr+I0hHRN1BU=
 | 
			
		||||
google.golang.org/protobuf v1.36.3/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE=
 | 
			
		||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM=
 | 
			
		||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo=
 | 
			
		||||
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
 | 
			
		||||
gopkg.in/ini.v1 v1.67.0 h1:Dgnx+6+nfE+IfzjUEISNeydPJh9AXNNsWbGP9KzCsOA=
 | 
			
		||||
gopkg.in/ini.v1 v1.67.0/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k=
 | 
			
		||||
gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
 | 
			
		||||
 
 | 
			
		||||
@@ -20,8 +20,25 @@ import (
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func Run(config types.Config, users types.UserList) {
 | 
			
		||||
func NewAPI(config types.APIConfig, hooks *hooks.Hooks, auth *auth.Auth) (*API) {
 | 
			
		||||
	return &API{
 | 
			
		||||
		Config: config,
 | 
			
		||||
		Hooks: hooks,
 | 
			
		||||
		Auth: auth,
 | 
			
		||||
		Router: nil,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type API struct {
 | 
			
		||||
	Config types.APIConfig
 | 
			
		||||
	Router *gin.Engine
 | 
			
		||||
	Hooks *hooks.Hooks
 | 
			
		||||
	Auth *auth.Auth
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (api *API) Init() {
 | 
			
		||||
	gin.SetMode(gin.ReleaseMode)
 | 
			
		||||
	
 | 
			
		||||
	router := gin.New()
 | 
			
		||||
	router.Use(zerolog())
 | 
			
		||||
	dist, distErr := fs.Sub(assets.Assets, "dist")
 | 
			
		||||
@@ -32,22 +49,35 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	fileServer := http.FileServer(http.FS(dist))
 | 
			
		||||
	store := cookie.NewStore([]byte(config.Secret))
 | 
			
		||||
	store := cookie.NewStore([]byte(api.Config.Secret))
 | 
			
		||||
 | 
			
		||||
	domain, domainErr := utils.GetRootURL(config.AppURL)
 | 
			
		||||
	domain, domainErr := utils.GetRootURL(api.Config.AppURL)
 | 
			
		||||
 | 
			
		||||
	log.Info().Str("domain", domain).Msg("Using domain for cookies")
 | 
			
		||||
 | 
			
		||||
	if domainErr != nil {
 | 
			
		||||
		log.Fatal().Err(domainErr).Msg("Failed to get domain")
 | 
			
		||||
		os.Exit(1)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	var isSecure bool
 | 
			
		||||
 | 
			
		||||
	if api.Config.CookieSecure {
 | 
			
		||||
		isSecure = true
 | 
			
		||||
	} else {
 | 
			
		||||
		isSecure = false
 | 
			
		||||
	}
 | 
			
		||||
	
 | 
			
		||||
	store.Options(sessions.Options{
 | 
			
		||||
		Domain: fmt.Sprintf(".%s", domain),
 | 
			
		||||
		Path: "/",
 | 
			
		||||
		HttpOnly: true,
 | 
			
		||||
		Secure: isSecure,
 | 
			
		||||
	})
 | 
			
		||||
	
 | 
			
		||||
  	router.Use(sessions.Sessions("tinyauth", store))
 | 
			
		||||
 | 
			
		||||
	router.Use(func(c *gin.Context) {
 | 
			
		||||
	  router.Use(func(c *gin.Context) {
 | 
			
		||||
		if !strings.HasPrefix(c.Request.URL.Path, "/api") {
 | 
			
		||||
			_, err := fs.Stat(dist, strings.TrimPrefix(c.Request.URL.Path, "/"))
 | 
			
		||||
			if os.IsNotExist(err) {
 | 
			
		||||
@@ -58,8 +88,12 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
		}
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	router.GET("/api/auth", func (c *gin.Context) {
 | 
			
		||||
		userContext := hooks.UseUserContext(c, users)
 | 
			
		||||
	api.Router = router
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (api *API) SetupRoutes() {
 | 
			
		||||
	api.Router.GET("/api/auth", func (c *gin.Context) {
 | 
			
		||||
		userContext := api.Hooks.UseUserContext(c)
 | 
			
		||||
 | 
			
		||||
		if userContext.IsLoggedIn {
 | 
			
		||||
			c.JSON(200, gin.H{
 | 
			
		||||
@@ -84,10 +118,10 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
			return
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		c.Redirect(http.StatusTemporaryRedirect, fmt.Sprintf("%s/?%s", config.AppURL, queries.Encode()))
 | 
			
		||||
		c.Redirect(http.StatusTemporaryRedirect, fmt.Sprintf("%s/?%s", api.Config.AppURL, queries.Encode()))
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	router.POST("/api/login", func (c *gin.Context) {
 | 
			
		||||
	api.Router.POST("/api/login", func (c *gin.Context) {
 | 
			
		||||
		var login types.LoginRequest
 | 
			
		||||
 | 
			
		||||
		err := c.BindJSON(&login)
 | 
			
		||||
@@ -100,7 +134,7 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
			return
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		user := auth.FindUser(users, login.Username)
 | 
			
		||||
		user := api.Auth.GetUser(login.Username)
 | 
			
		||||
 | 
			
		||||
		if user == nil {
 | 
			
		||||
			c.JSON(401, gin.H{
 | 
			
		||||
@@ -110,7 +144,7 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
			return
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if !auth.CheckPassword(*user, login.Password) {
 | 
			
		||||
		if !api.Auth.CheckPassword(*user, login.Password) {
 | 
			
		||||
			c.JSON(401, gin.H{
 | 
			
		||||
				"status": 401,
 | 
			
		||||
				"message": "Unauthorized",
 | 
			
		||||
@@ -128,7 +162,7 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	router.POST("/api/logout", func (c *gin.Context) {
 | 
			
		||||
	api.Router.POST("/api/logout", func (c *gin.Context) {
 | 
			
		||||
		session := sessions.Default(c)
 | 
			
		||||
		session.Delete("tinyauth")
 | 
			
		||||
		session.Save()
 | 
			
		||||
@@ -139,8 +173,8 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	router.GET("/api/status", func (c *gin.Context) {
 | 
			
		||||
		userContext := hooks.UseUserContext(c, users)
 | 
			
		||||
	api.Router.GET("/api/status", func (c *gin.Context) {
 | 
			
		||||
		userContext := api.Hooks.UseUserContext(c)
 | 
			
		||||
 | 
			
		||||
		if !userContext.IsLoggedIn {
 | 
			
		||||
			c.JSON(200, gin.H{
 | 
			
		||||
@@ -160,14 +194,18 @@ func Run(config types.Config, users types.UserList) {
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
 | 
			
		||||
	router.GET("/api/healthcheck", func (c *gin.Context) {
 | 
			
		||||
	api.Router.GET("/api/healthcheck", func (c *gin.Context) {
 | 
			
		||||
		c.JSON(200, gin.H{
 | 
			
		||||
			"status": 200,
 | 
			
		||||
			"message": "OK",
 | 
			
		||||
		})
 | 
			
		||||
	})
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
	router.Run(fmt.Sprintf("%s:%d", config.Address, config.Port))
 | 
			
		||||
 | 
			
		||||
func (api *API) Run() {
 | 
			
		||||
	log.Info().Str("address", api.Config.Address).Int("port", api.Config.Port).Msg("Starting server")
 | 
			
		||||
	api.Router.Run(fmt.Sprintf("%s:%d", api.Config.Address, api.Config.Port))
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func zerolog() gin.HandlerFunc {
 | 
			
		||||
 
 | 
			
		||||
@@ -1 +1 @@
 | 
			
		||||
v0.2.0
 | 
			
		||||
v0.3.0
 | 
			
		||||
@@ -6,8 +6,18 @@ import (
 | 
			
		||||
	"golang.org/x/crypto/bcrypt"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func FindUser(userList types.UserList, username string) (*types.User) {
 | 
			
		||||
	for _, user := range userList.Users {
 | 
			
		||||
func NewAuth(userList types.Users) *Auth {
 | 
			
		||||
	return &Auth{
 | 
			
		||||
		Users: userList,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Auth struct {
 | 
			
		||||
	Users types.Users
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (auth *Auth) GetUser(username string) *types.User {
 | 
			
		||||
	for _, user := range auth.Users {
 | 
			
		||||
		if user.Username == username {
 | 
			
		||||
			return &user
 | 
			
		||||
		}
 | 
			
		||||
@@ -15,7 +25,7 @@ func FindUser(userList types.UserList, username string) (*types.User) {
 | 
			
		||||
	return nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func CheckPassword(user types.User, password string) bool {
 | 
			
		||||
func (auth *Auth) CheckPassword(user types.User, password string) bool {
 | 
			
		||||
	hashedPasswordErr := bcrypt.CompareHashAndPassword([]byte(user.Password), []byte(password))
 | 
			
		||||
	return hashedPasswordErr == nil
 | 
			
		||||
}
 | 
			
		||||
@@ -8,7 +8,17 @@ import (
 | 
			
		||||
	"github.com/gin-gonic/gin"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func UseUserContext(c *gin.Context, userList types.UserList) (types.UserContext) {
 | 
			
		||||
func NewHooks(auth *auth.Auth) *Hooks {
 | 
			
		||||
	return &Hooks{
 | 
			
		||||
		Auth: auth,
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type Hooks struct {
 | 
			
		||||
	Auth *auth.Auth
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func (hooks *Hooks) UseUserContext(c *gin.Context) (types.UserContext) {
 | 
			
		||||
	session := sessions.Default(c)
 | 
			
		||||
	cookie := session.Get("tinyauth")
 | 
			
		||||
 | 
			
		||||
@@ -28,7 +38,7 @@ func UseUserContext(c *gin.Context, userList types.UserList) (types.UserContext)
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	user := auth.FindUser(userList, username)
 | 
			
		||||
	user := hooks.Auth.GetUser(username)
 | 
			
		||||
 | 
			
		||||
	if user == nil {
 | 
			
		||||
		return types.UserContext{
 | 
			
		||||
 
 | 
			
		||||
@@ -14,9 +14,7 @@ type User struct {
 | 
			
		||||
	Password string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type UserList struct {
 | 
			
		||||
	Users []User
 | 
			
		||||
}
 | 
			
		||||
type Users []User
 | 
			
		||||
 | 
			
		||||
type Config struct {
 | 
			
		||||
	Port int `validate:"number" mapstructure:"port"`
 | 
			
		||||
@@ -25,9 +23,18 @@ type Config struct {
 | 
			
		||||
	AppURL string `validate:"required,url" mapstructure:"app-url"`
 | 
			
		||||
	Users string `mapstructure:"users"`
 | 
			
		||||
	UsersFile string `mapstructure:"users-file"`
 | 
			
		||||
	CookieSecure bool `mapstructure:"cookie-secure"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type UserContext struct {
 | 
			
		||||
	Username string
 | 
			
		||||
	IsLoggedIn bool
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
type APIConfig struct {
 | 
			
		||||
	Port int
 | 
			
		||||
	Address string
 | 
			
		||||
	Secret string
 | 
			
		||||
	AppURL string
 | 
			
		||||
	CookieSecure bool
 | 
			
		||||
}
 | 
			
		||||
@@ -8,26 +8,26 @@ import (
 | 
			
		||||
	"tinyauth/internal/types"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func ParseUsers(users string) (types.UserList, error) {
 | 
			
		||||
	var userList types.UserList
 | 
			
		||||
	userListString := strings.Split(users, ",")
 | 
			
		||||
func ParseUsers(users string) (types.Users, error) {
 | 
			
		||||
	var usersParsed types.Users
 | 
			
		||||
	userList := strings.Split(users, ",")
 | 
			
		||||
 | 
			
		||||
	if len(userListString) == 0 {
 | 
			
		||||
		return types.UserList{}, errors.New("invalid user format")
 | 
			
		||||
	if len(userList) == 0 {
 | 
			
		||||
		return types.Users{}, errors.New("invalid user format")
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for _, user := range userListString {
 | 
			
		||||
	for _, user := range userList {
 | 
			
		||||
		userSplit := strings.Split(user, ":")
 | 
			
		||||
		if len(userSplit) != 2 {
 | 
			
		||||
			return types.UserList{}, errors.New("invalid user format")
 | 
			
		||||
			return types.Users{}, errors.New("invalid user format")
 | 
			
		||||
		}
 | 
			
		||||
		userList.Users = append(userList.Users, types.User{
 | 
			
		||||
		usersParsed = append(usersParsed, types.User{
 | 
			
		||||
			Username: userSplit[0],
 | 
			
		||||
			Password: userSplit[1],
 | 
			
		||||
		})
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return userList, nil
 | 
			
		||||
	return usersParsed, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
func GetRootURL(urlSrc string) (string, error) {
 | 
			
		||||
@@ -39,7 +39,7 @@ func GetRootURL(urlSrc string) (string, error) {
 | 
			
		||||
 | 
			
		||||
	urlSplitted := strings.Split(urlParsed.Host, ".")
 | 
			
		||||
 | 
			
		||||
	urlFinal := urlSplitted[len(urlSplitted)-2] + "." + urlSplitted[len(urlSplitted)-1]
 | 
			
		||||
	urlFinal := strings.Join(urlSplitted[1:], ".")
 | 
			
		||||
 | 
			
		||||
	return urlFinal, nil
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										15
									
								
								main.go
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								main.go
									
									
									
									
									
								
							@@ -1,7 +1,20 @@
 | 
			
		||||
package main
 | 
			
		||||
 | 
			
		||||
import "tinyauth/cmd"
 | 
			
		||||
import (
 | 
			
		||||
	"os"
 | 
			
		||||
	"time"
 | 
			
		||||
	"tinyauth/cmd"
 | 
			
		||||
	"tinyauth/internal/assets"
 | 
			
		||||
 | 
			
		||||
	"github.com/rs/zerolog"
 | 
			
		||||
	"github.com/rs/zerolog/log"
 | 
			
		||||
)
 | 
			
		||||
 | 
			
		||||
func main() {
 | 
			
		||||
	// Logger
 | 
			
		||||
	log.Logger = log.Output(zerolog.ConsoleWriter{Out: os.Stderr, TimeFormat: time.RFC3339}).With().Timestamp().Logger()
 | 
			
		||||
	log.Info().Str("version", assets.Version).Msg("Starting tinyauth")
 | 
			
		||||
	
 | 
			
		||||
	// Run cmd
 | 
			
		||||
	cmd.Execute()
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
										
											Binary file not shown.
										
									
								
							| 
		 Before Width: | Height: | Size: 61 KiB  | 
										
											Binary file not shown.
										
									
								
							| 
		 Before Width: | Height: | Size: 72 KiB  | 
										
											Binary file not shown.
										
									
								
							| 
		 Before Width: | Height: | Size: 66 KiB  | 
										
											Binary file not shown.
										
									
								
							| 
		 Before Width: | Height: | Size: 64 KiB  | 
		Reference in New Issue
	
	Block a user