sync.go 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. package sync
  2. import (
  3. "context"
  4. "github.com/Jguer/yay/v12/pkg/completion"
  5. "github.com/Jguer/yay/v12/pkg/db"
  6. "github.com/Jguer/yay/v12/pkg/dep"
  7. "github.com/Jguer/yay/v12/pkg/multierror"
  8. "github.com/Jguer/yay/v12/pkg/runtime"
  9. "github.com/Jguer/yay/v12/pkg/settings"
  10. "github.com/Jguer/yay/v12/pkg/settings/parser"
  11. "github.com/Jguer/yay/v12/pkg/sync/build"
  12. "github.com/Jguer/yay/v12/pkg/sync/srcinfo"
  13. "github.com/Jguer/yay/v12/pkg/sync/workdir"
  14. "github.com/Jguer/yay/v12/pkg/text"
  15. "github.com/leonelquinteros/gotext"
  16. )
  17. type OperationService struct {
  18. ctx context.Context
  19. cfg *settings.Configuration
  20. dbExecutor db.Executor
  21. logger *text.Logger
  22. }
  23. func NewOperationService(ctx context.Context,
  24. dbExecutor db.Executor,
  25. run *runtime.Runtime,
  26. ) *OperationService {
  27. return &OperationService{
  28. ctx: ctx,
  29. cfg: run.Cfg,
  30. dbExecutor: dbExecutor,
  31. logger: run.Logger.Child("operation"),
  32. }
  33. }
  34. func (o *OperationService) Run(ctx context.Context, run *runtime.Runtime,
  35. cmdArgs *parser.Arguments,
  36. targets []map[string]*dep.InstallInfo, excluded []string,
  37. ) error {
  38. if len(targets) == 0 {
  39. o.logger.Println("", gotext.Get("there is nothing to do"))
  40. return nil
  41. }
  42. preparer := workdir.NewPreparer(o.dbExecutor, run.CmdBuilder, o.cfg, o.logger.Child("workdir"))
  43. installer := build.NewInstaller(o.dbExecutor, run.CmdBuilder,
  44. run.VCSStore, o.cfg.Mode, o.cfg.ReBuild,
  45. cmdArgs.ExistsArg("w", "downloadonly"), run.Logger.Child("installer"))
  46. pkgBuildDirs, errInstall := preparer.Run(ctx, run, targets)
  47. if errInstall != nil {
  48. return errInstall
  49. }
  50. if cleanFunc := preparer.ShouldCleanMakeDeps(run, cmdArgs); cleanFunc != nil {
  51. installer.AddPostInstallHook(cleanFunc)
  52. }
  53. if cleanAURDirsFunc := preparer.ShouldCleanAURDirs(run, pkgBuildDirs); cleanAURDirsFunc != nil {
  54. installer.AddPostInstallHook(cleanAURDirsFunc)
  55. }
  56. go func() {
  57. errComp := completion.Update(ctx, run.HTTPClient, o.dbExecutor,
  58. o.cfg.AURURL, o.cfg.CompletionPath, o.cfg.CompletionInterval, false)
  59. if errComp != nil {
  60. o.logger.Warnln(errComp)
  61. }
  62. }()
  63. srcInfo, errInstall := srcinfo.NewService(o.dbExecutor, o.cfg,
  64. o.logger.Child("srcinfo"), run.CmdBuilder, run.VCSStore, pkgBuildDirs)
  65. if errInstall != nil {
  66. return errInstall
  67. }
  68. incompatible, errInstall := srcInfo.IncompatiblePkgs(ctx)
  69. if errInstall != nil {
  70. return errInstall
  71. }
  72. if errIncompatible := confirmIncompatible(o.logger, incompatible); errIncompatible != nil {
  73. return errIncompatible
  74. }
  75. if errPGP := srcInfo.CheckPGPKeys(ctx); errPGP != nil {
  76. return errPGP
  77. }
  78. if errInstall := installer.Install(ctx, cmdArgs, targets, pkgBuildDirs,
  79. excluded, o.manualConfirmRequired(cmdArgs)); errInstall != nil {
  80. return errInstall
  81. }
  82. var multiErr multierror.MultiError
  83. failedAndIgnored, err := installer.CompileFailedAndIgnored()
  84. if err != nil {
  85. multiErr.Add(err)
  86. }
  87. if !cmdArgs.ExistsArg("w", "downloadonly") {
  88. if err := srcInfo.UpdateVCSStore(ctx, targets, failedAndIgnored); err != nil {
  89. o.logger.Warnln(err)
  90. }
  91. }
  92. if err := installer.RunPostInstallHooks(ctx); err != nil {
  93. multiErr.Add(err)
  94. }
  95. return multiErr.Return()
  96. }
  97. func (o *OperationService) manualConfirmRequired(cmdArgs *parser.Arguments) bool {
  98. return (!cmdArgs.ExistsArg("u", "sysupgrade") && cmdArgs.Op != "Y") || o.cfg.DoubleConfirm
  99. }
  100. func confirmIncompatible(logger *text.Logger, incompatible []string) error {
  101. if len(incompatible) > 0 {
  102. logger.Warnln(gotext.Get("The following packages are not compatible with your architecture:"))
  103. for _, pkg := range incompatible {
  104. logger.Print(" " + text.Cyan(pkg))
  105. }
  106. logger.Println()
  107. if !logger.ContinueTask(gotext.Get("Try to build them anyway?"), true, settings.NoConfirm) {
  108. return &settings.ErrUserAbort{}
  109. }
  110. }
  111. return nil
  112. }