aur_install.go 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. package main
  2. import (
  3. "context"
  4. "fmt"
  5. "os"
  6. "github.com/Jguer/yay/v11/pkg/db"
  7. "github.com/Jguer/yay/v11/pkg/dep"
  8. "github.com/Jguer/yay/v11/pkg/multierror"
  9. "github.com/Jguer/yay/v11/pkg/settings"
  10. "github.com/Jguer/yay/v11/pkg/settings/parser"
  11. "github.com/Jguer/yay/v11/pkg/text"
  12. mapset "github.com/deckarep/golang-set/v2"
  13. "github.com/leonelquinteros/gotext"
  14. )
  15. type (
  16. PostInstallHookFunc func(ctx context.Context) error
  17. Installer struct {
  18. dbExecutor db.Executor
  19. postInstallHooks []PostInstallHookFunc
  20. }
  21. )
  22. func (installer *Installer) AddPostInstallHook(hook PostInstallHookFunc) {
  23. if hook == nil {
  24. return
  25. }
  26. installer.postInstallHooks = append(installer.postInstallHooks, hook)
  27. }
  28. func (installer *Installer) RunPostInstallHooks(ctx context.Context) error {
  29. var errMulti multierror.MultiError
  30. for _, hook := range installer.postInstallHooks {
  31. if err := hook(ctx); err != nil {
  32. errMulti.Add(err)
  33. }
  34. }
  35. return errMulti.Return()
  36. }
  37. func (installer *Installer) Install(ctx context.Context,
  38. cmdArgs *parser.Arguments,
  39. targets []map[string]*dep.InstallInfo,
  40. pkgBuildDirs map[string]string,
  41. ) error {
  42. // Reorganize targets into layers of dependencies
  43. for i := len(targets) - 1; i >= 0; i-- {
  44. err := installer.handleLayer(ctx, cmdArgs, targets[i], pkgBuildDirs)
  45. if err != nil {
  46. // rollback
  47. return err
  48. }
  49. }
  50. return nil
  51. }
  52. func (installer *Installer) handleLayer(ctx context.Context,
  53. cmdArgs *parser.Arguments, layer map[string]*dep.InstallInfo, pkgBuildDirs map[string]string,
  54. ) error {
  55. // Install layer
  56. nameToBaseMap := make(map[string]string, 0)
  57. syncDeps, syncExp := mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
  58. aurDeps, aurExp := mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
  59. for name, info := range layer {
  60. switch info.Source {
  61. case dep.AUR, dep.SrcInfo:
  62. nameToBaseMap[name] = *info.AURBase
  63. switch info.Reason {
  64. case dep.Explicit:
  65. if cmdArgs.ExistsArg("asdeps", "asdep") {
  66. aurDeps.Add(name)
  67. } else {
  68. aurExp.Add(name)
  69. }
  70. case dep.Dep, dep.MakeDep, dep.CheckDep:
  71. aurDeps.Add(name)
  72. }
  73. case dep.Sync:
  74. compositePkgName := fmt.Sprintf("%s/%s", *info.SyncDBName, name)
  75. switch info.Reason {
  76. case dep.Explicit:
  77. if cmdArgs.ExistsArg("asdeps", "asdep") {
  78. syncDeps.Add(compositePkgName)
  79. } else {
  80. syncExp.Add(compositePkgName)
  81. }
  82. case dep.Dep, dep.MakeDep, dep.CheckDep:
  83. syncDeps.Add(compositePkgName)
  84. }
  85. }
  86. }
  87. text.Debugln("syncDeps", syncDeps, "SyncExp", syncExp, "aurDeps", aurDeps, "aurExp", aurExp)
  88. errShow := installer.installSyncPackages(ctx, cmdArgs, syncDeps, syncExp)
  89. if errShow != nil {
  90. return ErrInstallRepoPkgs
  91. }
  92. errAur := installer.installAURPackages(ctx, cmdArgs, aurDeps, aurExp, nameToBaseMap, pkgBuildDirs, false)
  93. return errAur
  94. }
  95. func (installer *Installer) installAURPackages(ctx context.Context,
  96. cmdArgs *parser.Arguments,
  97. aurDepNames, aurExpNames mapset.Set[string],
  98. nameToBase, pkgBuildDirsByBase map[string]string,
  99. installIncompatible bool,
  100. ) error {
  101. deps, exps := make([]string, 0, aurDepNames.Cardinality()), make([]string, 0, aurExpNames.Cardinality())
  102. pkgArchives := make([]string, 0, len(exps)+len(deps))
  103. for _, name := range aurDepNames.Union(aurExpNames).ToSlice() {
  104. base := nameToBase[name]
  105. dir := pkgBuildDirsByBase[base]
  106. args := []string{"--nobuild", "-fC"}
  107. if installIncompatible {
  108. args = append(args, "--ignorearch")
  109. }
  110. // pkgver bump
  111. if err := config.Runtime.CmdBuilder.Show(
  112. config.Runtime.CmdBuilder.BuildMakepkgCmd(ctx, dir, args...)); err != nil {
  113. return fmt.Errorf("%s - %w", gotext.Get("error making: %s", base), err)
  114. }
  115. pkgdests, _, errList := parsePackageList(ctx, dir)
  116. if errList != nil {
  117. return errList
  118. }
  119. args = []string{"-cf", "--noconfirm", "--noextract", "--noprepare", "--holdver"}
  120. if installIncompatible {
  121. args = append(args, "--ignorearch")
  122. }
  123. if errMake := config.Runtime.CmdBuilder.Show(
  124. config.Runtime.CmdBuilder.BuildMakepkgCmd(ctx,
  125. dir, args...)); errMake != nil {
  126. return fmt.Errorf("%s - %w", gotext.Get("error making: %s", base), errMake)
  127. }
  128. newPKGArchives, hasDebug, err := installer.getNewTargets(pkgdests, name)
  129. if err != nil {
  130. return err
  131. }
  132. pkgArchives = append(pkgArchives, newPKGArchives...)
  133. if isDep := installer.isDep(cmdArgs, aurExpNames, name); isDep {
  134. deps = append(deps, name)
  135. } else {
  136. exps = append(exps, name)
  137. }
  138. if hasDebug {
  139. deps = append(deps, name+"-debug")
  140. }
  141. }
  142. if err := installPkgArchive(ctx, cmdArgs, pkgArchives); err != nil {
  143. return fmt.Errorf("%s - %w", fmt.Sprintf(gotext.Get("error installing:")+" %v", pkgArchives), err)
  144. }
  145. if err := setInstallReason(ctx, cmdArgs, deps, exps); err != nil {
  146. return fmt.Errorf("%s - %w", fmt.Sprintf(gotext.Get("error installing:")+" %v", pkgArchives), err)
  147. }
  148. return nil
  149. }
  150. func (*Installer) isDep(cmdArgs *parser.Arguments, aurExpNames mapset.Set[string], name string) bool {
  151. switch {
  152. case cmdArgs.ExistsArg("asdeps", "asdep"):
  153. return true
  154. case cmdArgs.ExistsArg("asexplicit", "asexp"):
  155. return false
  156. case aurExpNames.Contains(name):
  157. return false
  158. }
  159. return true
  160. }
  161. func (installer *Installer) getNewTargets(pkgdests map[string]string, name string,
  162. ) (archives []string, good bool, err error) {
  163. pkgdest, ok := pkgdests[name]
  164. if !ok {
  165. return nil, false, &PkgDestNotInListError{name: name}
  166. }
  167. pkgArchives := make([]string, 0, 2)
  168. if _, errStat := os.Stat(pkgdest); os.IsNotExist(errStat) {
  169. return nil, false, &FindPkgDestError{name: name, pkgDest: pkgdest}
  170. }
  171. pkgArchives = append(pkgArchives, pkgdest)
  172. debugName := pkgdest + "-debug"
  173. pkgdestDebug, ok := pkgdests[debugName]
  174. if ok {
  175. if _, errStat := os.Stat(pkgdestDebug); errStat == nil {
  176. pkgArchives = append(pkgArchives, debugName)
  177. }
  178. }
  179. return pkgArchives, ok, nil
  180. }
  181. func (*Installer) installSyncPackages(ctx context.Context, cmdArgs *parser.Arguments,
  182. syncDeps, // repo targets that are deps
  183. syncExp mapset.Set[string], // repo targets that are exp
  184. ) error {
  185. repoTargets := syncDeps.Union(syncExp).ToSlice()
  186. if len(repoTargets) == 0 {
  187. return nil
  188. }
  189. arguments := cmdArgs.Copy()
  190. arguments.DelArg("asdeps", "asdep")
  191. arguments.DelArg("asexplicit", "asexp")
  192. arguments.DelArg("i", "install")
  193. arguments.DelArg("u", "upgrade")
  194. arguments.Op = "S"
  195. arguments.ClearTargets()
  196. arguments.AddTarget(repoTargets...)
  197. errShow := config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  198. arguments, config.Runtime.Mode, settings.NoConfirm))
  199. if errD := asdeps(ctx, cmdArgs, syncDeps.ToSlice()); errD != nil {
  200. return errD
  201. }
  202. if errE := asexp(ctx, cmdArgs, syncExp.ToSlice()); errE != nil {
  203. return errE
  204. }
  205. return errShow
  206. }