mirror of
https://github.com/mudler/luet.git
synced 2025-06-26 07:22:59 +00:00
This changeset also drops --keep-exported-images, which is quite unused and can be replaced with a plugin, or either by manually exporting the resulting images.
327 lines
11 KiB
Go
327 lines
11 KiB
Go
// Copyright © 2019 Ettore Di Giacinto <mudler@gentoo.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
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"github.com/ghodss/yaml"
|
|
helpers "github.com/mudler/luet/cmd/helpers"
|
|
"github.com/mudler/luet/pkg/compiler"
|
|
"github.com/mudler/luet/pkg/compiler/backend"
|
|
. "github.com/mudler/luet/pkg/config"
|
|
. "github.com/mudler/luet/pkg/logger"
|
|
pkg "github.com/mudler/luet/pkg/package"
|
|
"github.com/mudler/luet/pkg/solver"
|
|
tree "github.com/mudler/luet/pkg/tree"
|
|
|
|
"github.com/spf13/cobra"
|
|
"github.com/spf13/viper"
|
|
)
|
|
|
|
var buildCmd = &cobra.Command{
|
|
Use: "build <package name> <package name> <package name> ...",
|
|
Short: "build a package or a tree",
|
|
Long: `Builds one or more packages from a tree (current directory is implied):
|
|
|
|
$ luet build utils/busybox utils/yq ...
|
|
|
|
Builds all packages
|
|
|
|
$ luet build --all
|
|
|
|
Builds only the leaf packages:
|
|
|
|
$ luet build --full
|
|
|
|
Build package revdeps:
|
|
|
|
$ luet build --revdeps utils/yq
|
|
|
|
Build package without dependencies (needs the images already in the host, or either need to be available online):
|
|
|
|
$ luet build --nodeps utils/yq ...
|
|
|
|
Build packages specifying multiple definition trees:
|
|
|
|
$ luet build --tree overlay/path --tree overlay/path2 utils/yq ...
|
|
`, PreRun: func(cmd *cobra.Command, args []string) {
|
|
viper.BindPFlag("tree", cmd.Flags().Lookup("tree"))
|
|
viper.BindPFlag("destination", cmd.Flags().Lookup("destination"))
|
|
viper.BindPFlag("backend", cmd.Flags().Lookup("backend"))
|
|
viper.BindPFlag("privileged", cmd.Flags().Lookup("privileged"))
|
|
viper.BindPFlag("database", cmd.Flags().Lookup("database"))
|
|
viper.BindPFlag("revdeps", cmd.Flags().Lookup("revdeps"))
|
|
viper.BindPFlag("all", cmd.Flags().Lookup("all"))
|
|
viper.BindPFlag("compression", cmd.Flags().Lookup("compression"))
|
|
viper.BindPFlag("nodeps", cmd.Flags().Lookup("nodeps"))
|
|
viper.BindPFlag("onlydeps", cmd.Flags().Lookup("onlydeps"))
|
|
viper.BindPFlag("values", cmd.Flags().Lookup("values"))
|
|
|
|
viper.BindPFlag("image-repository", cmd.Flags().Lookup("image-repository"))
|
|
viper.BindPFlag("push", cmd.Flags().Lookup("push"))
|
|
viper.BindPFlag("pull", cmd.Flags().Lookup("pull"))
|
|
viper.BindPFlag("wait", cmd.Flags().Lookup("wait"))
|
|
viper.BindPFlag("keep-images", cmd.Flags().Lookup("keep-images"))
|
|
|
|
LuetCfg.Viper.BindPFlag("solver.type", cmd.Flags().Lookup("solver-type"))
|
|
LuetCfg.Viper.BindPFlag("solver.discount", cmd.Flags().Lookup("solver-discount"))
|
|
LuetCfg.Viper.BindPFlag("solver.rate", cmd.Flags().Lookup("solver-rate"))
|
|
LuetCfg.Viper.BindPFlag("solver.max_attempts", cmd.Flags().Lookup("solver-attempts"))
|
|
},
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
|
|
treePaths := viper.GetStringSlice("tree")
|
|
dst := viper.GetString("destination")
|
|
concurrency := LuetCfg.GetGeneral().Concurrency
|
|
backendType := viper.GetString("backend")
|
|
privileged := viper.GetBool("privileged")
|
|
revdeps := viper.GetBool("revdeps")
|
|
all := viper.GetBool("all")
|
|
databaseType := viper.GetString("database")
|
|
compressionType := viper.GetString("compression")
|
|
imageRepository := viper.GetString("image-repository")
|
|
values := viper.GetString("values")
|
|
wait := viper.GetBool("wait")
|
|
push := viper.GetBool("push")
|
|
pull := viper.GetBool("pull")
|
|
keepImages := viper.GetBool("keep-images")
|
|
nodeps := viper.GetBool("nodeps")
|
|
onlydeps := viper.GetBool("onlydeps")
|
|
onlyTarget, _ := cmd.Flags().GetBool("only-target-package")
|
|
full, _ := cmd.Flags().GetBool("full")
|
|
concurrent, _ := cmd.Flags().GetBool("solver-concurrent")
|
|
var results Results
|
|
|
|
out, _ := cmd.Flags().GetString("output")
|
|
if out != "terminal" {
|
|
LuetCfg.GetLogging().SetLogLevel("error")
|
|
}
|
|
pretend, _ := cmd.Flags().GetBool("pretend")
|
|
compilerSpecs := compiler.NewLuetCompilationspecs()
|
|
var db pkg.PackageDatabase
|
|
|
|
compilerBackend := backend.NewBackend(backendType)
|
|
|
|
switch databaseType {
|
|
case "memory":
|
|
db = pkg.NewInMemoryDatabase(false)
|
|
|
|
case "boltdb":
|
|
tmpdir, err := ioutil.TempDir("", "package")
|
|
if err != nil {
|
|
Fatal(err)
|
|
}
|
|
db = pkg.NewBoltDatabase(tmpdir)
|
|
|
|
}
|
|
defer db.Clean()
|
|
|
|
generalRecipe := tree.NewCompilerRecipe(db)
|
|
|
|
for _, src := range treePaths {
|
|
Info("Loading tree", src)
|
|
|
|
err := generalRecipe.Load(src)
|
|
if err != nil {
|
|
Fatal("Error: " + err.Error())
|
|
}
|
|
}
|
|
|
|
Info("Building in", dst)
|
|
|
|
stype := LuetCfg.Viper.GetString("solver.type")
|
|
discount := LuetCfg.Viper.GetFloat64("solver.discount")
|
|
rate := LuetCfg.Viper.GetFloat64("solver.rate")
|
|
attempts := LuetCfg.Viper.GetInt("solver.max_attempts")
|
|
|
|
LuetCfg.GetSolverOptions().Type = stype
|
|
LuetCfg.GetSolverOptions().LearnRate = float32(rate)
|
|
LuetCfg.GetSolverOptions().Discount = float32(discount)
|
|
LuetCfg.GetSolverOptions().MaxAttempts = attempts
|
|
|
|
Debug("Solver", LuetCfg.GetSolverOptions().CompactString())
|
|
|
|
opts := compiler.NewDefaultCompilerOptions()
|
|
opts.SolverOptions = *LuetCfg.GetSolverOptions()
|
|
opts.ImageRepository = imageRepository
|
|
opts.PullFirst = pull
|
|
opts.KeepImg = keepImages
|
|
opts.Push = push
|
|
opts.OnlyDeps = onlydeps
|
|
opts.NoDeps = nodeps
|
|
opts.Wait = wait
|
|
opts.PackageTargetOnly = onlyTarget
|
|
opts.BuildValuesFile = values
|
|
var solverOpts solver.Options
|
|
if concurrent {
|
|
solverOpts = solver.Options{Type: solver.ParallelSimple, Concurrency: concurrency}
|
|
} else {
|
|
solverOpts = solver.Options{Type: solver.SingleCoreSimple, Concurrency: concurrency}
|
|
}
|
|
|
|
luetCompiler := compiler.NewLuetCompiler(compilerBackend, generalRecipe.GetDatabase(), opts, solverOpts)
|
|
luetCompiler.SetConcurrency(concurrency)
|
|
luetCompiler.SetCompressionType(compiler.CompressionImplementation(compressionType))
|
|
if full {
|
|
specs, err := luetCompiler.FromDatabase(generalRecipe.GetDatabase(), true, dst)
|
|
if err != nil {
|
|
Fatal(err.Error())
|
|
}
|
|
for _, spec := range specs {
|
|
Info(":package: Selecting ", spec.GetPackage().GetName(), spec.GetPackage().GetVersion())
|
|
|
|
compilerSpecs.Add(spec)
|
|
}
|
|
} else if !all {
|
|
for _, a := range args {
|
|
|
|
pack, err := helpers.ParsePackageStr(a)
|
|
if err != nil {
|
|
Fatal("Invalid package string ", a, ": ", err.Error())
|
|
}
|
|
|
|
spec, err := luetCompiler.FromPackage(pack)
|
|
if err != nil {
|
|
Fatal("Error: " + err.Error())
|
|
}
|
|
|
|
spec.SetOutputPath(dst)
|
|
compilerSpecs.Add(spec)
|
|
}
|
|
} else {
|
|
w := generalRecipe.GetDatabase().World()
|
|
|
|
for _, p := range w {
|
|
spec, err := luetCompiler.FromPackage(p)
|
|
if err != nil {
|
|
Fatal("Error: " + err.Error())
|
|
}
|
|
Info(":package: Selecting ", p.GetName(), p.GetVersion())
|
|
spec.SetOutputPath(dst)
|
|
compilerSpecs.Add(spec)
|
|
}
|
|
}
|
|
|
|
var artifact []compiler.Artifact
|
|
var errs []error
|
|
if revdeps {
|
|
artifact, errs = luetCompiler.CompileWithReverseDeps(privileged, compilerSpecs)
|
|
|
|
} else if pretend {
|
|
toCalculate := []compiler.CompilationSpec{}
|
|
if full {
|
|
var err error
|
|
toCalculate, err = luetCompiler.ComputeMinimumCompilableSet(compilerSpecs.All()...)
|
|
if err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
} else {
|
|
toCalculate = compilerSpecs.All()
|
|
}
|
|
|
|
for _, sp := range toCalculate {
|
|
packs, err := luetCompiler.ComputeDepTree(sp)
|
|
if err != nil {
|
|
errs = append(errs, err)
|
|
}
|
|
for _, p := range packs {
|
|
results.Packages = append(results.Packages,
|
|
PackageResult{
|
|
Name: p.Package.GetName(),
|
|
Version: p.Package.GetVersion(),
|
|
Category: p.Package.GetCategory(),
|
|
Repository: "",
|
|
Hidden: p.Package.IsHidden(),
|
|
Target: sp.GetPackage().HumanReadableString(),
|
|
})
|
|
}
|
|
}
|
|
|
|
y, err := yaml.Marshal(results)
|
|
if err != nil {
|
|
fmt.Printf("err: %v\n", err)
|
|
return
|
|
}
|
|
switch out {
|
|
case "yaml":
|
|
fmt.Println(string(y))
|
|
case "json":
|
|
j2, err := yaml.YAMLToJSON(y)
|
|
if err != nil {
|
|
fmt.Printf("err: %v\n", err)
|
|
return
|
|
}
|
|
fmt.Println(string(j2))
|
|
case "terminal":
|
|
for _, p := range results.Packages {
|
|
Info(p.String())
|
|
}
|
|
}
|
|
} else {
|
|
artifact, errs = luetCompiler.CompileParallel(privileged, compilerSpecs)
|
|
}
|
|
if len(errs) != 0 {
|
|
for _, e := range errs {
|
|
Error("Error: " + e.Error())
|
|
}
|
|
Fatal("Bailing out")
|
|
}
|
|
for _, a := range artifact {
|
|
Info("Artifact generated:", a.GetPath())
|
|
}
|
|
},
|
|
}
|
|
|
|
func init() {
|
|
path, err := os.Getwd()
|
|
if err != nil {
|
|
Fatal(err)
|
|
}
|
|
buildCmd.Flags().StringSliceP("tree", "t", []string{path}, "Path of the tree to use.")
|
|
buildCmd.Flags().String("backend", "docker", "backend used (docker,img)")
|
|
buildCmd.Flags().Bool("privileged", false, "Privileged (Keep permissions)")
|
|
buildCmd.Flags().String("database", "memory", "database used for solving (memory,boltdb)")
|
|
buildCmd.Flags().Bool("revdeps", false, "Build with revdeps")
|
|
buildCmd.Flags().Bool("all", false, "Build all specfiles in the tree")
|
|
buildCmd.Flags().Bool("full", false, "Build all packages (optimized)")
|
|
buildCmd.Flags().String("values", "", "Build values file to interpolate with each package")
|
|
|
|
buildCmd.Flags().String("destination", filepath.Join(path, "build"), "Destination folder")
|
|
buildCmd.Flags().String("compression", "none", "Compression alg: none, gzip, zstd")
|
|
buildCmd.Flags().String("image-repository", "luet/cache", "Default base image string for generated image")
|
|
buildCmd.Flags().Bool("push", false, "Push images to a hub")
|
|
buildCmd.Flags().Bool("pull", false, "Pull images from a hub")
|
|
buildCmd.Flags().Bool("wait", false, "Don't build all intermediate images, but wait for them until they are available")
|
|
buildCmd.Flags().Bool("keep-images", true, "Keep built docker images in the host")
|
|
buildCmd.Flags().Bool("nodeps", false, "Build only the target packages, skipping deps (it works only if you already built the deps locally, or by using --pull) ")
|
|
buildCmd.Flags().Bool("onlydeps", false, "Build only package dependencies")
|
|
buildCmd.Flags().Bool("only-target-package", false, "Build packages of only the required target. Otherwise builds all the necessary ones not present in the destination")
|
|
buildCmd.Flags().String("solver-type", "", "Solver strategy")
|
|
buildCmd.Flags().Float32("solver-rate", 0.7, "Solver learning rate")
|
|
buildCmd.Flags().Float32("solver-discount", 1.0, "Solver discount rate")
|
|
buildCmd.Flags().Int("solver-attempts", 9000, "Solver maximum attempts")
|
|
buildCmd.Flags().Bool("solver-concurrent", false, "Use concurrent solver (experimental)")
|
|
|
|
buildCmd.Flags().Bool("pretend", false, "Just print what packages will be compiled")
|
|
|
|
buildCmd.Flags().StringP("output", "o", "terminal", "Output format ( Defaults: terminal, available: json,yaml )")
|
|
|
|
RootCmd.AddCommand(buildCmd)
|
|
}
|