Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| a708c1980a | |||
| 16dd798f10 | |||
| fcd454691f |
31
install.go
31
install.go
@@ -21,6 +21,8 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"log/slog"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/leonelquinteros/gotext"
|
"github.com/leonelquinteros/gotext"
|
||||||
"github.com/urfave/cli/v2"
|
"github.com/urfave/cli/v2"
|
||||||
@@ -110,26 +112,53 @@ func InstallCmd() *cli.Command {
|
|||||||
return nil
|
return nil
|
||||||
}),
|
}),
|
||||||
BashComplete: cliutils.BashCompleteWithError(func(c *cli.Context) error {
|
BashComplete: cliutils.BashCompleteWithError(func(c *cli.Context) error {
|
||||||
|
|
||||||
ctx := c.Context
|
ctx := c.Context
|
||||||
deps, err := appbuilder.
|
deps, err := appbuilder.
|
||||||
New(ctx).
|
New(ctx).
|
||||||
WithConfig().
|
WithConfig().
|
||||||
WithDB().
|
WithDB().
|
||||||
|
WithManager().
|
||||||
Build()
|
Build()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer deps.Defer()
|
defer deps.Defer()
|
||||||
|
|
||||||
|
seen := make(map[string]struct{})
|
||||||
|
|
||||||
|
var prefix string
|
||||||
|
if c.Args().Len() > 0 {
|
||||||
|
prefix = c.Args().Get(c.Args().Len() - 1)
|
||||||
|
if strings.HasPrefix(prefix, "-") {
|
||||||
|
prefix = ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
result, err := deps.DB.GetPkgs(c.Context, "true")
|
result, err := deps.DB.GetPkgs(c.Context, "true")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return cliutils.FormatCliExit(gotext.Get("Error getting packages"), err)
|
return cliutils.FormatCliExit(gotext.Get("Error getting packages"), err)
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, pkg := range result {
|
for _, pkg := range result {
|
||||||
|
if prefix == "" || strings.HasPrefix(pkg.Name, prefix) {
|
||||||
|
if _, ok := seen[pkg.Name]; !ok {
|
||||||
|
seen[pkg.Name] = struct{}{}
|
||||||
fmt.Println(pkg.Name)
|
fmt.Println(pkg.Name)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sysPkgs, err := deps.Manager.ListAvailable(prefix)
|
||||||
|
if err != nil {
|
||||||
|
slog.Debug("failed to list system packages", "err", err)
|
||||||
|
} else {
|
||||||
|
for _, name := range sysPkgs {
|
||||||
|
if _, ok := seen[name]; !ok {
|
||||||
|
seen[name] = struct{}{}
|
||||||
|
fmt.Println(name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -614,9 +614,9 @@ func (b *Builder) BuildALRDeps(
|
|||||||
allPkgs = append(allPkgs, p.pkg)
|
allPkgs = append(allPkgs, p.pkg)
|
||||||
}
|
}
|
||||||
|
|
||||||
slog.Info("DEBUG: allPkgs count", "count", len(allPkgs))
|
slog.Debug("allPkgs count", "count", len(allPkgs))
|
||||||
for _, p := range allPkgsWithKeys {
|
for _, p := range allPkgsWithKeys {
|
||||||
slog.Info("DEBUG: package in depTree", "key", p.key, "name", p.pkg.Name, "repo", p.pkg.Repository)
|
slog.Debug("package in depTree", "key", p.key, "name", p.pkg.Name, "repo", p.pkg.Repository)
|
||||||
}
|
}
|
||||||
|
|
||||||
needBuildPkgs, err := b.installerExecutor.FilterPackagesByVersion(ctx, allPkgs, input.OSRelease())
|
needBuildPkgs, err := b.installerExecutor.FilterPackagesByVersion(ctx, allPkgs, input.OSRelease())
|
||||||
@@ -630,9 +630,9 @@ func (b *Builder) BuildALRDeps(
|
|||||||
needBuildNames[pkg.Name] = true
|
needBuildNames[pkg.Name] = true
|
||||||
}
|
}
|
||||||
|
|
||||||
slog.Info("DEBUG: needBuildPkgs count", "count", len(needBuildPkgs))
|
slog.Debug("needBuildPkgs count", "count", len(needBuildPkgs))
|
||||||
for _, pkg := range needBuildPkgs {
|
for _, pkg := range needBuildPkgs {
|
||||||
slog.Info("DEBUG: package needs build", "name", pkg.Name)
|
slog.Debug("package needs build", "name", pkg.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Строим needBuildSet по КЛЮЧАМ depTree, а не по pkg.Name
|
// Строим needBuildSet по КЛЮЧАМ depTree, а не по pkg.Name
|
||||||
@@ -647,13 +647,13 @@ func (b *Builder) BuildALRDeps(
|
|||||||
|
|
||||||
// Шаг 3: Группируем подпакеты по basePkgName для оптимизации сборки
|
// Шаг 3: Группируем подпакеты по basePkgName для оптимизации сборки
|
||||||
// Если несколько подпакетов из одного мультипакета, собираем их вместе
|
// Если несколько подпакетов из одного мультипакета, собираем их вместе
|
||||||
slog.Info("DEBUG: sortedPkgs", "pkgs", sortedPkgs)
|
slog.Debug("sortedPkgs", "pkgs", sortedPkgs)
|
||||||
|
|
||||||
// Шаг 4: Собираем пакеты в правильном порядке, проверяя кеш
|
// Шаг 4: Собираем пакеты в правильном порядке, проверяя кеш
|
||||||
for _, pkgName := range sortedPkgs {
|
for _, pkgName := range sortedPkgs {
|
||||||
node := depTree[pkgName]
|
node := depTree[pkgName]
|
||||||
if node == nil {
|
if node == nil {
|
||||||
slog.Info("DEBUG: node is nil", "pkgName", pkgName)
|
slog.Debug("node is nil", "pkgName", pkgName)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -662,7 +662,7 @@ func (b *Builder) BuildALRDeps(
|
|||||||
|
|
||||||
// Пропускаем уже установленные пакеты
|
// Пропускаем уже установленные пакеты
|
||||||
if !needBuildSet[pkgName] {
|
if !needBuildSet[pkgName] {
|
||||||
slog.Info("DEBUG: skipping (not in needBuildSet)", "pkgName", pkgName)
|
slog.Debug("skipping (not in needBuildSet)", "pkgName", pkgName)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -687,12 +687,12 @@ func (b *Builder) BuildALRDeps(
|
|||||||
|
|
||||||
if allInCache {
|
if allInCache {
|
||||||
// Подпакет в кеше, используем его
|
// Подпакет в кеше, используем его
|
||||||
slog.Info("DEBUG: using cached package", "pkgName", pkgName)
|
slog.Debug("using cached package", "pkgName", pkgName)
|
||||||
buildDeps = append(buildDeps, cachedDeps...)
|
buildDeps = append(buildDeps, cachedDeps...)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
slog.Info("DEBUG: building package", "pkgName", pkgName)
|
slog.Debug("building package", "pkgName", pkgName)
|
||||||
|
|
||||||
// Собираем только запрошенный подпакет
|
// Собираем только запрошенный подпакет
|
||||||
// SkipDepsBuilding: true предотвращает рекурсивный вызов BuildALRDeps
|
// SkipDepsBuilding: true предотвращает рекурсивный вызов BuildALRDeps
|
||||||
|
|||||||
@@ -116,6 +116,48 @@ func (a *APK) UpgradeAll(opts *Opts) error {
|
|||||||
return a.Upgrade(opts)
|
return a.Upgrade(opts)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (a *APK) ListAvailable(prefix string) ([]string, error) {
|
||||||
|
cmd := exec.Command("apk", "search", "-q", prefix)
|
||||||
|
stdout, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("apk: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Start(); err != nil {
|
||||||
|
return nil, fmt.Errorf("apk: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
seen := make(map[string]struct{})
|
||||||
|
var pkgs []string
|
||||||
|
scanner := bufio.NewScanner(stdout)
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
if line == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// apk search -q returns "name-version", extract name
|
||||||
|
lastDash := strings.LastIndex(line, "-")
|
||||||
|
name := line
|
||||||
|
if lastDash > 0 {
|
||||||
|
name = line[:lastDash]
|
||||||
|
}
|
||||||
|
if _, ok := seen[name]; !ok {
|
||||||
|
seen[name] = struct{}{}
|
||||||
|
pkgs = append(pkgs, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, fmt.Errorf("apk: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Wait(); err != nil {
|
||||||
|
return nil, fmt.Errorf("apk: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkgs, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (a *APK) ListInstalled(opts *Opts) (map[string]string, error) {
|
func (a *APK) ListInstalled(opts *Opts) (map[string]string, error) {
|
||||||
out := map[string]string{}
|
out := map[string]string{}
|
||||||
cmd := exec.Command("apk", "list", "-I")
|
cmd := exec.Command("apk", "list", "-I")
|
||||||
|
|||||||
@@ -196,6 +196,10 @@ func (a *APT) IsInstalled(pkg string) (bool, error) {
|
|||||||
return strings.Contains(status, "install ok installed"), nil
|
return strings.Contains(status, "install ok installed"), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (a *APT) ListAvailable(prefix string) ([]string, error) {
|
||||||
|
return aptCacheListAvailable(prefix)
|
||||||
|
}
|
||||||
|
|
||||||
func (a *APT) GetInstalledVersion(pkg string) (string, error) {
|
func (a *APT) GetInstalledVersion(pkg string) (string, error) {
|
||||||
resolved := a.resolvePackageName(pkg)
|
resolved := a.resolvePackageName(pkg)
|
||||||
cmd := exec.Command("dpkg-query", "-f", "${Version}", "-W", resolved)
|
cmd := exec.Command("dpkg-query", "-f", "${Version}", "-W", resolved)
|
||||||
|
|||||||
51
internal/manager/apt_common.go
Normal file
51
internal/manager/apt_common.go
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
// ALR - Any Linux Repository
|
||||||
|
// Copyright (C) 2025 The ALR Authors
|
||||||
|
//
|
||||||
|
// This program is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program is distributed in the hope that it will be useful,
|
||||||
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
// GNU General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
package manager
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"fmt"
|
||||||
|
"os/exec"
|
||||||
|
)
|
||||||
|
|
||||||
|
func aptCacheListAvailable(prefix string) ([]string, error) {
|
||||||
|
cmd := exec.Command("apt-cache", "pkgnames", prefix)
|
||||||
|
stdout, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("apt-cache: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Start(); err != nil {
|
||||||
|
return nil, fmt.Errorf("apt-cache: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var pkgs []string
|
||||||
|
scanner := bufio.NewScanner(stdout)
|
||||||
|
for scanner.Scan() {
|
||||||
|
pkgs = append(pkgs, scanner.Text())
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, fmt.Errorf("apt-cache: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Wait(); err != nil {
|
||||||
|
return nil, fmt.Errorf("apt-cache: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkgs, nil
|
||||||
|
}
|
||||||
@@ -100,6 +100,10 @@ func (a *APTRpm) Upgrade(opts *Opts, pkgs ...string) error {
|
|||||||
return a.Install(opts, pkgs...)
|
return a.Install(opts, pkgs...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (a *APTRpm) ListAvailable(prefix string) ([]string, error) {
|
||||||
|
return aptCacheListAvailable(prefix)
|
||||||
|
}
|
||||||
|
|
||||||
func (a *APTRpm) UpgradeAll(opts *Opts) error {
|
func (a *APTRpm) UpgradeAll(opts *Opts) error {
|
||||||
opts = ensureOpts(opts)
|
opts = ensureOpts(opts)
|
||||||
cmd := a.getCmd(opts, "apt-get", "dist-upgrade")
|
cmd := a.getCmd(opts, "apt-get", "dist-upgrade")
|
||||||
|
|||||||
@@ -20,6 +20,7 @@
|
|||||||
package manager
|
package manager
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
)
|
)
|
||||||
@@ -107,6 +108,42 @@ func (d *DNF) Upgrade(opts *Opts, pkgs ...string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (d *DNF) ListAvailable(prefix string) ([]string, error) {
|
||||||
|
cmd := exec.Command("dnf", "repoquery", "--qf", "%{name}\n", "--quiet", prefix+"*")
|
||||||
|
stdout, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("dnf: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Start(); err != nil {
|
||||||
|
return nil, fmt.Errorf("dnf: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
seen := make(map[string]struct{})
|
||||||
|
var pkgs []string
|
||||||
|
scanner := bufio.NewScanner(stdout)
|
||||||
|
for scanner.Scan() {
|
||||||
|
name := scanner.Text()
|
||||||
|
if name == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, ok := seen[name]; !ok {
|
||||||
|
seen[name] = struct{}{}
|
||||||
|
pkgs = append(pkgs, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, fmt.Errorf("dnf: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Wait(); err != nil {
|
||||||
|
return nil, fmt.Errorf("dnf: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkgs, nil
|
||||||
|
}
|
||||||
|
|
||||||
// UpgradeAll обновляет все установленные пакеты
|
// UpgradeAll обновляет все установленные пакеты
|
||||||
func (d *DNF) UpgradeAll(opts *Opts) error {
|
func (d *DNF) UpgradeAll(opts *Opts) error {
|
||||||
opts = ensureOpts(opts)
|
opts = ensureOpts(opts)
|
||||||
|
|||||||
@@ -79,6 +79,9 @@ type Manager interface {
|
|||||||
// GetInstalledVersion returns the version of an installed package.
|
// GetInstalledVersion returns the version of an installed package.
|
||||||
// Returns empty string and no error if package is not installed.
|
// Returns empty string and no error if package is not installed.
|
||||||
GetInstalledVersion(string) (string, error)
|
GetInstalledVersion(string) (string, error)
|
||||||
|
// ListAvailable returns names of available packages matching the given prefix.
|
||||||
|
// The prefix is used for filtering to avoid returning all packages.
|
||||||
|
ListAvailable(prefix string) ([]string, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Detect returns the package manager detected on the system
|
// Detect returns the package manager detected on the system
|
||||||
|
|||||||
@@ -115,6 +115,42 @@ func (p *Pacman) UpgradeAll(opts *Opts) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *Pacman) ListAvailable(prefix string) ([]string, error) {
|
||||||
|
cmd := exec.Command("pacman", "-Ssq", "^"+prefix)
|
||||||
|
stdout, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("pacman: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Start(); err != nil {
|
||||||
|
return nil, fmt.Errorf("pacman: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
seen := make(map[string]struct{})
|
||||||
|
var pkgs []string
|
||||||
|
scanner := bufio.NewScanner(stdout)
|
||||||
|
for scanner.Scan() {
|
||||||
|
name := scanner.Text()
|
||||||
|
if _, ok := seen[name]; !ok {
|
||||||
|
seen[name] = struct{}{}
|
||||||
|
pkgs = append(pkgs, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, fmt.Errorf("pacman: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Wait(); err != nil {
|
||||||
|
if exitErr, ok := err.(*exec.ExitError); ok && exitErr.ExitCode() == 1 {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("pacman: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkgs, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (p *Pacman) ListInstalled(opts *Opts) (map[string]string, error) {
|
func (p *Pacman) ListInstalled(opts *Opts) (map[string]string, error) {
|
||||||
out := map[string]string{}
|
out := map[string]string{}
|
||||||
cmd := exec.Command("pacman", "-Q")
|
cmd := exec.Command("pacman", "-Q")
|
||||||
|
|||||||
@@ -20,6 +20,7 @@
|
|||||||
package manager
|
package manager
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
)
|
)
|
||||||
@@ -103,6 +104,42 @@ func (y *YUM) Upgrade(opts *Opts, pkgs ...string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (y *YUM) ListAvailable(prefix string) ([]string, error) {
|
||||||
|
cmd := exec.Command("yum", "repoquery", "--qf", "%{name}\n", "--quiet", prefix+"*")
|
||||||
|
stdout, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("yum: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Start(); err != nil {
|
||||||
|
return nil, fmt.Errorf("yum: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
seen := make(map[string]struct{})
|
||||||
|
var pkgs []string
|
||||||
|
scanner := bufio.NewScanner(stdout)
|
||||||
|
for scanner.Scan() {
|
||||||
|
name := scanner.Text()
|
||||||
|
if name == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, ok := seen[name]; !ok {
|
||||||
|
seen[name] = struct{}{}
|
||||||
|
pkgs = append(pkgs, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, fmt.Errorf("yum: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Wait(); err != nil {
|
||||||
|
return nil, fmt.Errorf("yum: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkgs, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (y *YUM) UpgradeAll(opts *Opts) error {
|
func (y *YUM) UpgradeAll(opts *Opts) error {
|
||||||
opts = ensureOpts(opts)
|
opts = ensureOpts(opts)
|
||||||
cmd := y.getCmd(opts, "yum", "upgrade")
|
cmd := y.getCmd(opts, "yum", "upgrade")
|
||||||
|
|||||||
@@ -20,8 +20,10 @@
|
|||||||
package manager
|
package manager
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Zypper represents the Zypper package manager
|
// Zypper represents the Zypper package manager
|
||||||
@@ -103,6 +105,55 @@ func (z *Zypper) Upgrade(opts *Opts, pkgs ...string) error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (z *Zypper) ListAvailable(prefix string) ([]string, error) {
|
||||||
|
cmd := exec.Command("zypper", "--quiet", "search", "--type", "package", prefix+"*")
|
||||||
|
stdout, err := cmd.StdoutPipe()
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("zypper: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Start(); err != nil {
|
||||||
|
return nil, fmt.Errorf("zypper: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
seen := make(map[string]struct{})
|
||||||
|
var pkgs []string
|
||||||
|
scanner := bufio.NewScanner(stdout)
|
||||||
|
for scanner.Scan() {
|
||||||
|
line := scanner.Text()
|
||||||
|
// zypper table format: "S | Name | Summary | Type"
|
||||||
|
// Skip separator lines and headers
|
||||||
|
if !strings.Contains(line, "|") {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
fields := strings.Split(line, "|")
|
||||||
|
if len(fields) < 2 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
name := strings.TrimSpace(fields[1])
|
||||||
|
if name == "" || name == "Name" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, ok := seen[name]; !ok {
|
||||||
|
seen[name] = struct{}{}
|
||||||
|
pkgs = append(pkgs, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := scanner.Err(); err != nil {
|
||||||
|
return nil, fmt.Errorf("zypper: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := cmd.Wait(); err != nil {
|
||||||
|
if exitErr, ok := err.(*exec.ExitError); ok && exitErr.ExitCode() == 104 {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
return nil, fmt.Errorf("zypper: listavailable: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return pkgs, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (z *Zypper) UpgradeAll(opts *Opts) error {
|
func (z *Zypper) UpgradeAll(opts *Opts) error {
|
||||||
opts = ensureOpts(opts)
|
opts = ensureOpts(opts)
|
||||||
cmd := z.getCmd(opts, "zypper", "update", "-y")
|
cmd := z.getCmd(opts, "zypper", "update", "-y")
|
||||||
|
|||||||
6
main.go
6
main.go
@@ -122,7 +122,7 @@ func GetApp() *cli.App {
|
|||||||
|
|
||||||
func setLogLevel(newLevel string) {
|
func setLogLevel(newLevel string) {
|
||||||
level := slog.LevelInfo
|
level := slog.LevelInfo
|
||||||
switch newLevel {
|
switch strings.ToUpper(newLevel) {
|
||||||
case "DEBUG":
|
case "DEBUG":
|
||||||
level = slog.LevelDebug
|
level = slog.LevelDebug
|
||||||
case "INFO":
|
case "INFO":
|
||||||
@@ -131,6 +131,10 @@ func setLogLevel(newLevel string) {
|
|||||||
level = slog.LevelWarn
|
level = slog.LevelWarn
|
||||||
case "ERROR":
|
case "ERROR":
|
||||||
level = slog.LevelError
|
level = slog.LevelError
|
||||||
|
default:
|
||||||
|
if newLevel != "" {
|
||||||
|
slog.Warn("unknown logLevel value, falling back to INFO", "value", newLevel)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
logger, ok := slog.Default().Handler().(*logger.Logger)
|
logger, ok := slog.Default().Handler().(*logger.Logger)
|
||||||
if !ok {
|
if !ok {
|
||||||
|
|||||||
@@ -172,14 +172,13 @@ func (s *ScriptFile) createPackageFromMeta(
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// DEBUG: Выводим что в metaRunner.Vars и dec.Runner.Vars для deps_debian
|
|
||||||
if depsDebianMeta, ok := metaRunner.Vars["deps_debian"]; ok {
|
if depsDebianMeta, ok := metaRunner.Vars["deps_debian"]; ok {
|
||||||
slog.Info("DEBUG createPackageFromMeta: metaRunner.Vars[deps_debian]", "value", depsDebianMeta.String(), "list", depsDebianMeta.List)
|
slog.Debug("createPackageFromMeta: metaRunner.Vars[deps_debian]", "value", depsDebianMeta.String(), "list", depsDebianMeta.List)
|
||||||
} else {
|
} else {
|
||||||
slog.Info("DEBUG createPackageFromMeta: metaRunner.Vars[deps_debian] NOT FOUND")
|
slog.Debug("createPackageFromMeta: metaRunner.Vars[deps_debian] NOT FOUND")
|
||||||
}
|
}
|
||||||
if depsDebianParent, ok := dec.Runner.Vars["deps_debian"]; ok {
|
if depsDebianParent, ok := dec.Runner.Vars["deps_debian"]; ok {
|
||||||
slog.Info("DEBUG createPackageFromMeta: parent Vars[deps_debian]", "value", depsDebianParent.String(), "list", depsDebianParent.List)
|
slog.Debug("createPackageFromMeta: parent Vars[deps_debian]", "value", depsDebianParent.String(), "list", depsDebianParent.List)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Сливаем переменные родительского runner'а с переменными мета-функции.
|
// Сливаем переменные родительского runner'а с переменными мета-функции.
|
||||||
|
|||||||
Reference in New Issue
Block a user