123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471 |
- package build
- import (
- "context"
- "fmt"
- "os"
- "github.com/Jguer/yay/v12/pkg/db"
- "github.com/Jguer/yay/v12/pkg/dep"
- "github.com/Jguer/yay/v12/pkg/multierror"
- "github.com/Jguer/yay/v12/pkg/settings"
- "github.com/Jguer/yay/v12/pkg/settings/exe"
- "github.com/Jguer/yay/v12/pkg/settings/parser"
- "github.com/Jguer/yay/v12/pkg/text"
- "github.com/Jguer/yay/v12/pkg/vcs"
- mapset "github.com/deckarep/golang-set/v2"
- "github.com/leonelquinteros/gotext"
- )
- type (
- PostInstallHookFunc func(ctx context.Context) error
- Installer struct {
- dbExecutor db.Executor
- postInstallHooks []PostInstallHookFunc
- failedAndIgnored map[string]error
- exeCmd exe.ICmdBuilder
- vcsStore vcs.Store
- targetMode parser.TargetMode
- rebuildMode parser.RebuildMode
- origTargets mapset.Set[string]
- downloadOnly bool
- log *text.Logger
- manualConfirmRequired bool
- }
- )
- func NewInstaller(dbExecutor db.Executor,
- exeCmd exe.ICmdBuilder, vcsStore vcs.Store, targetMode parser.TargetMode,
- rebuildMode parser.RebuildMode, downloadOnly bool, logger *text.Logger,
- ) *Installer {
- return &Installer{
- dbExecutor: dbExecutor,
- postInstallHooks: []PostInstallHookFunc{},
- failedAndIgnored: map[string]error{},
- exeCmd: exeCmd,
- vcsStore: vcsStore,
- targetMode: targetMode,
- rebuildMode: rebuildMode,
- downloadOnly: downloadOnly,
- log: logger,
- manualConfirmRequired: true,
- }
- }
- func (installer *Installer) CompileFailedAndIgnored() (map[string]error, error) {
- if len(installer.failedAndIgnored) == 0 {
- return installer.failedAndIgnored, nil
- }
- return installer.failedAndIgnored, &FailedIgnoredPkgError{
- pkgErrors: installer.failedAndIgnored,
- }
- }
- func (installer *Installer) AddPostInstallHook(hook PostInstallHookFunc) {
- if hook == nil {
- return
- }
- installer.postInstallHooks = append(installer.postInstallHooks, hook)
- }
- func (installer *Installer) RunPostInstallHooks(ctx context.Context) error {
- var errMulti multierror.MultiError
- for _, hook := range installer.postInstallHooks {
- if err := hook(ctx); err != nil {
- errMulti.Add(err)
- }
- }
- return errMulti.Return()
- }
- func (installer *Installer) Install(ctx context.Context,
- cmdArgs *parser.Arguments,
- targets []map[string]*dep.InstallInfo,
- pkgBuildDirs map[string]string,
- excluded []string,
- manualConfirmRequired bool,
- ) error {
- installer.log.Debugln("manualConfirmRequired:", manualConfirmRequired)
- installer.manualConfirmRequired = manualConfirmRequired
- installer.origTargets = mapset.NewThreadUnsafeSet[string]()
- for _, targetString := range cmdArgs.Targets {
- installer.origTargets.Add(dep.ToTarget(targetString).Name)
- }
- installer.log.Debugln("origTargets:", installer.origTargets)
- // Reorganize targets into layers of dependencies
- var errMulti multierror.MultiError
- for i := len(targets) - 1; i >= 0; i-- {
- lastLayer := i == 0
- errI := installer.handleLayer(ctx, cmdArgs, targets[i], pkgBuildDirs, lastLayer, excluded)
- if errI == nil && lastLayer {
- // success after rollups
- return nil
- }
- if errI != nil {
- errMulti.Add(errI)
- if lastLayer {
- break
- }
- // rollup
- installer.log.Warnln(gotext.Get("Failed to install layer, rolling up to next layer."), "error:", errI)
- targets[i-1] = mergeLayers(targets[i-1], targets[i])
- }
- }
- return errMulti.Return()
- }
- func mergeLayers(layer1, layer2 map[string]*dep.InstallInfo) map[string]*dep.InstallInfo {
- for name, info := range layer2 {
- layer1[name] = info
- }
- return layer1
- }
- func (installer *Installer) appendNoConfirm() bool {
- return !installer.manualConfirmRequired || settings.NoConfirm
- }
- func (installer *Installer) handleLayer(ctx context.Context,
- cmdArgs *parser.Arguments,
- layer map[string]*dep.InstallInfo,
- pkgBuildDirs map[string]string,
- lastLayer bool,
- excluded []string,
- ) error {
- // Install layer
- nameToBaseMap := make(map[string]string, 0)
- syncDeps, syncExp, syncGroups := mapset.NewThreadUnsafeSet[string](),
- mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
- aurDeps, aurExp, aurOrigTargetBases := mapset.NewThreadUnsafeSet[string](),
- mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
- upgradeSync := false
- for name, info := range layer {
- switch info.Source {
- case dep.AUR, dep.SrcInfo:
- nameToBaseMap[name] = *info.AURBase
- if installer.origTargets.Contains(name) {
- aurOrigTargetBases.Add(*info.AURBase)
- }
- switch info.Reason {
- case dep.Explicit:
- if cmdArgs.ExistsArg("asdeps", "asdep") {
- aurDeps.Add(name)
- } else {
- aurExp.Add(name)
- }
- case dep.Dep, dep.MakeDep, dep.CheckDep:
- aurDeps.Add(name)
- }
- case dep.Sync:
- if info.Upgrade {
- upgradeSync = true
- continue // do not add to targets, let pacman handle it
- }
- compositePkgName := fmt.Sprintf("%s/%s", *info.SyncDBName, name)
- if info.IsGroup {
- syncGroups.Add(compositePkgName)
- continue
- }
- switch info.Reason {
- case dep.Explicit:
- if cmdArgs.ExistsArg("asdeps", "asdep") {
- syncDeps.Add(compositePkgName)
- } else {
- syncExp.Add(compositePkgName)
- }
- case dep.Dep, dep.MakeDep, dep.CheckDep:
- syncDeps.Add(compositePkgName)
- }
- }
- }
- installer.log.Debugln("syncDeps", syncDeps, "SyncExp", syncExp,
- "aurDeps", aurDeps, "aurExp", aurExp, "upgrade", upgradeSync)
- errShow := installer.installSyncPackages(ctx, cmdArgs, syncDeps, syncExp, syncGroups,
- excluded, upgradeSync, installer.appendNoConfirm())
- if errShow != nil {
- return ErrInstallRepoPkgs
- }
- errAur := installer.installAURPackages(ctx, cmdArgs, aurDeps, aurExp,
- aurOrigTargetBases, nameToBaseMap, pkgBuildDirs, true, lastLayer,
- installer.appendNoConfirm())
- return errAur
- }
- func (installer *Installer) installAURPackages(ctx context.Context,
- cmdArgs *parser.Arguments,
- aurDepNames, aurExpNames, aurOrigTargetBases mapset.Set[string],
- nameToBase, pkgBuildDirsByBase map[string]string,
- installIncompatible bool,
- lastLayer bool,
- noConfirm bool,
- ) error {
- all := aurDepNames.Union(aurExpNames).ToSlice()
- if len(all) == 0 {
- return nil
- }
- builtPkgDests := make(map[string]map[string]string)
- deps, exps := make([]string, 0, aurDepNames.Cardinality()), make([]string, 0, aurExpNames.Cardinality())
- pkgArchives := make([]string, 0, len(exps)+len(deps))
- for _, name := range all {
- base := nameToBase[name]
- dir := pkgBuildDirsByBase[base]
- pkgdests, ok := builtPkgDests[base]
- if ok {
- installer.log.Debugln("skipping built pkgbase", base, "package", name)
- } else {
- var errMake error
- installer.log.Debugln("building pkgbase", base, "package", name)
- pkgdests, errMake = installer.buildPkg(ctx, dir, base,
- installIncompatible, cmdArgs.ExistsArg("needed"), aurOrigTargetBases.Contains(base))
- if errMake != nil {
- if !lastLayer {
- return fmt.Errorf("%s - %w", gotext.Get("error making: %s", base), errMake)
- }
- installer.failedAndIgnored[name] = errMake
- installer.log.Errorln(gotext.Get("error making: %s", base), "-", errMake)
- continue
- }
- builtPkgDests[base] = pkgdests
- }
- if len(pkgdests) == 0 {
- installer.log.Warnln(gotext.Get("nothing to install for %s", text.Cyan(base)))
- continue
- }
- newPKGArchives, hasDebug, err := installer.getNewTargets(pkgdests, name)
- if err != nil {
- return err
- }
- pkgArchives = append(pkgArchives, newPKGArchives...)
- if isDep := installer.isDep(cmdArgs, aurExpNames, name); isDep {
- deps = append(deps, name)
- } else {
- exps = append(exps, name)
- }
- if hasDebug {
- deps = append(deps, name+"-debug")
- }
- }
- if err := installPkgArchive(ctx, installer.exeCmd, installer.targetMode,
- installer.vcsStore, cmdArgs, pkgArchives, noConfirm); err != nil {
- return fmt.Errorf("%s - %w", fmt.Sprintf(gotext.Get("error installing:")+" %v", pkgArchives), err)
- }
- if err := setInstallReason(ctx, installer.exeCmd, installer.targetMode, cmdArgs, deps, exps); err != nil {
- return fmt.Errorf("%s - %w", fmt.Sprintf(gotext.Get("error installing:")+" %v", pkgArchives), err)
- }
- return nil
- }
- func (installer *Installer) buildPkg(ctx context.Context,
- dir, base string,
- installIncompatible, needed, isTarget bool,
- ) (map[string]string, error) {
- args := []string{"--nobuild", "-f"}
- if !installer.exeCmd.GetKeepSrc() {
- args = append(args, "-C")
- }
- if installIncompatible {
- args = append(args, "--ignorearch")
- }
- // pkgver bump
- if err := installer.exeCmd.Show(
- installer.exeCmd.BuildMakepkgCmd(ctx, dir, args...)); err != nil {
- return nil, err
- }
- pkgdests, pkgVersion, errList := parsePackageList(ctx, installer.exeCmd, dir)
- if errList != nil {
- return nil, errList
- }
- switch {
- case needed && installer.pkgsAreAlreadyInstalled(pkgdests, pkgVersion) || installer.downloadOnly:
- args = []string{"--nobuild", "--noextract", "--ignorearch"}
- pkgdests = map[string]string{}
- installer.log.Warnln(gotext.Get("%s is up to date -- skipping", text.Cyan(base+"-"+pkgVersion)))
- case installer.skipAlreadyBuiltPkg(isTarget, pkgdests):
- args = []string{"--nobuild", "--noextract", "--ignorearch"}
- installer.log.Warnln(gotext.Get("%s already made -- skipping build", text.Cyan(base+"-"+pkgVersion)))
- default:
- args = []string{"-f", "--noconfirm", "--noextract", "--noprepare", "--holdver"}
- if installIncompatible {
- args = append(args, "--ignorearch")
- }
- }
- if !installer.exeCmd.GetKeepSrc() {
- args = append(args, "-c")
- }
- errMake := installer.exeCmd.Show(
- installer.exeCmd.BuildMakepkgCmd(ctx,
- dir, args...))
- if errMake != nil {
- return nil, errMake
- }
- if installer.downloadOnly {
- return map[string]string{}, nil
- }
- return pkgdests, nil
- }
- func (installer *Installer) pkgsAreAlreadyInstalled(pkgdests map[string]string, pkgVersion string) bool {
- for pkgName := range pkgdests {
- if !installer.dbExecutor.IsCorrectVersionInstalled(pkgName, pkgVersion) {
- return false
- }
- }
- return true
- }
- func pkgsAreBuilt(logger *text.Logger, pkgdests map[string]string) bool {
- for _, pkgdest := range pkgdests {
- if _, err := os.Stat(pkgdest); err != nil {
- logger.Debugln("pkgIsBuilt:", pkgdest, "does not exist")
- return false
- }
- }
- return true
- }
- func (installer *Installer) skipAlreadyBuiltPkg(isTarget bool, pkgdests map[string]string) bool {
- switch installer.rebuildMode {
- case parser.RebuildModeNo:
- return pkgsAreBuilt(installer.log, pkgdests)
- case parser.RebuildModeYes:
- return !isTarget && pkgsAreBuilt(installer.log, pkgdests)
- // case parser.RebuildModeTree: // TODO
- // case parser.RebuildModeAll: // TODO
- default:
- // same as RebuildModeNo
- return pkgsAreBuilt(installer.log, pkgdests)
- }
- }
- func (*Installer) isDep(cmdArgs *parser.Arguments, aurExpNames mapset.Set[string], name string) bool {
- switch {
- case cmdArgs.ExistsArg("asdeps", "asdep"):
- return true
- case cmdArgs.ExistsArg("asexplicit", "asexp"):
- return false
- case aurExpNames.Contains(name):
- return false
- }
- return true
- }
- func (installer *Installer) getNewTargets(pkgdests map[string]string, name string,
- ) (archives []string, good bool, err error) {
- pkgdest, ok := pkgdests[name]
- if !ok {
- return nil, false, &PkgDestNotInListError{name: name}
- }
- pkgArchives := make([]string, 0, 2)
- if _, errStat := os.Stat(pkgdest); os.IsNotExist(errStat) {
- return nil, false, &FindPkgDestError{name: name, pkgDest: pkgdest}
- }
- pkgArchives = append(pkgArchives, pkgdest)
- debugName := name + "-debug"
- pkgdestDebug, ok := pkgdests[debugName]
- if ok {
- if _, errStat := os.Stat(pkgdestDebug); errStat == nil {
- pkgArchives = append(pkgArchives, pkgdestDebug)
- } else {
- ok = false
- }
- }
- return pkgArchives, ok, nil
- }
- func (installer *Installer) installSyncPackages(ctx context.Context, cmdArgs *parser.Arguments,
- syncDeps, // repo targets that are deps
- syncExp mapset.Set[string], // repo targets that are exp
- syncGroups mapset.Set[string], // repo targets that are groups
- excluded []string,
- upgrade bool, // run even without targets
- noConfirm bool,
- ) error {
- repoTargets := syncDeps.Union(syncExp).Union(syncGroups).ToSlice()
- if len(repoTargets) == 0 && !upgrade {
- return nil
- }
- arguments := cmdArgs.Copy()
- arguments.DelArg("asdeps", "asdep")
- arguments.DelArg("asexplicit", "asexp")
- arguments.DelArg("i", "install")
- arguments.Op = "S"
- arguments.ClearTargets()
- arguments.AddTarget(repoTargets...)
- // Don't upgrade all repo packages if only AUR upgrades are specified
- if installer.targetMode == parser.ModeAUR {
- arguments.DelArg("u", "upgrades")
- }
- if len(excluded) > 0 {
- arguments.CreateOrAppendOption("ignore", excluded...)
- }
- errShow := installer.exeCmd.Show(installer.exeCmd.BuildPacmanCmd(ctx,
- arguments, installer.targetMode, noConfirm))
- if errShow != nil {
- return errShow
- }
- if errD := asdeps(ctx, installer.exeCmd, installer.targetMode, cmdArgs, syncDeps.ToSlice()); errD != nil {
- return errD
- }
- if errE := asexp(ctx, installer.exeCmd, installer.targetMode, cmdArgs, syncExp.ToSlice()); errE != nil {
- return errE
- }
- return nil
- }
|