This repository has been archived on 2024-03-03. You can view files and clone it, but cannot push or open issues or pull requests.
doggo-fetcher/cmd/main.go

356 lines
8.5 KiB
Go

package main
import (
"errors"
"fmt"
"os"
"os/exec"
"path/filepath"
"regexp"
"runtime"
"strings"
"github.com/datahearth/doggo-fetcher/pkg"
"github.com/sirupsen/logrus"
"github.com/urfave/cli/v2"
)
var (
dgfFolder = strings.Replace(pkg.DGF_FOLDER, "~", os.Getenv("HOME"), 1)
re = regexp.MustCompile(`\d*\.?\d*\.?\d(rc|beta)?\d*?`)
logger = &logrus.Logger{
Out: os.Stdout,
Formatter: new(pkg.LoggerFormatter),
Hooks: make(logrus.LevelHooks),
Level: logrus.InfoLevel,
}
app = &cli.App{
Name: "dg",
Usage: "I bring you your latest GoLang release with ease and efficiency (like a stick) !",
Description: `Doggo-fetcher is a utility tool that manage for you your installed GoLang releases.
You can select a specific GoLang release or even set a specific one for directories.`,
EnableBashCompletion: true,
Authors: []*cli.Author{
{
Name: "Antoine <DataHearth> Langlois",
Email: "antoine.l@antoine-langlois.net",
},
},
Suggest: true,
Version: "0.1.0",
Commands: []*cli.Command{
{
Name: "use",
Usage: "Set a specific golang version",
Description: `Use a specific golang version as primary golang binary for the user.
If the version is not already downloaded, it'll downloaded and installed automatically.`,
Flags: []cli.Flag{
&cli.BoolFlag{
Name: "latest",
Aliases: []string{"lts"},
Usage: "Use latest release",
},
&cli.BoolFlag{
Name: "rc",
Usage: "Allow \"rc\" version to be fetched",
},
&cli.BoolFlag{
Name: "beta",
Usage: "Allow \"beta\" version to be fetched",
},
},
Action: use,
},
{
Name: "init",
Usage: "Initialize doggofetcher",
Description: `Initialize doggofetcher by adding a custom path inside your sheel configuration file.
It also creates a folder at "~/.local/doggofetcher" to store your custom golang binaries.`,
Action: initFunc,
},
{
Name: "uninstall",
Usage: "Uninstall golang release",
Description: `Uninstall current golang release.
It will remove the folder at "~/.local/doggofetcher/go"`,
Action: uninstall,
},
{
Name: "ls",
Usage: "List available releases",
Description: `List available releases.
It will list all available releases in the "~/.local/doggofetcher/go*" folder.`,
Action: ls,
},
{
Name: "remove",
Usage: "Remove one or more releases",
Description: `Remove one or more releases.
Release(s) folder will be removed from "~/.local/doggofetcher" and thus not available.`,
Action: remove,
Flags: []cli.Flag{
&cli.BoolFlag{
Name: "all",
Aliases: []string{"a"},
Usage: "Remove all releases",
},
},
},
{
Name: "exec",
Usage: "Execute a go command",
Action: execCommand,
},
},
Flags: []cli.Flag{
&cli.BoolFlag{Name: "verbose", Aliases: []string{"v"}, Usage: "Enable verbose mode"},
},
}
)
// todo: ls-remote (list remote releases)
// todo: alias (set alias for a release)
// todo: auto-use (automatically switch when changing directory)
func main() {
cli.VersionFlag = &cli.BoolFlag{
Name: "version",
Aliases: []string{"V"},
Usage: "print the version",
}
if err := app.Run(os.Args); err != nil {
logger.Error(err)
}
}
func use(ctx *cli.Context) error {
if ctx.Bool("verbose") {
logger.Level = logrus.DebugLevel
}
var release string
if ctx.NArg() == 0 {
if !ctx.Bool("latest") {
return errors.New("a release is required if \"--latest|--lts\" is not passed")
}
release = pkg.LTS
} else {
release = ctx.Args().First()
if !re.Match([]byte(release)) {
return errors.New("release doesn't match \"\\d*\\.?\\d*\\.?\\d(rc|beta)?\\d*?\" format")
}
}
localFolder := dgfFolder
fi, err := os.Stat(localFolder)
if err != nil {
if !os.IsNotExist(err) {
return err
}
if err := os.MkdirAll(localFolder, 0755); err != nil {
return err
}
} else {
if !fi.IsDir() {
if err := os.RemoveAll(localFolder); err != nil {
return err
}
if err := os.MkdirAll(localFolder, 0755); err != nil {
return err
}
}
}
release, err = pkg.NewTags(release, ctx.Context).GetRelease(ctx.Bool("beta"), ctx.Bool("rc"))
if err != nil {
return err
}
hash, err := pkg.NewHash(localFolder)
if err != nil {
return err
}
r := pkg.NewRelease(release, filepath.Join(localFolder, release))
if err := r.CheckReleaseExists(); err != nil {
if err != pkg.ErrReleaseNotFound {
return err
}
logger.Info("Release not found, downloading...")
if err := r.DownloadRelease(); err != nil {
return err
}
logger.Info("Release downloaded, installing...")
if err := r.ExtractRelease(); err != nil {
return err
}
logger.Debug("Release installed, generating hash...")
h, err := hash.GetFolderHash(r.GetReleaseFolder())
if err != nil {
return err
}
if err := hash.AddHash(r.GetReleaseFolder(), h); err != nil {
return err
}
} else {
logger.Debug("Release found, checking hash...")
h, err := hash.GetFolderHash(r.GetReleaseFolder())
if err != nil {
return err
}
if err := hash.CompareReleaseHash(r.GetReleaseFolder(), h); err != nil {
if err == pkg.ErrHashNotFound {
logger.Warnln("Hash not found in hash table, adding...")
if err := hash.AddHash(r.GetReleaseFolder(), h); err != nil {
return err
}
} else if err == pkg.ErrHashInvalid {
logger.Warnln("Hash invalid, replacing...")
if err := hash.ReplaceHash(r.GetReleaseFolder(), h); err != nil {
return err
}
} else {
return err
}
}
}
logger.Info("Setting golang binary")
if err := pkg.UpdateRelease(r.GetReleaseFolder()); err != nil {
return err
}
logger.Info("Everything done!")
return nil
}
func initFunc(ctx *cli.Context) error {
if ctx.Bool("verbose") {
logger.Level = logrus.DebugLevel
}
return pkg.Init(logger)
}
func uninstall(ctx *cli.Context) error {
return os.RemoveAll(filepath.Join(dgfFolder, "go"))
}
func ls(ctx *cli.Context) error {
if ctx.Bool("verbose") {
logger.Level = logrus.DebugLevel
}
items, err := os.ReadDir(dgfFolder)
if err != nil {
return err
}
for _, i := range items {
if i.IsDir() && i.Name() != "go" {
fmt.Println(i.Name())
}
}
return nil
}
func remove(ctx *cli.Context) error {
if ctx.Bool("verbose") {
logger.Level = logrus.DebugLevel
}
if !ctx.Bool("all") {
if ctx.NArg() == 0 {
return errors.New("a release is required")
}
release := ctx.Args().First()
if !re.Match([]byte(release)) {
return errors.New("release doesn't match \"\\d*\\.?\\d*\\.?\\d(rc|beta)?\\d*?\" format")
}
hash, err := pkg.NewHash(dgfFolder)
if err != nil {
return err
}
if err := os.RemoveAll(filepath.Join(dgfFolder, fmt.Sprintf("go%s", release))); err != nil {
return fmt.Errorf("could not remove release %s: %s", release, err)
}
if err := hash.RemoveHash(fmt.Sprintf("%s/go%s", dgfFolder, release)); err != nil {
return err
}
if strings.Contains(runtime.Version(), ctx.Args().First()) {
logger.Infoln("Removing installed version...")
if err := os.RemoveAll(filepath.Join(dgfFolder, "go")); err != nil {
return fmt.Errorf("could not remove installed release %s: %s", release, err)
}
}
logger.Info("Release removed")
return nil
}
entries, err := os.ReadDir(dgfFolder)
if err != nil {
return fmt.Errorf("could not read local folder: %s", err)
}
hash, err := pkg.NewHash(dgfFolder)
if err != nil {
return err
}
for _, f := range entries {
if !f.IsDir() {
continue
}
if strings.Contains(f.Name(), "go") && f.Name() != "go" {
folderPath := filepath.Join(dgfFolder, f.Name())
if err := os.RemoveAll(folderPath); err != nil {
return fmt.Errorf("could not remove %s: %s", folderPath, err)
}
if err := hash.RemoveHash(folderPath); err != nil {
return err
}
}
}
logger.Infoln("Removing installed version...")
if err := os.RemoveAll(filepath.Join(dgfFolder, "go")); err != nil {
return fmt.Errorf("could not remove installed release: %s", err)
}
logger.Info("All releases removed")
return nil
}
func execCommand(ctx *cli.Context) error {
if ctx.Bool("verbose") {
logger.Level = logrus.DebugLevel
}
if ctx.NArg() == 0 {
return errors.New("a release is required")
}
release := ctx.Args().First()
if !re.Match([]byte(release)) {
return errors.New("release doesn't match \"\\d*\\.?\\d*\\.?\\d(rc|beta)?\\d*?\" format")
}
cmd := exec.Command(filepath.Join(dgfFolder, fmt.Sprintf("go%s", release), "bin", "go"), ctx.Args().Tail()...)
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
return cmd.Run()
}