sync.go 5.5 KB

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