sync.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. package main
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "os"
  7. "strings"
  8. "github.com/Jguer/yay/v11/pkg/completion"
  9. "github.com/Jguer/yay/v11/pkg/db"
  10. "github.com/Jguer/yay/v11/pkg/dep"
  11. "github.com/Jguer/yay/v11/pkg/multierror"
  12. "github.com/Jguer/yay/v11/pkg/settings"
  13. "github.com/Jguer/yay/v11/pkg/settings/parser"
  14. "github.com/Jguer/yay/v11/pkg/srcinfo"
  15. "github.com/Jguer/yay/v11/pkg/text"
  16. "github.com/Jguer/yay/v11/pkg/upgrade"
  17. "github.com/leonelquinteros/gotext"
  18. )
  19. func syncInstall(ctx context.Context,
  20. config *settings.Configuration,
  21. cmdArgs *parser.Arguments,
  22. dbExecutor db.Executor,
  23. ) error {
  24. aurCache := config.Runtime.AURCache
  25. refreshArg := cmdArgs.ExistsArg("y", "refresh")
  26. noDeps := cmdArgs.ExistsArg("d", "nodeps")
  27. noCheck := strings.Contains(config.MFlags, "--nocheck")
  28. if noDeps {
  29. config.Runtime.CmdBuilder.AddMakepkgFlag("-d")
  30. }
  31. if refreshArg && config.Runtime.Mode.AtLeastRepo() {
  32. if errR := earlyRefresh(ctx, cmdArgs); errR != nil {
  33. return fmt.Errorf("%s - %w", gotext.Get("error refreshing databases"), errR)
  34. }
  35. // we may have done -Sy, our handle now has an old
  36. // database.
  37. if errRefresh := dbExecutor.RefreshHandle(); errRefresh != nil {
  38. return errRefresh
  39. }
  40. }
  41. grapher := dep.NewGrapher(dbExecutor, aurCache, false, settings.NoConfirm, os.Stdout, noDeps, noCheck)
  42. graph, err := grapher.GraphFromTargets(ctx, nil, cmdArgs.Targets)
  43. if err != nil {
  44. return err
  45. }
  46. if cmdArgs.ExistsArg("u", "sysupgrade") {
  47. var errSysUp error
  48. upService := upgrade.NewUpgradeService(
  49. grapher, aurCache, config.Runtime.AURClient,
  50. dbExecutor, config.Runtime.VCSStore, config.Runtime, config, settings.NoConfirm, config.Runtime.Logger.Child("upgrade"))
  51. graph, errSysUp = upService.GraphUpgrades(ctx, graph, cmdArgs.ExistsDouble("u", "sysupgrade"))
  52. if errSysUp != nil {
  53. return errSysUp
  54. }
  55. }
  56. opService := NewOperationService(ctx, config, dbExecutor)
  57. multiErr := &multierror.MultiError{}
  58. targets := graph.TopoSortedLayerMap(func(s string, ii *dep.InstallInfo) error {
  59. if ii.Source == dep.Missing {
  60. multiErr.Add(errors.New(gotext.Get("could not find %s%s", s, ii.Version)))
  61. }
  62. return nil
  63. })
  64. if err := multiErr.Return(); err != nil {
  65. return err
  66. }
  67. return opService.Run(ctx, cmdArgs, targets)
  68. }
  69. type OperationService struct {
  70. ctx context.Context
  71. cfg *settings.Configuration
  72. dbExecutor db.Executor
  73. }
  74. func NewOperationService(ctx context.Context, cfg *settings.Configuration, dbExecutor db.Executor) *OperationService {
  75. return &OperationService{
  76. ctx: ctx,
  77. cfg: cfg,
  78. dbExecutor: dbExecutor,
  79. }
  80. }
  81. func (o *OperationService) Run(ctx context.Context,
  82. cmdArgs *parser.Arguments,
  83. targets []map[string]*dep.InstallInfo,
  84. ) error {
  85. if len(targets) == 0 {
  86. fmt.Fprintln(os.Stdout, "", gotext.Get("there is nothing to do"))
  87. return nil
  88. }
  89. preparer := NewPreparer(o.dbExecutor, o.cfg.Runtime.CmdBuilder, o.cfg)
  90. installer := NewInstaller(o.dbExecutor, o.cfg.Runtime.CmdBuilder, o.cfg.Runtime.VCSStore, o.cfg.Runtime.Mode)
  91. pkgBuildDirs, errInstall := preparer.Run(ctx, os.Stdout, targets)
  92. if errInstall != nil {
  93. return errInstall
  94. }
  95. cleanFunc := preparer.ShouldCleanMakeDeps()
  96. if cleanFunc != nil {
  97. installer.AddPostInstallHook(cleanFunc)
  98. }
  99. if cleanAURDirsFunc := preparer.ShouldCleanAURDirs(pkgBuildDirs); cleanAURDirsFunc != nil {
  100. installer.AddPostInstallHook(cleanAURDirsFunc)
  101. }
  102. go func() {
  103. errComp := completion.Update(ctx, o.cfg.Runtime.HTTPClient, o.dbExecutor,
  104. o.cfg.AURURL, o.cfg.Runtime.CompletionPath, o.cfg.CompletionInterval, false)
  105. if errComp != nil {
  106. text.Warnln(errComp)
  107. }
  108. }()
  109. srcInfo, errInstall := srcinfo.NewService(o.dbExecutor, o.cfg, o.cfg.Runtime.CmdBuilder, o.cfg.Runtime.VCSStore, pkgBuildDirs)
  110. if errInstall != nil {
  111. return errInstall
  112. }
  113. incompatible, errInstall := srcInfo.IncompatiblePkgs(ctx)
  114. if errInstall != nil {
  115. return errInstall
  116. }
  117. if errIncompatible := confirmIncompatible(incompatible); errIncompatible != nil {
  118. return errIncompatible
  119. }
  120. if errPGP := srcInfo.CheckPGPKeys(ctx); errPGP != nil {
  121. return errPGP
  122. }
  123. if errInstall := installer.Install(ctx, cmdArgs, targets, pkgBuildDirs); errInstall != nil {
  124. return errInstall
  125. }
  126. var multiErr multierror.MultiError
  127. if err := installer.CompileFailedAndIgnored(); err != nil {
  128. multiErr.Add(err)
  129. }
  130. if err := srcInfo.UpdateVCSStore(ctx, targets, installer.failedAndIgnored); err != nil {
  131. text.Warnln(err)
  132. }
  133. if err := installer.RunPostInstallHooks(ctx); err != nil {
  134. multiErr.Add(err)
  135. }
  136. return multiErr.Return()
  137. }
  138. func confirmIncompatible(incompatible []string) error {
  139. if len(incompatible) > 0 {
  140. text.Warnln(gotext.Get("The following packages are not compatible with your architecture:"))
  141. for _, pkg := range incompatible {
  142. fmt.Print(" " + text.Cyan(pkg))
  143. }
  144. fmt.Println()
  145. if !text.ContinueTask(os.Stdin, gotext.Get("Try to build them anyway?"), true, settings.NoConfirm) {
  146. return &settings.ErrUserAbort{}
  147. }
  148. }
  149. return nil
  150. }