aur_install.go 7.8 KB

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