diff --git a/README.md b/README.md index a6d338f..b460ccc 100644 --- a/README.md +++ b/README.md @@ -15,6 +15,9 @@ `rbinstall` is a utility for installing prebuilt Ruby to [rbenv](https://github.com/rbenv/rbenv). +> [!NOTE] +> Take a look at our [FAQ](https://kaos.sh/rbinstall/w/FAQ) for more information. + ### Usage demo [![demo](https://gh.kaos.st/rbinstall-300.gif)](#usage-demo) diff --git a/cli/cli.go b/cli/cli.go index 6273ef2..64540e8 100644 --- a/cli/cli.go +++ b/cli/cli.go @@ -66,7 +66,7 @@ import ( // App info const ( APP = "RBInstall" - VER = "3.3.0" + VER = "3.4.0" DESC = "Utility for installing prebuilt Ruby versions to rbenv" ) @@ -544,13 +544,23 @@ func printPrettyListing(dist, arch string) { } } - ruby := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_RUBY, options.GetB(OPT_ALL)) - jruby := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_JRUBY, options.GetB(OPT_ALL)) - truffle := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_TRUFFLE, options.GetB(OPT_ALL)) - other := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_OTHER, options.GetB(OPT_ALL)) + ruby := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_RUBY, true) + jruby := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_JRUBY, true) + truffle := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_TRUFFLE, true) + other := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_OTHER, true) + + rubyTotal := ruby.Total() + jrubyTotal := jruby.Total() + truffleTotal := truffle.Total() + otherTotal := other.Total() installed := getInstalledVersionsMap() + ruby = filterCategoryData(ruby, installed) + jruby = filterCategoryData(jruby, installed) + truffle = filterCategoryData(truffle, installed) + other = filterCategoryData(other, installed) + configureCategorySizes(map[string]index.CategoryData{ index.CATEGORY_RUBY: ruby, index.CATEGORY_JRUBY: jruby, @@ -558,11 +568,6 @@ func printPrettyListing(dist, arch string) { index.CATEGORY_OTHER: other, }) - rubyTotal := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_RUBY, true).Total() - jrubyTotal := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_JRUBY, true).Total() - truffleTotal := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_TRUFFLE, true).Total() - otherTotal := repoIndex.GetCategoryData(dist, arch, index.CATEGORY_OTHER, true).Total() - headerTemplate := getCategoryHeaderStyle(index.CATEGORY_RUBY) + " " + getCategoryHeaderStyle(index.CATEGORY_JRUBY) + " " + getCategoryHeaderStyle(index.CATEGORY_TRUFFLE) + " " + @@ -570,10 +575,10 @@ func printPrettyListing(dist, arch string) { fmtc.Printf( headerTemplate, - fmt.Sprintf("%s (%d)", strings.ToUpper(index.CATEGORY_RUBY), rubyTotal), - fmt.Sprintf("%s (%d)", strings.ToUpper(index.CATEGORY_JRUBY), jrubyTotal), - fmt.Sprintf("%s (%d)", strings.ToUpper(index.CATEGORY_TRUFFLE), truffleTotal), - fmt.Sprintf("%s (%d)", strings.ToUpper(index.CATEGORY_OTHER), otherTotal), + fmt.Sprintf("%s (%d/%d)", strings.ToUpper(index.CATEGORY_RUBY), countVersions(ruby), rubyTotal), + fmt.Sprintf("%s (%d/%d)", strings.ToUpper(index.CATEGORY_JRUBY), countVersions(jruby), jrubyTotal), + fmt.Sprintf("%s (%d/%d)", strings.ToUpper(index.CATEGORY_TRUFFLE), countVersions(truffle), truffleTotal), + fmt.Sprintf("%s (%d/%d)", strings.ToUpper(index.CATEGORY_OTHER), countVersions(other), otherTotal), ) var counter int @@ -597,7 +602,7 @@ func printPrettyListing(dist, arch string) { if !options.GetB(OPT_ALL) { fmtc.NewLine() - fmtc.Println("{s-}For listing outdated versions use option '--all'{!}") + fmtc.Printf("{s-}For listing outdated versions use option '%s'{!}\n", options.Format(OPT_ALL)) } } @@ -655,6 +660,10 @@ func installVersion(rubyVersion string, reinstall bool) { printErrorAndExit(err.Error()) } + progress.DefaultSettings.BarFgColorTag = "{" + categoryColor[category] + "}" + spinner.SpinnerColorTag = "{" + categoryColor[category] + "}" + fmtc.NameColor("category", "{"+categoryColor[category]+"}") + checkRBEnv() checkDependencies(info, category) @@ -672,10 +681,6 @@ func installVersion(rubyVersion string, reinstall bool) { var file string - progress.DefaultSettings.BarFgColorTag = "{" + categoryColor[category] + "}" - spinner.SpinnerColorTag = "{" + categoryColor[category] + "}" - fmtc.NameColor("category", "{"+categoryColor[category]+"}") - if !noProgress { fmtc.Printf("Fetching {*}{?category}%s{!} from storage…\n", info.Name) file, err = downloadFile(info) @@ -746,13 +751,8 @@ func installVersion(rubyVersion string, reinstall bool) { if knf.GetB(GEMS_RUBYGEMS_UPDATE) && strutil.HasPrefixAny(info.Name, "1", "2", "3") { rgVersion := getAdvisableRubyGemsVersion(info.Name) - gemVersion := rgVersion - - if gemVersion != "latest" && strings.Count(gemVersion, ".") < 2 { - gemVersion += ".x" - } - spinner.Show("Updating RubyGems to %s", gemVersion) + spinner.Show("Updating RubyGems to %s", formatGemVersion(rgVersion)) err = updateRubygemsTaskHandler(info.Name, rgVersion) spinner.Done(err == nil) @@ -766,15 +766,8 @@ func installVersion(rubyVersion string, reinstall bool) { if knf.GetS(GEMS_INSTALL) != "" { for _, gem := range strings.Split(knf.GetS(GEMS_INSTALL), " ") { gemName, gemVersion := parseGemInfo(gem) - taskDesc := fmt.Sprintf("Installing %s", gemName) - - if gemVersion != "" { - taskDesc += fmt.Sprintf(" (%s.x)", gemVersion) - } else { - taskDesc += fmt.Sprintf(" (latest)") - } - spinner.Show(taskDesc) + spinner.Show(fmt.Sprintf("Installing %s (%s)", gemName, formatGemVersion(gemVersion))) _, err = installGemTaskHandler(info.Name, gemName, gemVersion) spinner.Done(err == nil) @@ -941,18 +934,18 @@ func unistallTaskHandler(versionName string) error { var err error - // Remove symlink - if fsutil.IsExist(path.Join(versionsDir, cleanVersionName)) { - err = os.Remove(path.Join(versionsDir, cleanVersionName)) + // Remove directory with files + if fsutil.IsExist(path.Join(versionsDir, versionName)) { + err = os.RemoveAll(path.Join(versionsDir, versionName)) if err != nil { return err } } - // Remove directory with files - if fsutil.IsExist(path.Join(versionsDir, versionName)) { - err = os.RemoveAll(path.Join(versionsDir, versionName)) + // Remove symlink + if fsutil.IsExist(path.Join(versionsDir, cleanVersionName)) { + err = os.Remove(path.Join(versionsDir, cleanVersionName)) if err != nil { return err @@ -1037,12 +1030,21 @@ func updateGems(rubyVersion string) { printErrorAndExit("Version %s is not installed", rubyVersion) } + _, category, err := getVersionInfo(rubyVersion) + + if err == nil { + fmtc.NameColor("category", "{"+categoryColor[category]+"}") + spinner.SpinnerColorTag = "{" + categoryColor[category] + "}" + } else { + fmtc.NameColor("category", "{"+categoryColor[index.CATEGORY_RUBY]+"}") + } + checkRBEnv() runDate = time.Now() installed := false - fmtc.Printf("Updating gems for {c}%s{!}…\n\n", rubyVersion) + fmtc.Printf("Updating gems for {?category}%s{!}…\n\n", rubyVersion) // //////////////////////////////////////////////////////////////////////////////// // @@ -1063,22 +1065,16 @@ func updateGems(rubyVersion string) { // //////////////////////////////////////////////////////////////////////////////// // if knf.GetS(GEMS_INSTALL) != "" { - var gemVerInfo, installedVersion string + var installedVersion string for _, gem := range strings.Split(knf.GetS(GEMS_INSTALL), " ") { gemName, gemVersion := parseGemInfo(gem) - if gemVersion != "" { - gemVerInfo = fmt.Sprintf("(%s.x)", gemVersion) - } else { - gemVerInfo = fmt.Sprintf("(latest)") - } - if isGemInstalled(rubyVersion, gemName) { - spinner.Show("Updating %s %s", gemName, gemVerInfo) + spinner.Show("Updating %s (%s)", gemName, formatGemVersion(gemVersion)) installedVersion, err = updateGemTaskHandler(rubyVersion, gemName, gemVersion) } else { - spinner.Show("Installing %s %s", gemName, gemVerInfo) + spinner.Show("Installing %s (%s)", gemName, formatGemVersion(gemVersion)) installedVersion, err = installGemTaskHandler(rubyVersion, gemName, gemVersion) } @@ -1119,7 +1115,11 @@ func runGemCmd(rubyVersion, cmd, gem, gemVersion string) (string, error) { gemCmd := exec.Command(rubyPath+"/bin/ruby", rubyPath+"/bin/gem", cmd, gem, "--force") if gemVersion != "" { - gemCmd.Args = append(gemCmd.Args, "--version", fmt.Sprintf("~>%s", gemVersion)) + if strings.Count(gemVersion, ".") >= 2 { + gemCmd.Args = append(gemCmd.Args, "--version", fmt.Sprintf("%s", gemVersion)) + } else { + gemCmd.Args = append(gemCmd.Args, "--version", fmt.Sprintf("~>%s.0", gemVersion)) + } } if knf.GetB(GEMS_NO_DOCUMENT) { @@ -1142,45 +1142,38 @@ func runGemCmd(rubyVersion, cmd, gem, gemVersion string) (string, error) { return version, nil } - if gemVersion == "" { - gemVersion = "latest" - } else if strings.Count(gemVersion, ".") < 2 { - gemVersion += ".x" - } - actionLog, err := logFailedAction(strings.TrimRight(string(output), "\r\n")) if err == nil { switch cmd { case "update": - return "", fmtc.Errorf("Can't update gem %s (%s). Gem command output saved as %s.", gem, gemVersion, actionLog) + return "", fmtc.Errorf( + "Can't update gem %s (%s). Gem command output saved as %s.", + gem, formatGemVersion(gemVersion), actionLog, + ) default: - return "", fmtc.Errorf("Can't install gem %s (%s). Gem command output saved as %s.", gem, gemVersion, actionLog) + return "", fmtc.Errorf( + "Can't install gem %s (%s). Gem command output saved as %s.", + gem, formatGemVersion(gemVersion), actionLog, + ) } } switch cmd { case "update": - return "", fmtc.Errorf("Can't update gem %s (%s)", gem, gemVersion) + return "", fmtc.Errorf("Can't update gem %s (%s)", gem, formatGemVersion(gemVersion)) default: - return "", fmtc.Errorf("Can't install gem %s (%s)", gem, gemVersion) + return "", fmtc.Errorf("Can't install gem %s (%s)", gem, formatGemVersion(gemVersion)) } } // updateRubygems update rubygems to defined version func updateRubygems(rubyVersion, gemVersion string) error { - var gemCmd *exec.Cmd - rubyPath := getVersionPath(rubyVersion) + gemCmd := exec.Command(rubyPath+"/bin/ruby", rubyPath+"/bin/gem", "update", "--no-document", "--system") - if gemVersion == "latest" { - gemCmd = exec.Command(rubyPath+"/bin/ruby", rubyPath+"/bin/gem", "update", "--system") - } else { - gemCmd = exec.Command(rubyPath+"/bin/ruby", rubyPath+"/bin/gem", "update", "--system", gemVersion) - } - - if knf.GetB(GEMS_NO_DOCUMENT) { - gemCmd.Args = append(gemCmd.Args, "--no-document") + if gemVersion != "latest" { + gemCmd.Args = append(gemCmd.Args, gemVersion) } if knf.GetS(GEMS_SOURCE) != "" { @@ -1306,7 +1299,7 @@ func printCurrentVersionName(category string, versions index.CategoryData, insta } } - printRubyVersion(category, prettyName) + printRubyVersion(category, prettyName, info.EOL) return true } @@ -1362,8 +1355,12 @@ func printSized(format string, size int, a ...any) { } // printRubyVersion print version with align spaces -func printRubyVersion(category, name string) { - fmtc.Printf(" " + name + getAlignSpaces(fmtc.Clean(name), categorySize[category]) + " ") +func printRubyVersion(category, name string, eol bool) { + if !eol { + fmtc.Printf(" " + name + getAlignSpaces(fmtc.Clean(name), categorySize[category]) + " ") + } else { + fmtc.Printf(" {s}" + name + "{!}" + getAlignSpaces(fmtc.Clean(name), categorySize[category]) + " ") + } } // configureCategorySizes configure column size for each category @@ -1469,6 +1466,17 @@ func isGemInstalled(rubyVersion string, gemName string) bool { return false } +// formatGemVersion formats info about gem +func formatGemVersion(gemVersion string) string { + if gemVersion == "" || gemVersion == "latest" { + return "latest" + } else if strings.Count(gemVersion, ".") < 2 { + return fmt.Sprintf("%s.x", gemVersion) + } + + return gemVersion +} + // isVersionInstalled return true is given version already installed func isVersionInstalled(rubyVersion string) bool { fullPath := getVersionPath(rubyVersion) @@ -1559,6 +1567,49 @@ func getInstalledVersionsMap() map[string]bool { return result } +// filterCategoryData filters category data removing EOL versions +func filterCategoryData(versions index.CategoryData, installed map[string]bool) index.CategoryData { + var result index.CategoryData + + showAll := options.GetB(OPT_ALL) + +MAIN: + for _, info := range versions { + if info.EOL && !showAll { + for _, vInfo := range info.Variations { + if installed[vInfo.Name] { + result = append(result, info) + continue MAIN + } + } + + if installed[info.Name] { + result = append(result, info) + continue MAIN + } + } else { + result = append(result, info) + } + } + + return result +} + +// countVersions counts versions in category data +func countVersions(versions index.CategoryData) int { + var result int + + for _, info := range versions { + if info.EOL { + continue + } + + result += len(info.Variations) + 1 + } + + return result +} + // getVersionGemPath returns path to directory with installed gems func getVersionGemDirPath(rubyVersion string) string { gemsPath := getVersionPath(rubyVersion) + "/lib/ruby/gems" @@ -1583,12 +1634,12 @@ func getVersionPath(rubyVersion string) string { // getRBEnvVersionsPath return path to rbenv directory with all versions func getRBEnvVersionsPath() string { - return knf.GetS(RBENV_DIR) + "/versions" + return path.Join(knf.GetS(RBENV_DIR), "versions") } // getUnpackDirPath return path to directory for unpacking data func getUnpackDirPath() string { - return getRBEnvVersionsPath() + "/.rbinstall" + return path.Join(getRBEnvVersionsPath(), ".rbinstall") } // getAlignSpaces return spaces for output align @@ -1598,15 +1649,22 @@ func getAlignSpaces(t string, l int) string { // getGemSourceURL return url of gem source func getGemSourceURL(rubyVersion string) string { + source := knf.GetS(GEMS_SOURCE) + + if strutil.HasPrefixAny(source, "https://", "http://") { + source = strutil.Exclude(source, "https://") + source = strutil.Exclude(source, "http://") + } + if strings.HasPrefix(rubyVersion, "1.8") { - return "http://" + knf.GetS(GEMS_SOURCE) + return "http://" + source } if !options.GetB(OPT_GEMS_INSECURE) && knf.GetB(GEMS_SOURCE_SECURE, false) { - return "https://" + knf.GetS(GEMS_SOURCE) + return "https://" + source } - return "http://" + knf.GetS(GEMS_SOURCE) + return "http://" + source } // checkRBEnv check rbenv directory and state diff --git a/clone/clone.go b/clone/clone.go index 787e015..d7bcce7 100644 --- a/clone/clone.go +++ b/clone/clone.go @@ -43,7 +43,7 @@ import ( // App info const ( APP = "RBInstall Clone" - VER = "3.1.1" + VER = "3.1.2" DESC = "Utility for cloning RBInstall repository" ) diff --git a/common/rbinstall.spec b/common/rbinstall.spec index ae0a082..10a5981 100644 --- a/common/rbinstall.spec +++ b/common/rbinstall.spec @@ -10,7 +10,7 @@ Summary: Utility for installing prebuilt Ruby to rbenv Name: rbinstall -Version: 3.3.0 +Version: 3.4.0 Release: 0%{?dist} Group: Applications/System License: Apache License, Version 2.0 @@ -24,7 +24,7 @@ BuildRoot: %{_tmppath}/%{name}-%{version}-%{release}-root-%(%{__id_u} -n) Requires: rbenv libyaml ca-certificates zlib >= 1.2.11 -BuildRequires: golang >= 1.20 +BuildRequires: golang >= 1.21 Provides: %{name} = %{version}-%{release} @@ -38,7 +38,7 @@ Utility for installing different prebuilt versions of Ruby to rbenv. %package gen Summary: Utility for generating RBInstall index -Version: 3.2.1 +Version: 3.2.2 Release: 0%{?dist} Group: Development/Tools @@ -50,7 +50,7 @@ Utility for generating RBInstall index. %package clone Summary: Utility for cloning RBInstall repository -Version: 3.1.1 +Version: 3.1.2 Release: 0%{?dist} Group: Development/Tools @@ -118,6 +118,13 @@ rm -rf %{buildroot} ################################################################################ %changelog +* Mon Jan 15 2024 Anton Novojilov - 3.4.0-0 +- [cli] Improved versions listing +- [cli] Fixed bug with uninstalling +- [cli] UI fixes +- [cli] Code refactoring +- Dependencies update + * Fri Jan 12 2024 Anton Novojilov - 3.3.0-0 - [cli] Improved rubygems gem update - [cli|gen|clone] Code refactoring diff --git a/gen/gen.go b/gen/gen.go index 511aa1c..f44feb4 100644 --- a/gen/gen.go +++ b/gen/gen.go @@ -41,7 +41,7 @@ import ( // App info const ( APP = "RBInstall Gen" - VER = "3.2.1" + VER = "3.2.2" DESC = "Utility for generating RBInstall index" ) diff --git a/go.mod b/go.mod index 10ea326..233fd33 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.18 require ( github.com/essentialkaos/depsy v1.1.0 - github.com/essentialkaos/ek/v12 v12.93.0 + github.com/essentialkaos/ek/v12 v12.94.0 github.com/essentialkaos/npck v1.6.2 ) diff --git a/go.sum b/go.sum index 81e5ef5..ec481b7 100644 --- a/go.sum +++ b/go.sum @@ -1,8 +1,8 @@ github.com/essentialkaos/check v1.4.0 h1:kWdFxu9odCxUqo1NNFNJmguGrDHgwi3A8daXX1nkuKk= github.com/essentialkaos/depsy v1.1.0 h1:U6dp687UkQwXlZU17Hg2KMxbp3nfZAoZ8duaeUFYvJI= github.com/essentialkaos/depsy v1.1.0/go.mod h1:kpiTAV17dyByVnrbNaMcZt2jRwvuXClUYOzpyJQwtG8= -github.com/essentialkaos/ek/v12 v12.93.0 h1:5lwuNYgUYjQHy2h57adWt2t1w5L2MxqTPKgHjyH3h6k= -github.com/essentialkaos/ek/v12 v12.93.0/go.mod h1:peB5w8zUkRuDs7m/QG5Z2gMmqzSIs2viAmxzzNFIIoo= +github.com/essentialkaos/ek/v12 v12.94.0 h1:3mPRCOsJKictpqOuydZ0HleCQb/3B3STFo8GJ4yjFOM= +github.com/essentialkaos/ek/v12 v12.94.0/go.mod h1:peB5w8zUkRuDs7m/QG5Z2gMmqzSIs2viAmxzzNFIIoo= github.com/essentialkaos/go-linenoise/v3 v3.4.0 h1:g72w8x+/HIwOMBVvNaPYp+wMWVHrYZwzFAF7OfZR5Ts= github.com/essentialkaos/go-linenoise/v3 v3.4.0/go.mod h1:t1kNLY2bSMQCy1JXOefD2BDLs/TTPMtTv3DFNV5uDSI= github.com/essentialkaos/npck v1.6.2 h1:OqV74UfA9qoRYbFVsZ+KYH9kY//1nU6uX/HBfAltUQ4= diff --git a/index/index.go b/index/index.go index effd966..a87e3ae 100644 --- a/index/index.go +++ b/index/index.go @@ -58,7 +58,7 @@ type CategoryData []*VersionInfo // VersionInfo contains info about particular version type VersionInfo struct { - Variations []*VersionInfo `json:"variations,omitempty"` // Info about version variations (railsexpress/jemalloc) + Variations []*VersionInfo `json:"variations,omitempty"` // Info about version variations (jemalloc) Name string `json:"name"` // Base version name File string `json:"file"` // Full filename (with extension) Path string `json:"path"` // Relative path to file