mirror of
https://github.com/mudler/luet.git
synced 2025-04-29 12:04:17 +00:00
This refactors DefaultPackage into types.Package and gets rid of the interface. This is a preceeding for a follow up where accessors will be removed from the code. It also does several cleanup, so we get rid also of some unneeded dependencies.
503 lines
13 KiB
Go
503 lines
13 KiB
Go
// Copyright © 2020 Ettore Di Giacinto <mudler@gentoo.org>
|
|
// Daniele Rondina <geaaru@sabayonlinux.org>
|
|
//
|
|
// 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 2 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 cmd_tree
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"regexp"
|
|
"sort"
|
|
"strconv"
|
|
"sync"
|
|
|
|
"github.com/mudler/luet/pkg/api/core/types"
|
|
|
|
helpers "github.com/mudler/luet/cmd/helpers"
|
|
"github.com/mudler/luet/cmd/util"
|
|
|
|
pkg "github.com/mudler/luet/pkg/database"
|
|
"github.com/mudler/luet/pkg/solver"
|
|
tree "github.com/mudler/luet/pkg/tree"
|
|
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
type ValidateOpts struct {
|
|
WithSolver bool
|
|
OnlyRuntime bool
|
|
OnlyBuildtime bool
|
|
RegExcludes []*regexp.Regexp
|
|
RegMatches []*regexp.Regexp
|
|
Excludes []string
|
|
Matches []string
|
|
|
|
// Runtime validate stuff
|
|
RuntimeCacheDeps *pkg.InMemoryDatabase
|
|
RuntimeReciper *tree.InstallerRecipe
|
|
|
|
// Buildtime validate stuff
|
|
BuildtimeCacheDeps *pkg.InMemoryDatabase
|
|
BuildtimeReciper *tree.CompilerRecipe
|
|
|
|
Mutex sync.Mutex
|
|
BrokenPkgs int
|
|
BrokenDeps int
|
|
|
|
Errors []error
|
|
}
|
|
|
|
func (o *ValidateOpts) IncrBrokenPkgs() {
|
|
o.Mutex.Lock()
|
|
defer o.Mutex.Unlock()
|
|
o.BrokenPkgs++
|
|
}
|
|
|
|
func (o *ValidateOpts) IncrBrokenDeps() {
|
|
o.Mutex.Lock()
|
|
defer o.Mutex.Unlock()
|
|
o.BrokenDeps++
|
|
}
|
|
|
|
func (o *ValidateOpts) AddError(err error) {
|
|
o.Mutex.Lock()
|
|
defer o.Mutex.Unlock()
|
|
o.Errors = append(o.Errors, err)
|
|
}
|
|
|
|
func validatePackage(p *types.Package, checkType string, opts *ValidateOpts, reciper tree.Builder, cacheDeps *pkg.InMemoryDatabase) error {
|
|
var errstr string
|
|
var ans error
|
|
|
|
var depSolver types.PackageSolver
|
|
|
|
if opts.WithSolver {
|
|
emptyInstallationDb := pkg.NewInMemoryDatabase(false)
|
|
depSolver = solver.NewSolver(types.SolverOptions{Type: types.SolverSingleCoreSimple}, pkg.NewInMemoryDatabase(false),
|
|
reciper.GetDatabase(),
|
|
emptyInstallationDb)
|
|
}
|
|
|
|
found, err := reciper.GetDatabase().FindPackages(
|
|
&types.Package{
|
|
Name: p.GetName(),
|
|
Category: p.GetCategory(),
|
|
Version: ">=0",
|
|
},
|
|
)
|
|
|
|
if err != nil || len(found) < 1 {
|
|
if err != nil {
|
|
errstr = err.Error()
|
|
} else {
|
|
errstr = "No packages"
|
|
}
|
|
util.DefaultContext.Error(fmt.Sprintf("[%9s] %s/%s-%s: Broken. No versions could be found by database %s",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
errstr,
|
|
))
|
|
|
|
opts.IncrBrokenDeps()
|
|
|
|
return errors.New(
|
|
fmt.Sprintf("[%9s] %s/%s-%s: Broken. No versions could be found by database %s",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
errstr,
|
|
))
|
|
}
|
|
|
|
// Ensure that we use the right package from right recipier for deps
|
|
pReciper, err := reciper.GetDatabase().FindPackage(
|
|
&types.Package{
|
|
Name: p.GetName(),
|
|
Category: p.GetCategory(),
|
|
Version: p.GetVersion(),
|
|
},
|
|
)
|
|
if err != nil {
|
|
errstr = fmt.Sprintf("[%9s] %s/%s-%s: Error on retrieve package - %s.",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
err.Error(),
|
|
)
|
|
util.DefaultContext.Error(errstr)
|
|
|
|
return errors.New(errstr)
|
|
}
|
|
p = pReciper
|
|
|
|
pkgstr := fmt.Sprintf("%s/%s-%s", p.GetCategory(), p.GetName(),
|
|
p.GetVersion())
|
|
|
|
validpkg := true
|
|
|
|
if len(opts.Matches) > 0 {
|
|
matched := false
|
|
for _, rgx := range opts.RegMatches {
|
|
if rgx.MatchString(pkgstr) {
|
|
matched = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if !matched {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
if len(opts.Excludes) > 0 {
|
|
excluded := false
|
|
for _, rgx := range opts.RegExcludes {
|
|
if rgx.MatchString(pkgstr) {
|
|
excluded = true
|
|
break
|
|
}
|
|
}
|
|
|
|
if excluded {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
util.DefaultContext.Info(fmt.Sprintf("[%9s] Checking package ", checkType)+
|
|
fmt.Sprintf("%s/%s-%s", p.GetCategory(), p.GetName(), p.GetVersion()),
|
|
"with", len(p.GetRequires()), "dependencies and", len(p.GetConflicts()), "conflicts.")
|
|
|
|
all := p.GetRequires()
|
|
all = append(all, p.GetConflicts()...)
|
|
for idx, r := range all {
|
|
|
|
var deps types.Packages
|
|
var err error
|
|
if r.IsSelector() {
|
|
deps, err = reciper.GetDatabase().FindPackages(
|
|
&types.Package{
|
|
Name: r.GetName(),
|
|
Category: r.GetCategory(),
|
|
Version: r.GetVersion(),
|
|
},
|
|
)
|
|
} else {
|
|
deps = append(deps, r)
|
|
}
|
|
|
|
if err != nil || len(deps) < 1 {
|
|
if err != nil {
|
|
errstr = err.Error()
|
|
} else {
|
|
errstr = "No packages"
|
|
}
|
|
util.DefaultContext.Error(fmt.Sprintf("[%9s] %s/%s-%s: Broken Dep %s/%s-%s - %s",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
r.GetCategory(), r.GetName(), r.GetVersion(),
|
|
errstr,
|
|
))
|
|
|
|
opts.IncrBrokenDeps()
|
|
|
|
ans = errors.New(
|
|
fmt.Sprintf("[%9s] %s/%s-%s: Broken Dep %s/%s-%s - %s",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
r.GetCategory(), r.GetName(), r.GetVersion(),
|
|
errstr))
|
|
|
|
validpkg = false
|
|
|
|
} else {
|
|
|
|
util.DefaultContext.Debug(fmt.Sprintf("[%9s] Find packages for dep", checkType),
|
|
fmt.Sprintf("%s/%s-%s", r.GetCategory(), r.GetName(), r.GetVersion()))
|
|
|
|
if opts.WithSolver {
|
|
|
|
util.DefaultContext.Info(fmt.Sprintf("[%9s] :soap: [%2d/%2d] %s/%s-%s: %s/%s-%s",
|
|
checkType,
|
|
idx+1, len(all),
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
r.GetCategory(), r.GetName(), r.GetVersion(),
|
|
))
|
|
|
|
// Check if the solver is already been done for the deep
|
|
_, err := cacheDeps.Get(r.HashFingerprint(""))
|
|
if err == nil {
|
|
util.DefaultContext.Debug(fmt.Sprintf("[%9s] :direct_hit: Cache Hit for dep", checkType),
|
|
fmt.Sprintf("%s/%s-%s", r.GetCategory(), r.GetName(), r.GetVersion()))
|
|
continue
|
|
}
|
|
|
|
util.DefaultContext.Spinner()
|
|
solution, err := depSolver.Install(types.Packages{r})
|
|
ass := solution.SearchByName(r.GetPackageName())
|
|
util.DefaultContext.SpinnerStop()
|
|
if err == nil {
|
|
if ass == nil {
|
|
|
|
ans = errors.New(
|
|
fmt.Sprintf("[%9s] %s/%s-%s: solution doesn't retrieve package %s/%s-%s.",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
r.GetCategory(), r.GetName(), r.GetVersion(),
|
|
))
|
|
|
|
if util.DefaultContext.Config.General.Debug {
|
|
for idx, pa := range solution {
|
|
fmt.Println(fmt.Sprintf("[%9s] %s/%s-%s: solution %d: %s",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(), idx,
|
|
pa.Package.GetPackageName()))
|
|
}
|
|
}
|
|
|
|
util.DefaultContext.Error(ans.Error())
|
|
opts.IncrBrokenDeps()
|
|
validpkg = false
|
|
} else {
|
|
_, err = solution.Order(reciper.GetDatabase(), ass.Package.GetFingerPrint())
|
|
}
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
util.DefaultContext.Error(fmt.Sprintf("[%9s] %s/%s-%s: solver broken for dep %s/%s-%s - %s",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
r.GetCategory(), r.GetName(), r.GetVersion(),
|
|
err.Error(),
|
|
))
|
|
|
|
ans = errors.New(
|
|
fmt.Sprintf("[%9s] %s/%s-%s: solver broken for Dep %s/%s-%s - %s",
|
|
checkType,
|
|
p.GetCategory(), p.GetName(), p.GetVersion(),
|
|
r.GetCategory(), r.GetName(), r.GetVersion(),
|
|
err.Error()))
|
|
|
|
opts.IncrBrokenDeps()
|
|
validpkg = false
|
|
}
|
|
|
|
// Register the key
|
|
cacheDeps.Set(r.HashFingerprint(""), "1")
|
|
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
if !validpkg {
|
|
opts.IncrBrokenPkgs()
|
|
}
|
|
|
|
return ans
|
|
}
|
|
|
|
func validateWorker(i int,
|
|
wg *sync.WaitGroup,
|
|
c <-chan *types.Package,
|
|
opts *ValidateOpts) {
|
|
|
|
defer wg.Done()
|
|
|
|
for p := range c {
|
|
|
|
if opts.OnlyBuildtime {
|
|
// Check buildtime compiler/deps
|
|
err := validatePackage(p, "buildtime", opts, opts.BuildtimeReciper, opts.BuildtimeCacheDeps)
|
|
if err != nil {
|
|
opts.AddError(err)
|
|
continue
|
|
}
|
|
} else if opts.OnlyRuntime {
|
|
|
|
// Check runtime installer/deps
|
|
err := validatePackage(p, "runtime", opts, opts.RuntimeReciper, opts.RuntimeCacheDeps)
|
|
if err != nil {
|
|
opts.AddError(err)
|
|
continue
|
|
}
|
|
|
|
} else {
|
|
|
|
// Check runtime installer/deps
|
|
err := validatePackage(p, "runtime", opts, opts.RuntimeReciper, opts.RuntimeCacheDeps)
|
|
if err != nil {
|
|
opts.AddError(err)
|
|
continue
|
|
}
|
|
|
|
// Check buildtime compiler/deps
|
|
err = validatePackage(p, "buildtime", opts, opts.BuildtimeReciper, opts.BuildtimeCacheDeps)
|
|
if err != nil {
|
|
opts.AddError(err)
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
func initOpts(opts *ValidateOpts, onlyRuntime, onlyBuildtime, withSolver bool, treePaths []string) {
|
|
var err error
|
|
|
|
opts.OnlyBuildtime = onlyBuildtime
|
|
opts.OnlyRuntime = onlyRuntime
|
|
opts.WithSolver = withSolver
|
|
opts.RuntimeReciper = nil
|
|
opts.BuildtimeReciper = nil
|
|
opts.BrokenPkgs = 0
|
|
opts.BrokenDeps = 0
|
|
|
|
if onlyBuildtime {
|
|
opts.BuildtimeReciper = (tree.NewCompilerRecipe(pkg.NewInMemoryDatabase(false))).(*tree.CompilerRecipe)
|
|
} else if onlyRuntime {
|
|
opts.RuntimeReciper = (tree.NewInstallerRecipe(pkg.NewInMemoryDatabase(false))).(*tree.InstallerRecipe)
|
|
} else {
|
|
opts.BuildtimeReciper = (tree.NewCompilerRecipe(pkg.NewInMemoryDatabase(false))).(*tree.CompilerRecipe)
|
|
opts.RuntimeReciper = (tree.NewInstallerRecipe(pkg.NewInMemoryDatabase(false))).(*tree.InstallerRecipe)
|
|
}
|
|
|
|
opts.RuntimeCacheDeps = pkg.NewInMemoryDatabase(false).(*pkg.InMemoryDatabase)
|
|
opts.BuildtimeCacheDeps = pkg.NewInMemoryDatabase(false).(*pkg.InMemoryDatabase)
|
|
|
|
for _, treePath := range treePaths {
|
|
util.DefaultContext.Info(fmt.Sprintf("Loading :deciduous_tree: %s...", treePath))
|
|
if opts.BuildtimeReciper != nil {
|
|
err = opts.BuildtimeReciper.Load(treePath)
|
|
if err != nil {
|
|
util.DefaultContext.Fatal("Error on load tree ", err)
|
|
}
|
|
}
|
|
if opts.RuntimeReciper != nil {
|
|
err = opts.RuntimeReciper.Load(treePath)
|
|
if err != nil {
|
|
util.DefaultContext.Fatal("Error on load tree ", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
opts.RegExcludes, err = helpers.CreateRegexArray(opts.Excludes)
|
|
if err != nil {
|
|
util.DefaultContext.Fatal(err.Error())
|
|
}
|
|
opts.RegMatches, err = helpers.CreateRegexArray(opts.Matches)
|
|
if err != nil {
|
|
util.DefaultContext.Fatal(err.Error())
|
|
}
|
|
|
|
}
|
|
|
|
func NewTreeValidateCommand() *cobra.Command {
|
|
var excludes []string
|
|
var matches []string
|
|
var treePaths []string
|
|
var opts ValidateOpts
|
|
|
|
var ans = &cobra.Command{
|
|
Use: "validate [OPTIONS]",
|
|
Short: "Validate a tree or a list of packages",
|
|
Args: cobra.OnlyValidArgs,
|
|
PreRun: func(cmd *cobra.Command, args []string) {
|
|
onlyRuntime, _ := cmd.Flags().GetBool("only-runtime")
|
|
onlyBuildtime, _ := cmd.Flags().GetBool("only-buildtime")
|
|
|
|
if len(treePaths) < 1 {
|
|
util.DefaultContext.Fatal("Mandatory tree param missing.")
|
|
}
|
|
if onlyRuntime && onlyBuildtime {
|
|
util.DefaultContext.Fatal("Both --only-runtime and --only-buildtime options are not possibile.")
|
|
}
|
|
},
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
var reciper tree.Builder
|
|
|
|
concurrency := util.DefaultContext.Config.General.Concurrency
|
|
|
|
withSolver, _ := cmd.Flags().GetBool("with-solver")
|
|
onlyRuntime, _ := cmd.Flags().GetBool("only-runtime")
|
|
onlyBuildtime, _ := cmd.Flags().GetBool("only-buildtime")
|
|
|
|
opts.Excludes = excludes
|
|
opts.Matches = matches
|
|
initOpts(&opts, onlyRuntime, onlyBuildtime, withSolver, treePaths)
|
|
|
|
// We need at least one valid reciper for get list of the packages.
|
|
if onlyBuildtime {
|
|
reciper = opts.BuildtimeReciper
|
|
} else {
|
|
reciper = opts.RuntimeReciper
|
|
}
|
|
|
|
all := make(chan *types.Package)
|
|
|
|
var wg = new(sync.WaitGroup)
|
|
|
|
for i := 0; i < concurrency; i++ {
|
|
wg.Add(1)
|
|
go validateWorker(i, wg, all, &opts)
|
|
}
|
|
for _, p := range reciper.GetDatabase().World() {
|
|
all <- p
|
|
}
|
|
close(all)
|
|
|
|
// Wait separately and once done close the channel
|
|
go func() {
|
|
wg.Wait()
|
|
}()
|
|
|
|
stringerrs := []string{}
|
|
for _, e := range opts.Errors {
|
|
stringerrs = append(stringerrs, e.Error())
|
|
}
|
|
sort.Strings(stringerrs)
|
|
for _, e := range stringerrs {
|
|
fmt.Println(e)
|
|
}
|
|
|
|
// fmt.Println("Broken packages:", brokenPkgs, "(", brokenDeps, "deps ).")
|
|
if len(stringerrs) != 0 {
|
|
util.DefaultContext.Error(fmt.Sprintf("Found %d broken packages and %d broken deps.",
|
|
opts.BrokenPkgs, opts.BrokenDeps))
|
|
util.DefaultContext.Fatal("Errors: " + strconv.Itoa(len(stringerrs)))
|
|
} else {
|
|
util.DefaultContext.Info("All good! :white_check_mark:")
|
|
os.Exit(0)
|
|
}
|
|
},
|
|
}
|
|
path, err := os.Getwd()
|
|
if err != nil {
|
|
util.DefaultContext.Fatal(err)
|
|
}
|
|
ans.Flags().Bool("only-runtime", false, "Check only runtime dependencies.")
|
|
ans.Flags().Bool("only-buildtime", false, "Check only buildtime dependencies.")
|
|
ans.Flags().BoolP("with-solver", "s", false,
|
|
"Enable check of requires also with solver.")
|
|
ans.Flags().StringSliceVarP(&treePaths, "tree", "t", []string{path},
|
|
"Path of the tree to use.")
|
|
ans.Flags().StringSliceVarP(&excludes, "exclude", "e", []string{},
|
|
"Exclude matched packages from analysis. (Use string as regex).")
|
|
ans.Flags().StringSliceVarP(&matches, "matches", "m", []string{},
|
|
"Analyze only matched packages. (Use string as regex).")
|
|
|
|
return ans
|
|
}
|