aur_install.go 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  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/exe"
  11. "github.com/Jguer/yay/v11/pkg/settings/parser"
  12. "github.com/Jguer/yay/v11/pkg/text"
  13. "github.com/Jguer/yay/v11/pkg/vcs"
  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. failedAndIgnored map[string]error
  23. exeCmd exe.ICmdBuilder
  24. vcsStore vcs.Store
  25. targetMode parser.TargetMode
  26. downloadOnly bool
  27. }
  28. )
  29. func NewInstaller(dbExecutor db.Executor,
  30. exeCmd exe.ICmdBuilder, vcsStore vcs.Store, targetMode parser.TargetMode,
  31. downloadOnly bool,
  32. ) *Installer {
  33. return &Installer{
  34. dbExecutor: dbExecutor,
  35. postInstallHooks: []PostInstallHookFunc{},
  36. failedAndIgnored: map[string]error{},
  37. exeCmd: exeCmd,
  38. vcsStore: vcsStore,
  39. targetMode: targetMode,
  40. downloadOnly: downloadOnly,
  41. }
  42. }
  43. func (installer *Installer) CompileFailedAndIgnored() error {
  44. if len(installer.failedAndIgnored) == 0 {
  45. return nil
  46. }
  47. return &FailedIgnoredPkgError{
  48. pkgErrors: installer.failedAndIgnored,
  49. }
  50. }
  51. func (installer *Installer) AddPostInstallHook(hook PostInstallHookFunc) {
  52. if hook == nil {
  53. return
  54. }
  55. installer.postInstallHooks = append(installer.postInstallHooks, hook)
  56. }
  57. func (installer *Installer) RunPostInstallHooks(ctx context.Context) error {
  58. var errMulti multierror.MultiError
  59. for _, hook := range installer.postInstallHooks {
  60. if err := hook(ctx); err != nil {
  61. errMulti.Add(err)
  62. }
  63. }
  64. return errMulti.Return()
  65. }
  66. func (installer *Installer) Install(ctx context.Context,
  67. cmdArgs *parser.Arguments,
  68. targets []map[string]*dep.InstallInfo,
  69. pkgBuildDirs map[string]string,
  70. ) error {
  71. // Reorganize targets into layers of dependencies
  72. for i := len(targets) - 1; i >= 0; i-- {
  73. err := installer.handleLayer(ctx, cmdArgs, targets[i], pkgBuildDirs, i == 0)
  74. if err != nil {
  75. // rollback
  76. return err
  77. }
  78. }
  79. return nil
  80. }
  81. func (installer *Installer) handleLayer(ctx context.Context,
  82. cmdArgs *parser.Arguments,
  83. layer map[string]*dep.InstallInfo,
  84. pkgBuildDirs map[string]string,
  85. lastLayer bool,
  86. ) error {
  87. // Install layer
  88. nameToBaseMap := make(map[string]string, 0)
  89. syncDeps, syncExp := mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
  90. aurDeps, aurExp := mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
  91. for name, info := range layer {
  92. switch info.Source {
  93. case dep.AUR, dep.SrcInfo:
  94. nameToBaseMap[name] = *info.AURBase
  95. switch info.Reason {
  96. case dep.Explicit:
  97. if cmdArgs.ExistsArg("asdeps", "asdep") {
  98. aurDeps.Add(name)
  99. } else {
  100. aurExp.Add(name)
  101. }
  102. case dep.Dep, dep.MakeDep, dep.CheckDep:
  103. aurDeps.Add(name)
  104. }
  105. case dep.Sync:
  106. compositePkgName := fmt.Sprintf("%s/%s", *info.SyncDBName, name)
  107. switch info.Reason {
  108. case dep.Explicit:
  109. if cmdArgs.ExistsArg("asdeps", "asdep") {
  110. syncDeps.Add(compositePkgName)
  111. } else {
  112. syncExp.Add(compositePkgName)
  113. }
  114. case dep.Dep, dep.MakeDep, dep.CheckDep:
  115. syncDeps.Add(compositePkgName)
  116. }
  117. }
  118. }
  119. text.Debugln("syncDeps", syncDeps, "SyncExp", syncExp, "aurDeps", aurDeps, "aurExp", aurExp)
  120. errShow := installer.installSyncPackages(ctx, cmdArgs, syncDeps, syncExp)
  121. if errShow != nil {
  122. return ErrInstallRepoPkgs
  123. }
  124. errAur := installer.installAURPackages(ctx, cmdArgs, aurDeps, aurExp,
  125. nameToBaseMap, pkgBuildDirs, true, lastLayer)
  126. return errAur
  127. }
  128. func (installer *Installer) installAURPackages(ctx context.Context,
  129. cmdArgs *parser.Arguments,
  130. aurDepNames, aurExpNames mapset.Set[string],
  131. nameToBase, pkgBuildDirsByBase map[string]string,
  132. installIncompatible bool,
  133. lastLayer bool,
  134. ) error {
  135. all := aurDepNames.Union(aurExpNames).ToSlice()
  136. if len(all) == 0 {
  137. return nil
  138. }
  139. deps, exps := make([]string, 0, aurDepNames.Cardinality()), make([]string, 0, aurExpNames.Cardinality())
  140. pkgArchives := make([]string, 0, len(exps)+len(deps))
  141. for _, name := range all {
  142. base := nameToBase[name]
  143. dir := pkgBuildDirsByBase[base]
  144. pkgdests, errMake := installer.buildPkg(ctx, dir, base, installIncompatible, cmdArgs.ExistsArg("needed"))
  145. if errMake != nil {
  146. if !lastLayer {
  147. return fmt.Errorf("%s - %w", gotext.Get("error making: %s", base), errMake)
  148. }
  149. installer.failedAndIgnored[name] = errMake
  150. text.Errorln(gotext.Get("error making: %s", base), "-", errMake)
  151. continue
  152. }
  153. if len(pkgdests) == 0 {
  154. text.Warnln(gotext.Get("nothing to install for %s", text.Cyan(base)))
  155. continue
  156. }
  157. newPKGArchives, hasDebug, err := installer.getNewTargets(pkgdests, name)
  158. if err != nil {
  159. return err
  160. }
  161. pkgArchives = append(pkgArchives, newPKGArchives...)
  162. if isDep := installer.isDep(cmdArgs, aurExpNames, name); isDep {
  163. deps = append(deps, name)
  164. } else {
  165. exps = append(exps, name)
  166. }
  167. if hasDebug {
  168. deps = append(deps, name+"-debug")
  169. }
  170. }
  171. if err := installPkgArchive(ctx, installer.exeCmd, installer.targetMode, installer.vcsStore, cmdArgs, pkgArchives); err != nil {
  172. return fmt.Errorf("%s - %w", fmt.Sprintf(gotext.Get("error installing:")+" %v", pkgArchives), err)
  173. }
  174. if err := setInstallReason(ctx, installer.exeCmd, installer.targetMode, cmdArgs, deps, exps); err != nil {
  175. return fmt.Errorf("%s - %w", fmt.Sprintf(gotext.Get("error installing:")+" %v", pkgArchives), err)
  176. }
  177. return nil
  178. }
  179. func (installer *Installer) buildPkg(ctx context.Context,
  180. dir, base string,
  181. installIncompatible, needed bool,
  182. ) (map[string]string, error) {
  183. args := []string{"--nobuild", "-fC"}
  184. if installIncompatible {
  185. args = append(args, "--ignorearch")
  186. }
  187. // pkgver bump
  188. if err := installer.exeCmd.Show(
  189. installer.exeCmd.BuildMakepkgCmd(ctx, dir, args...)); err != nil {
  190. return nil, err
  191. }
  192. pkgdests, pkgVersion, errList := parsePackageList(ctx, installer.exeCmd, dir)
  193. if errList != nil {
  194. return nil, errList
  195. }
  196. switch {
  197. case needed && installer.pkgsAreAlreadyInstalled(pkgdests, pkgVersion) || installer.downloadOnly:
  198. args = []string{"-c", "--nobuild", "--noextract", "--ignorearch"}
  199. pkgdests = map[string]string{}
  200. text.Warnln(gotext.Get("%s is up to date -- skipping", text.Cyan(base+"-"+pkgVersion)))
  201. case pkgsAreBuilt(pkgdests):
  202. args = []string{"-c", "--nobuild", "--noextract", "--ignorearch"}
  203. text.Warnln(gotext.Get("%s already made -- skipping build", text.Cyan(base+"-"+pkgVersion)))
  204. default:
  205. args = []string{"-cf", "--noconfirm", "--noextract", "--noprepare", "--holdver"}
  206. if installIncompatible {
  207. args = append(args, "--ignorearch")
  208. }
  209. }
  210. errMake := installer.exeCmd.Show(
  211. installer.exeCmd.BuildMakepkgCmd(ctx,
  212. dir, args...))
  213. if errMake != nil {
  214. return nil, errMake
  215. }
  216. if installer.downloadOnly {
  217. return map[string]string{}, nil
  218. }
  219. return pkgdests, nil
  220. }
  221. func (installer *Installer) pkgsAreAlreadyInstalled(pkgdests map[string]string, pkgVersion string) bool {
  222. for pkgName := range pkgdests {
  223. if !installer.dbExecutor.IsCorrectVersionInstalled(pkgName, pkgVersion) {
  224. return false
  225. }
  226. }
  227. return true
  228. }
  229. func pkgsAreBuilt(pkgdests map[string]string) bool {
  230. for _, pkgdest := range pkgdests {
  231. if _, err := os.Stat(pkgdest); err != nil {
  232. text.Debugln("pkgIsBuilt:", pkgdest, "does not exist")
  233. return false
  234. }
  235. }
  236. return true
  237. }
  238. func (*Installer) isDep(cmdArgs *parser.Arguments, aurExpNames mapset.Set[string], name string) bool {
  239. switch {
  240. case cmdArgs.ExistsArg("asdeps", "asdep"):
  241. return true
  242. case cmdArgs.ExistsArg("asexplicit", "asexp"):
  243. return false
  244. case aurExpNames.Contains(name):
  245. return false
  246. }
  247. return true
  248. }
  249. func (installer *Installer) getNewTargets(pkgdests map[string]string, name string,
  250. ) (archives []string, good bool, err error) {
  251. pkgdest, ok := pkgdests[name]
  252. if !ok {
  253. return nil, false, &PkgDestNotInListError{name: name}
  254. }
  255. pkgArchives := make([]string, 0, 2)
  256. if _, errStat := os.Stat(pkgdest); os.IsNotExist(errStat) {
  257. return nil, false, &FindPkgDestError{name: name, pkgDest: pkgdest}
  258. }
  259. pkgArchives = append(pkgArchives, pkgdest)
  260. debugName := pkgdest + "-debug"
  261. pkgdestDebug, ok := pkgdests[debugName]
  262. if ok {
  263. if _, errStat := os.Stat(pkgdestDebug); errStat == nil {
  264. pkgArchives = append(pkgArchives, debugName)
  265. }
  266. }
  267. return pkgArchives, ok, nil
  268. }
  269. func (installer *Installer) installSyncPackages(ctx context.Context, cmdArgs *parser.Arguments,
  270. syncDeps, // repo targets that are deps
  271. syncExp mapset.Set[string], // repo targets that are exp
  272. ) error {
  273. repoTargets := syncDeps.Union(syncExp).ToSlice()
  274. if len(repoTargets) == 0 {
  275. return nil
  276. }
  277. arguments := cmdArgs.Copy()
  278. arguments.DelArg("asdeps", "asdep")
  279. arguments.DelArg("asexplicit", "asexp")
  280. arguments.DelArg("i", "install")
  281. arguments.DelArg("u", "upgrade")
  282. arguments.Op = "S"
  283. arguments.ClearTargets()
  284. arguments.AddTarget(repoTargets...)
  285. errShow := installer.exeCmd.Show(installer.exeCmd.BuildPacmanCmd(ctx,
  286. arguments, installer.targetMode, settings.NoConfirm))
  287. if errD := asdeps(ctx, installer.exeCmd, installer.targetMode, cmdArgs, syncDeps.ToSlice()); errD != nil {
  288. return errD
  289. }
  290. if errE := asexp(ctx, installer.exeCmd, installer.targetMode, cmdArgs, syncExp.ToSlice()); errE != nil {
  291. return errE
  292. }
  293. return errShow
  294. }