local_install.go 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. // Experimental code for install local with dependency refactoring
  2. // Not at feature parity with install.go
  3. package main
  4. import (
  5. "context"
  6. "fmt"
  7. "os"
  8. "path/filepath"
  9. "github.com/Jguer/yay/v11/pkg/db"
  10. "github.com/Jguer/yay/v11/pkg/dep"
  11. "github.com/Jguer/yay/v11/pkg/download"
  12. "github.com/Jguer/yay/v11/pkg/metadata"
  13. "github.com/Jguer/yay/v11/pkg/settings"
  14. "github.com/Jguer/yay/v11/pkg/settings/exe"
  15. "github.com/Jguer/yay/v11/pkg/settings/parser"
  16. "github.com/Jguer/yay/v11/pkg/text"
  17. "github.com/leonelquinteros/gotext"
  18. "github.com/pkg/errors"
  19. gosrc "github.com/Morganamilo/go-srcinfo"
  20. mapset "github.com/deckarep/golang-set/v2"
  21. )
  22. var ErrInstallRepoPkgs = errors.New(gotext.Get("error installing repo packages"))
  23. func installLocalPKGBUILD(
  24. ctx context.Context,
  25. cmdArgs *parser.Arguments,
  26. dbExecutor db.Executor,
  27. ) error {
  28. aurCache, err := metadata.NewAURCache(filepath.Join(config.BuildDir, "aur.json"))
  29. if err != nil {
  30. return errors.Wrap(err, gotext.Get("failed to retrieve aur Cache"))
  31. }
  32. wd, err := os.Getwd()
  33. if err != nil {
  34. return errors.Wrap(err, gotext.Get("failed to retrieve working directory"))
  35. }
  36. if len(cmdArgs.Targets) > 1 {
  37. return errors.New(gotext.Get("only one target is allowed"))
  38. }
  39. if len(cmdArgs.Targets) == 1 {
  40. wd = cmdArgs.Targets[0]
  41. }
  42. pkgbuild, err := gosrc.ParseFile(filepath.Join(wd, ".SRCINFO"))
  43. if err != nil {
  44. return errors.Wrap(err, gotext.Get("failed to parse .SRCINFO"))
  45. }
  46. grapher := dep.NewGrapher(dbExecutor, aurCache, false, settings.NoConfirm, os.Stdout)
  47. graph, err := grapher.GraphFromSrcInfo(wd, pkgbuild)
  48. if err != nil {
  49. return err
  50. }
  51. topoSorted := graph.TopoSortedLayerMap()
  52. fmt.Println(topoSorted, len(topoSorted))
  53. preparer := &Preparer{dbExecutor: dbExecutor, cmdBuilder: config.Runtime.CmdBuilder}
  54. installer := &Installer{dbExecutor: dbExecutor}
  55. pkgBuildDirs, err := preparer.PrepareWorkspace(ctx, topoSorted)
  56. if err != nil {
  57. return err
  58. }
  59. return installer.Install(ctx, cmdArgs, topoSorted, pkgBuildDirs)
  60. }
  61. type Preparer struct {
  62. dbExecutor db.Executor
  63. cmdBuilder exe.ICmdBuilder
  64. aurBases []string
  65. pkgBuildDirs []string
  66. }
  67. func (preper *Preparer) PrepareWorkspace(ctx context.Context, targets []map[string]*dep.InstallInfo,
  68. ) (map[string]string, error) {
  69. pkgBuildDirs := make(map[string]string, 0)
  70. for _, layer := range targets {
  71. for pkgBase, info := range layer {
  72. if info.Source == dep.AUR {
  73. preper.aurBases = append(preper.aurBases, pkgBase)
  74. pkgBuildDirs[pkgBase] = filepath.Join(config.BuildDir, pkgBase)
  75. } else if info.Source == dep.SrcInfo {
  76. pkgBuildDirs[pkgBase] = *info.SrcinfoPath
  77. }
  78. }
  79. }
  80. if _, errA := download.AURPKGBUILDRepos(ctx,
  81. preper.cmdBuilder, preper.aurBases, config.AURURL, config.BuildDir, false); errA != nil {
  82. return nil, errA
  83. }
  84. if errP := downloadPKGBUILDSourceFanout(ctx, config.Runtime.CmdBuilder,
  85. preper.pkgBuildDirs, false, config.MaxConcurrentDownloads); errP != nil {
  86. text.Errorln(errP)
  87. }
  88. return pkgBuildDirs, nil
  89. }
  90. type Installer struct {
  91. dbExecutor db.Executor
  92. }
  93. func (installer *Installer) Install(ctx context.Context,
  94. cmdArgs *parser.Arguments,
  95. targets []map[string]*dep.InstallInfo,
  96. pkgBuildDirs map[string]string,
  97. ) error {
  98. // Reorganize targets into layers of dependencies
  99. for i := len(targets) - 1; i >= 0; i-- {
  100. err := installer.handleLayer(ctx, cmdArgs, targets[i], pkgBuildDirs)
  101. if err != nil {
  102. // rollback
  103. return err
  104. }
  105. }
  106. return nil
  107. }
  108. func (installer *Installer) handleLayer(ctx context.Context,
  109. cmdArgs *parser.Arguments, layer map[string]*dep.InstallInfo, pkgBuildDirs map[string]string,
  110. ) error {
  111. // Install layer
  112. nameToBaseMap := make(map[string]string, 0)
  113. syncDeps, syncExp := mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
  114. aurDeps, aurExp := mapset.NewThreadUnsafeSet[string](), mapset.NewThreadUnsafeSet[string]()
  115. for name, info := range layer {
  116. switch info.Source {
  117. case dep.SrcInfo:
  118. fallthrough
  119. case dep.AUR:
  120. nameToBaseMap[name] = *info.AURBase
  121. switch info.Reason {
  122. case dep.Explicit:
  123. if cmdArgs.ExistsArg("asdeps", "asdep") {
  124. aurDeps.Add(name)
  125. } else {
  126. aurExp.Add(name)
  127. }
  128. case dep.CheckDep:
  129. fallthrough
  130. case dep.MakeDep:
  131. fallthrough
  132. case dep.Dep:
  133. aurDeps.Add(name)
  134. }
  135. case dep.Sync:
  136. switch info.Reason {
  137. case dep.Explicit:
  138. if cmdArgs.ExistsArg("asdeps", "asdep") {
  139. syncDeps.Add(name)
  140. } else {
  141. syncExp.Add(name)
  142. }
  143. case dep.CheckDep:
  144. fallthrough
  145. case dep.MakeDep:
  146. fallthrough
  147. case dep.Dep:
  148. syncDeps.Add(name)
  149. }
  150. }
  151. }
  152. fmt.Println(syncDeps, syncExp)
  153. errShow := installer.installSyncPackages(ctx, cmdArgs, syncDeps, syncExp)
  154. if errShow != nil {
  155. return ErrInstallRepoPkgs
  156. }
  157. errAur := installer.installAURPackages(ctx, cmdArgs, aurDeps, aurExp, nameToBaseMap, pkgBuildDirs, false)
  158. return errAur
  159. }
  160. func (installer *Installer) installAURPackages(ctx context.Context,
  161. cmdArgs *parser.Arguments,
  162. aurDepNames, aurExpNames mapset.Set[string],
  163. nameToBase, pkgBuildDirsByBase map[string]string,
  164. installIncompatible bool,
  165. ) error {
  166. deps, exp := make([]string, 0, aurDepNames.Cardinality()), make([]string, 0, aurExpNames.Cardinality())
  167. for _, name := range aurDepNames.Union(aurExpNames).ToSlice() {
  168. base := nameToBase[name]
  169. dir := pkgBuildDirsByBase[base]
  170. args := []string{"--nobuild", "-fC"}
  171. if installIncompatible {
  172. args = append(args, "--ignorearch")
  173. }
  174. // pkgver bump
  175. if err := config.Runtime.CmdBuilder.Show(
  176. config.Runtime.CmdBuilder.BuildMakepkgCmd(ctx, dir, args...)); err != nil {
  177. return errors.New(gotext.Get("error making: %s", base))
  178. }
  179. pkgdests, _, errList := parsePackageList(ctx, dir)
  180. if errList != nil {
  181. return errList
  182. }
  183. args = []string{"-cf", "--noconfirm", "--noextract", "--noprepare", "--holdver"}
  184. if installIncompatible {
  185. args = append(args, "--ignorearch")
  186. }
  187. if errMake := config.Runtime.CmdBuilder.Show(
  188. config.Runtime.CmdBuilder.BuildMakepkgCmd(ctx,
  189. dir, args...)); errMake != nil {
  190. return errors.New(gotext.Get("error making: %s", base))
  191. }
  192. names, err := installer.getNewTargets(pkgdests, name)
  193. if err != nil {
  194. return err
  195. }
  196. isDep := installer.isDep(cmdArgs, aurExpNames, name)
  197. if isDep {
  198. deps = append(deps, names...)
  199. } else {
  200. exp = append(exp, names...)
  201. }
  202. }
  203. if err := doInstall(ctx, cmdArgs, deps, exp); err != nil {
  204. return errors.New(fmt.Sprintf(gotext.Get("error installing:")+" %v %v", deps, exp))
  205. }
  206. return nil
  207. }
  208. func (*Installer) isDep(cmdArgs *parser.Arguments, aurExpNames mapset.Set[string], name string) bool {
  209. switch {
  210. case cmdArgs.ExistsArg("asdeps", "asdep"):
  211. return true
  212. case cmdArgs.ExistsArg("asexplicit", "asexp"):
  213. return false
  214. case aurExpNames.Contains(name):
  215. return false
  216. }
  217. return true
  218. }
  219. func (installer *Installer) getNewTargets(pkgdests map[string]string, name string,
  220. ) ([]string, error) {
  221. pkgdest, ok := pkgdests[name]
  222. names := make([]string, 0, 2)
  223. if !ok {
  224. return nil, errors.New(gotext.Get("could not find PKGDEST for: %s", name))
  225. }
  226. if _, errStat := os.Stat(pkgdest); os.IsNotExist(errStat) {
  227. return nil, errors.New(
  228. gotext.Get(
  229. "the PKGDEST for %s is listed by makepkg but does not exist: %s",
  230. name, pkgdest))
  231. }
  232. names = append(names, name)
  233. debugName := pkgdest + "-debug"
  234. pkgdestDebug, ok := pkgdests[debugName]
  235. if ok {
  236. if _, errStat := os.Stat(pkgdestDebug); errStat == nil {
  237. names = append(names, debugName)
  238. }
  239. }
  240. return names, nil
  241. }
  242. func (*Installer) installSyncPackages(ctx context.Context, cmdArgs *parser.Arguments,
  243. syncDeps, // repo targets that are deps
  244. syncExp mapset.Set[string], // repo targets that are exp
  245. ) error {
  246. repoTargets := syncDeps.Union(syncExp).ToSlice()
  247. if len(repoTargets) == 0 {
  248. return nil
  249. }
  250. arguments := cmdArgs.Copy()
  251. arguments.DelArg("asdeps", "asdep")
  252. arguments.DelArg("asexplicit", "asexp")
  253. arguments.DelArg("i", "install")
  254. arguments.Op = "S"
  255. arguments.ClearTargets()
  256. arguments.AddTarget(repoTargets...)
  257. errShow := config.Runtime.CmdBuilder.Show(config.Runtime.CmdBuilder.BuildPacmanCmd(ctx,
  258. arguments, config.Runtime.Mode, settings.NoConfirm))
  259. if errD := asdeps(ctx, cmdArgs, syncDeps.ToSlice()); errD != nil {
  260. return errD
  261. }
  262. if errE := asexp(ctx, cmdArgs, syncExp.ToSlice()); errE != nil {
  263. return errE
  264. }
  265. return errShow
  266. }