From a582636e16d9b43ce88591a68cf8b25fb5a5d56f Mon Sep 17 00:00:00 2001 From: Suraj B M Date: Mon, 22 Sep 2025 23:38:55 +0530 Subject: [PATCH] feat: implement create command --- Makefile | 2 + cmd/create.go | 103 +++++++++++++++++++++++++++++++++++++++++++++ cmd/utils/db.go | 2 +- cmd/utils/tools.go | 22 +++++++++- go.mod | 11 +++-- go.sum | 4 ++ 6 files changed, 137 insertions(+), 7 deletions(-) create mode 100644 Makefile create mode 100644 cmd/create.go diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..2098ff4 --- /dev/null +++ b/Makefile @@ -0,0 +1,2 @@ +all: + go build -o ~/bin/test/envy . diff --git a/cmd/create.go b/cmd/create.go new file mode 100644 index 0000000..668e503 --- /dev/null +++ b/cmd/create.go @@ -0,0 +1,103 @@ +/* +Copyright © 2025 NAME HERE +*/ +package cmd + +import ( + "envy/cmd/utils" + "fmt" + "log" + "os" + "path" + + "github.com/charmbracelet/huh" + "github.com/spf13/cobra" +) + +// createCmd represents the create command +var createCmd = &cobra.Command{ + Use: "create", + Short: "Create a project/environment in the current directory", + Run: func(cmd *cobra.Command, args []string) { + cwd, err := os.Getwd() + if err != nil { + utils.ErrPrint("Not able to fetch working directory:", err.Error()) + return + } + + _, err = os.Stat(path.Join(cwd, ".envy")) + + if os.IsExist(err) { + return + } else { + _, err = os.Stat(path.Join(cwd, ".env")) + + if os.IsNotExist(err) { + utils.ErrPrint(".env file does not exist in the current directory.") + return + } + + project := "" + environment := "" + + form := huh.NewForm( + huh.NewGroup( + huh.NewInput(). + Title("Project name"). + Prompt("? "). + Validate(utils.NoSpace). + Value(&project), + + huh.NewInput(). + Title("Environment name"). + Prompt("? "). + Validate(utils.NoSpace). + Value(&environment), + ), + ) + + err := form.Run() + if err != nil { + utils.ErrPrint("Some error occured:", err.Error()) + } + + envFile, err := os.ReadFile(path.Join(cwd, ".env")) + if err != nil { + utils.ErrPrint("Error occured while opening .env:", err.Error()) + return + } + + utils.InitDb() + + proj := utils.Project{Name: project} + utils.DB.Create(&proj) + + env := utils.Environment{Name: environment, ProjectID: proj.ID, Active: true, Data: string(envFile)} + utils.DB.Create(&env) + + filePath := path.Join(cwd, ".envy") + + err = os.WriteFile(filePath, []byte(fmt.Sprintf("[envy]\nproject = \"%s\"\nenv = \"%s\"\n", project, environment)), 0644) + if err != nil { + log.Println("Error creating .envy file:", err) + return + } + + fmt.Printf("Created environment %s inside of project %s.\n", environment, project) + } + }, +} + +func init() { + rootCmd.AddCommand(createCmd) + + // Here you will define your flags and configuration settings. + + // Cobra supports Persistent Flags which will work for this command + // and all subcommands, e.g.: + // createCmd.PersistentFlags().String("foo", "", "A help for foo") + + // Cobra supports local flags which will only run when this command + // is called directly, e.g.: + // createCmd.Flags().BoolP("toggle", "t", false, "Help message for toggle") +} diff --git a/cmd/utils/db.go b/cmd/utils/db.go index ad5169a..d116b63 100644 --- a/cmd/utils/db.go +++ b/cmd/utils/db.go @@ -28,7 +28,7 @@ func InitDb() { var openErr error DB, openErr = gorm.Open(sqlite.Open(path.Join(EnvyPath, "envy.db")), &gorm.Config{}) if openErr != nil { - ErrPrint("Error while opening database:" + openErr.Error()) + ErrPrint("Error while opening database:", openErr.Error()) } } diff --git a/cmd/utils/tools.go b/cmd/utils/tools.go index e9fea45..caa0288 100644 --- a/cmd/utils/tools.go +++ b/cmd/utils/tools.go @@ -1,17 +1,19 @@ package utils import ( + "errors" "fmt" "os" "path" + "strings" "github.com/charmbracelet/lipgloss" ) var ErrStyle = lipgloss.NewStyle().Foreground(lipgloss.Color("#F00")) -func ErrPrint(data string) { - fmt.Println(ErrStyle.Render(data)) +func ErrPrint(data ...string) { + fmt.Println(ErrStyle.Render(strings.Join(data, " "))) } var EnvyPath string @@ -24,3 +26,19 @@ func Init() { EnvyPath = path.Join(conf, "envy") } + +func NoSpace(input string) error { + for _, r := range input { + if !(isAlphaNum(r) || r == '-' || r == '_') { + return errors.New("input can only contain letters, numbers, '-' and '_'") + } + } + return nil +} + +// helper function +func isAlphaNum(r rune) bool { + return (r >= 'a' && r <= 'z') || + (r >= 'A' && r <= 'Z') || + (r >= '0' && r <= '9') +} diff --git a/go.mod b/go.mod index cb5ee97..9490a96 100644 --- a/go.mod +++ b/go.mod @@ -2,6 +2,13 @@ module envy go 1.25.0 +require ( + github.com/charmbracelet/lipgloss v1.1.0 + github.com/spf13/cobra v1.10.1 + gorm.io/driver/sqlite v1.6.0 + gorm.io/gorm v1.31.0 +) + require ( github.com/atotto/clipboard v0.1.4 // indirect github.com/aymanbagabas/go-osc52/v2 v2.0.1 // indirect @@ -10,7 +17,6 @@ require ( github.com/charmbracelet/bubbletea v1.3.6 // indirect github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc // indirect github.com/charmbracelet/huh v0.7.0 // indirect - github.com/charmbracelet/lipgloss v1.1.0 // indirect github.com/charmbracelet/x/ansi v0.9.3 // indirect github.com/charmbracelet/x/cellbuf v0.0.13 // indirect github.com/charmbracelet/x/exp/strings v0.0.0-20240722160745-212f7b056ed0 // indirect @@ -30,14 +36,11 @@ require ( github.com/muesli/cancelreader v0.2.2 // indirect github.com/muesli/termenv v0.16.0 // indirect github.com/rivo/uniseg v0.4.7 // indirect - github.com/spf13/cobra v1.10.1 // indirect github.com/spf13/pflag v1.0.9 // indirect github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e // indirect golang.org/x/sync v0.17.0 // indirect golang.org/x/sys v0.33.0 // indirect golang.org/x/text v0.29.0 // indirect - gorm.io/driver/sqlite v1.6.0 // indirect - gorm.io/gorm v1.31.0 // indirect ) replace github.com/charmbracelet/huh => github.com/silicoflare/huh v0.0.0-20250921143325-6504e5f1a84d diff --git a/go.sum b/go.sum index 1744868..358dc7a 100644 --- a/go.sum +++ b/go.sum @@ -6,9 +6,11 @@ github.com/catppuccin/go v0.3.0 h1:d+0/YicIq+hSTo5oPuRi5kOpqkVA5tAsU6dNhvRu+aY= github.com/catppuccin/go v0.3.0/go.mod h1:8IHJuMGaUUjQM82qBrGNBv7LFq6JI3NnQCF6MOlZjpc= github.com/charmbracelet/bubbles v0.21.0 h1:9TdC97SdRVg/1aaXNVWfFH3nnLAwOXr8Fn6u6mfQdFs= github.com/charmbracelet/bubbles v0.21.0/go.mod h1:HF+v6QUR4HkEpz62dx7ym2xc71/KBHg+zKwJtMw+qtg= +github.com/charmbracelet/bubbles v0.21.1-0.20250623103423-23b8fd6302d7 h1:JFgG/xnwFfbezlUnFMJy0nusZvytYysV4SCS2cYbvws= github.com/charmbracelet/bubbles v0.21.1-0.20250623103423-23b8fd6302d7/go.mod h1:ISC1gtLcVilLOf23wvTfoQuYbW2q0JevFxPfUzZ9Ybw= github.com/charmbracelet/bubbletea v1.3.4 h1:kCg7B+jSCFPLYRA52SDZjr51kG/fMUEoPoZrkaDHyoI= github.com/charmbracelet/bubbletea v1.3.4/go.mod h1:dtcUCyCGEX3g9tosuYiut3MXgY/Jsv9nKVdibKKRRXo= +github.com/charmbracelet/bubbletea v1.3.6 h1:VkHIxPJQeDt0aFJIsVxw8BQdh/F/L2KKZGsK6et5taU= github.com/charmbracelet/bubbletea v1.3.6/go.mod h1:oQD9VCRQFF8KplacJLo28/jofOI2ToOfGYeFgBBxHOc= github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc h1:4pZI35227imm7yK2bGPcfpFEmuY1gc2YSTShr4iJBfs= github.com/charmbracelet/colorprofile v0.2.3-0.20250311203215-f60798e515dc/go.mod h1:X4/0JoqgTIPSFcRA/P6INZzIuyqdFY5rm8tb41s9okk= @@ -59,6 +61,7 @@ github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJ github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ= github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88= github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= +github.com/silicoflare/huh v0.0.0-20250921143325-6504e5f1a84d h1:mHyHR5yYBhJvdPsB2ra7+AQKOqKdaGHsq6eS/EyuLV8= github.com/silicoflare/huh v0.0.0-20250921143325-6504e5f1a84d/go.mod h1:5YVc+SlZ1IhQALxRPpkGwwEKftN/+OlJlnJYlDRFqN4= github.com/spf13/cobra v1.10.1 h1:lJeBwCfmrnXthfAupyUTzJ/J4Nc1RsHC/mSRU2dll/s= github.com/spf13/cobra v1.10.1/go.mod h1:7SmJGaTHFVBY0jW4NXGluQoLvhqFQM+6XSKD+P4XaB0= @@ -69,6 +72,7 @@ github.com/xo/terminfo v0.0.0-20220910002029-abceb7e1c41e/go.mod h1:RbqR21r5mrJu golang.org/x/sync v0.12.0 h1:MHc5BpPuC30uJk597Ri8TV3CNZcTLu6B6z4lJy+g6Jw= golang.org/x/sync v0.12.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA= golang.org/x/sync v0.15.0/go.mod h1:1dzgHSNfp02xaA81J2MS99Qcpr2w7fw1gpm99rleRqA= +golang.org/x/sync v0.17.0 h1:l60nONMj9l5drqw6jlhIELNv9I0A4OFgRsG9k2oT9Ug= golang.org/x/sync v0.17.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI= golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=