luet/cmd/root.go

182 lines
4.9 KiB
Go
Raw Normal View History

2019-10-29 16:36:48 +00:00
// 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"
2019-10-29 16:36:48 +00:00
"os"
"os/user"
2019-10-29 16:36:48 +00:00
"path/filepath"
"runtime"
2019-12-27 19:12:08 +00:00
"strings"
2019-10-29 16:36:48 +00:00
"github.com/marcsauter/single"
2019-12-27 19:12:08 +00:00
config "github.com/mudler/luet/pkg/config"
helpers "github.com/mudler/luet/pkg/helpers"
. "github.com/mudler/luet/pkg/logger"
2019-12-27 19:12:08 +00:00
repo "github.com/mudler/luet/pkg/repository"
2019-10-29 16:36:48 +00:00
"github.com/spf13/cobra"
"github.com/spf13/viper"
)
var cfgFile string
var Verbose bool
var LockedCommands = []string{"install", "uninstall", "upgrade"}
2019-10-29 16:36:48 +00:00
2019-12-27 19:12:08 +00:00
const (
2020-04-11 16:30:11 +00:00
LuetCLIVersion = "0.8-dev"
2019-12-27 19:12:08 +00:00
LuetEnvPrefix = "LUET"
)
2019-10-29 16:36:48 +00:00
// RootCmd represents the base command when called without any subcommands
var RootCmd = &cobra.Command{
Use: "luet",
Short: "Package manager for the XXth century!",
2019-11-17 18:45:20 +00:00
Long: `Package manager which uses containers to build packages`,
Version: LuetCLIVersion,
2019-12-27 19:12:08 +00:00
PersistentPreRun: func(cmd *cobra.Command, args []string) {
err := LoadConfig(config.LuetCfg)
if err != nil {
Fatal("failed to load configuration:", err.Error())
}
},
SilenceErrors: true,
2019-12-27 19:12:08 +00:00
}
func LoadConfig(c *config.LuetConfig) error {
// If a config file is found, read it in.
if err := c.Viper.ReadInConfig(); err != nil {
Debug(err)
2019-12-27 19:12:08 +00:00
}
err := c.Viper.Unmarshal(&config.LuetCfg)
if err != nil {
return err
}
Debug("Using config file:", c.Viper.ConfigFileUsed())
NewSpinner()
2019-12-27 19:12:08 +00:00
if c.GetLogging().Path != "" {
2020-01-03 14:22:55 +00:00
// Init zap logger
err = ZapLogger()
if err != nil {
return err
}
2019-12-27 19:12:08 +00:00
}
// Load repositories
err = repo.LoadRepositories(c)
if err != nil {
return err
}
return nil
2019-10-29 16:36:48 +00:00
}
// Execute adds all child commands to the root command sets flags appropriately.
// This is called by main.main(). It only needs to happen once to the rootCmd.
func Execute() {
2019-11-26 17:05:13 +00:00
if os.Getenv("LUET_NOLOCK") != "true" {
if len(os.Args) > 1 {
for _, lockedCmd := range LockedCommands {
if os.Args[1] == lockedCmd {
s := single.New("luet")
if err := s.CheckLock(); err != nil && err == single.ErrAlreadyRunning {
Fatal("another instance of the app is already running, exiting")
} else if err != nil {
// Another error occurred, might be worth handling it as well
Fatal("failed to acquire exclusive app lock:", err.Error())
}
defer s.TryUnlock()
break
}
}
2019-11-26 17:05:13 +00:00
}
}
2019-10-29 16:36:48 +00:00
if err := RootCmd.Execute(); err != nil {
if len(os.Args) > 0 {
for _, c := range RootCmd.Commands() {
if c.Name() == os.Args[1] {
os.Exit(-1) // Something failed
}
}
// Try to load a bin from path.
helpers.Exec("luet-"+os.Args[1], os.Args[1:], os.Environ())
}
fmt.Println(err)
2019-10-29 16:36:48 +00:00
os.Exit(-1)
}
}
func init() {
cobra.OnInitialize(initConfig)
pflags := RootCmd.PersistentFlags()
pflags.StringVar(&cfgFile, "config", "", "config file (default is $HOME/.luet.yaml)")
2020-03-06 15:53:36 +00:00
pflags.BoolP("debug", "d", false, "verbose output")
2020-01-03 14:41:45 +00:00
pflags.Bool("fatal", false, "Enables Warnings to exit")
sameOwner := false
u, err := user.Current()
// os/user doesn't work in from scratch environments
if err != nil {
Warning("failed to retrieve user identity:", err.Error())
sameOwner = true
}
if u != nil && u.Uid == "0" {
sameOwner = true
}
pflags.Bool("same-owner", sameOwner, "Maintain same owner on uncompress.")
pflags.Int("concurrency", runtime.NumCPU(), "Concurrency")
config.LuetCfg.Viper.BindPFlag("general.same_owner", pflags.Lookup("same-owner"))
2020-03-06 15:53:36 +00:00
config.LuetCfg.Viper.BindPFlag("general.debug", pflags.Lookup("debug"))
config.LuetCfg.Viper.BindPFlag("general.concurrency", pflags.Lookup("concurrency"))
2020-01-03 14:41:45 +00:00
config.LuetCfg.Viper.BindPFlag("general.fatal_warnings", pflags.Lookup("fatal"))
2019-10-29 16:36:48 +00:00
}
// initConfig reads in config file and ENV variables if set.
func initConfig() {
2019-10-29 16:36:48 +00:00
dir, err := filepath.Abs(filepath.Dir(os.Args[0]))
if err != nil {
Error(err)
os.Exit(1)
2019-10-29 16:36:48 +00:00
}
2019-12-27 19:12:08 +00:00
viper.SetEnvPrefix(LuetEnvPrefix)
viper.SetConfigType("yaml")
2019-10-29 16:36:48 +00:00
viper.SetConfigName(".luet") // name of config file (without extension)
if cfgFile != "" { // enable ability to specify config file via flag
2019-10-29 16:36:48 +00:00
viper.SetConfigFile(cfgFile)
2019-12-27 19:12:08 +00:00
} else {
viper.AddConfigPath(dir)
viper.AddConfigPath(".")
viper.AddConfigPath("$HOME")
viper.AddConfigPath("/etc/luet")
2019-10-29 16:36:48 +00:00
}
viper.AutomaticEnv() // read in environment variables that match
2019-12-27 19:12:08 +00:00
// Create EnvKey Replacer for handle complex structure
replacer := strings.NewReplacer(".", "__")
viper.SetEnvKeyReplacer(replacer)
viper.SetTypeByDefaultValue(true)
2019-10-29 16:36:48 +00:00
}